diff --git a/v1.46/404.html b/v1.46/404.html new file mode 100644 index 000000000..635e5f3bb --- /dev/null +++ b/v1.46/404.html @@ -0,0 +1,710 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Page not found | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + Page not found + + +

+ +

+ + + We can’t seem to find the page you’re looking for. + + +

+ + +

Error code: 404

+ +

+ +

+ Back to Docs +

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/Gemfile b/v1.46/Gemfile new file mode 100644 index 000000000..43490db0e --- /dev/null +++ b/v1.46/Gemfile @@ -0,0 +1,6 @@ +source 'https://rubygems.org' +gem 'github-pages', group: :jekyll_plugins +gem 'just-the-docs' +gem "webrick", "~> 1.8" +gem 'jekyll-sitemap' +gem 'jekyll-seo-tag' diff --git a/v1.46/Gemfile.lock b/v1.46/Gemfile.lock new file mode 100644 index 000000000..14e07e2fa --- /dev/null +++ b/v1.46/Gemfile.lock @@ -0,0 +1,270 @@ +GEM + remote: https://rubygems.org/ + specs: + activesupport (7.0.8) + concurrent-ruby (~> 1.0, >= 1.0.2) + i18n (>= 1.6, < 2) + minitest (>= 5.1) + tzinfo (~> 2.0) + addressable (2.8.4) + public_suffix (>= 2.0.2, < 6.0) + coffee-script (2.4.1) + coffee-script-source + execjs + coffee-script-source (1.11.1) + colorator (1.1.0) + commonmarker (0.23.10) + concurrent-ruby (1.2.2) + dnsruby (1.70.0) + simpleidn (~> 0.2.1) + em-websocket (0.5.3) + eventmachine (>= 0.12.9) + http_parser.rb (~> 0) + ethon (0.16.0) + ffi (>= 1.15.0) + eventmachine (1.2.7) + execjs (2.8.1) + faraday (2.7.5) + faraday-net_http (>= 2.0, < 3.1) + ruby2_keywords (>= 0.0.4) + faraday-net_http (3.0.2) + ffi (1.15.5) + forwardable-extended (2.6.0) + gemoji (3.0.1) + github-pages (228) + github-pages-health-check (= 1.17.9) + jekyll (= 3.9.3) + jekyll-avatar (= 0.7.0) + jekyll-coffeescript (= 1.1.1) + jekyll-commonmark-ghpages (= 0.4.0) + jekyll-default-layout (= 0.1.4) + jekyll-feed (= 0.15.1) + jekyll-gist (= 1.5.0) + jekyll-github-metadata (= 2.13.0) + jekyll-include-cache (= 0.2.1) + jekyll-mentions (= 1.6.0) + jekyll-optional-front-matter (= 0.3.2) + jekyll-paginate (= 1.1.0) + jekyll-readme-index (= 0.3.0) + jekyll-redirect-from (= 0.16.0) + jekyll-relative-links (= 0.6.1) + jekyll-remote-theme (= 0.4.3) + jekyll-sass-converter (= 1.5.2) + jekyll-seo-tag (= 2.8.0) + jekyll-sitemap (= 1.4.0) + jekyll-swiss (= 1.0.0) + jekyll-theme-architect (= 0.2.0) + jekyll-theme-cayman (= 0.2.0) + jekyll-theme-dinky (= 0.2.0) + jekyll-theme-hacker (= 0.2.0) + jekyll-theme-leap-day (= 0.2.0) + jekyll-theme-merlot (= 0.2.0) + jekyll-theme-midnight (= 0.2.0) + jekyll-theme-minimal (= 0.2.0) + jekyll-theme-modernist (= 0.2.0) + jekyll-theme-primer (= 0.6.0) + jekyll-theme-slate (= 0.2.0) + jekyll-theme-tactile (= 0.2.0) + jekyll-theme-time-machine (= 0.2.0) + jekyll-titles-from-headings (= 0.5.3) + jemoji (= 0.12.0) + kramdown (= 2.3.2) + kramdown-parser-gfm (= 1.1.0) + liquid (= 4.0.4) + mercenary (~> 0.3) + minima (= 2.5.1) + nokogiri (>= 1.13.6, < 2.0) + rouge (= 3.26.0) + terminal-table (~> 1.4) + github-pages-health-check (1.17.9) + addressable (~> 2.3) + dnsruby (~> 1.60) + octokit (~> 4.0) + public_suffix (>= 3.0, < 5.0) + typhoeus (~> 1.3) + html-pipeline (2.14.3) + activesupport (>= 2) + nokogiri (>= 1.4) + http_parser.rb (0.8.0) + i18n (1.14.1) + concurrent-ruby (~> 1.0) + jekyll (3.9.3) + addressable (~> 2.4) + colorator (~> 1.0) + em-websocket (~> 0.5) + i18n (>= 0.7, < 2) + jekyll-sass-converter (~> 1.0) + jekyll-watch (~> 2.0) + kramdown (>= 1.17, < 3) + liquid (~> 4.0) + mercenary (~> 0.3.3) + pathutil (~> 0.9) + rouge (>= 1.7, < 4) + safe_yaml (~> 1.0) + jekyll-avatar (0.7.0) + jekyll (>= 3.0, < 5.0) + jekyll-coffeescript (1.1.1) + coffee-script (~> 2.2) + coffee-script-source (~> 1.11.1) + jekyll-commonmark (1.4.0) + commonmarker (~> 0.22) + jekyll-commonmark-ghpages (0.4.0) + commonmarker (~> 0.23.7) + jekyll (~> 3.9.0) + jekyll-commonmark (~> 1.4.0) + rouge (>= 2.0, < 5.0) + jekyll-default-layout (0.1.4) + jekyll (~> 3.0) + jekyll-feed (0.15.1) + jekyll (>= 3.7, < 5.0) + jekyll-gist (1.5.0) + octokit (~> 4.2) + jekyll-github-metadata (2.13.0) + jekyll (>= 3.4, < 5.0) + octokit (~> 4.0, != 4.4.0) + jekyll-include-cache (0.2.1) + jekyll (>= 3.7, < 5.0) + jekyll-mentions (1.6.0) + html-pipeline (~> 2.3) + jekyll (>= 3.7, < 5.0) + jekyll-optional-front-matter (0.3.2) + jekyll (>= 3.0, < 5.0) + jekyll-paginate (1.1.0) + jekyll-readme-index (0.3.0) + jekyll (>= 3.0, < 5.0) + jekyll-redirect-from (0.16.0) + jekyll (>= 3.3, < 5.0) + jekyll-relative-links (0.6.1) + jekyll (>= 3.3, < 5.0) + jekyll-remote-theme (0.4.3) + addressable (~> 2.0) + jekyll (>= 3.5, < 5.0) + jekyll-sass-converter (>= 1.0, <= 3.0.0, != 2.0.0) + rubyzip (>= 1.3.0, < 3.0) + jekyll-sass-converter (1.5.2) + sass (~> 3.4) + jekyll-seo-tag (2.8.0) + jekyll (>= 3.8, < 5.0) + jekyll-sitemap (1.4.0) + jekyll (>= 3.7, < 5.0) + jekyll-swiss (1.0.0) + jekyll-theme-architect (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-cayman (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-dinky (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-hacker (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-leap-day (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-merlot (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-midnight (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-minimal (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-modernist (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-primer (0.6.0) + jekyll (> 3.5, < 5.0) + jekyll-github-metadata (~> 2.9) + jekyll-seo-tag (~> 2.0) + jekyll-theme-slate (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-tactile (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-theme-time-machine (0.2.0) + jekyll (> 3.5, < 5.0) + jekyll-seo-tag (~> 2.0) + jekyll-titles-from-headings (0.5.3) + jekyll (>= 3.3, < 5.0) + jekyll-watch (2.2.1) + listen (~> 3.0) + jemoji (0.12.0) + gemoji (~> 3.0) + html-pipeline (~> 2.2) + jekyll (>= 3.0, < 5.0) + just-the-docs (0.5.1) + jekyll (>= 3.8.5) + jekyll-seo-tag (>= 2.0) + rake (>= 12.3.1) + kramdown (2.3.2) + rexml + kramdown-parser-gfm (1.1.0) + kramdown (~> 2.0) + liquid (4.0.4) + listen (3.8.0) + rb-fsevent (~> 0.10, >= 0.10.3) + rb-inotify (~> 0.9, >= 0.9.10) + mercenary (0.3.6) + mini_portile2 (2.8.5) + minima (2.5.1) + jekyll (>= 3.5, < 5.0) + jekyll-feed (~> 0.9) + jekyll-seo-tag (~> 2.1) + minitest (5.20.0) + nokogiri (1.16.2) + mini_portile2 (~> 2.8.2) + racc (~> 1.4) + octokit (4.25.1) + faraday (>= 1, < 3) + sawyer (~> 0.9) + pathutil (0.16.2) + forwardable-extended (~> 2.6) + public_suffix (4.0.7) + racc (1.7.3) + rake (13.0.6) + rb-fsevent (0.11.2) + rb-inotify (0.10.1) + ffi (~> 1.0) + rexml (3.2.5) + rouge (3.26.0) + ruby2_keywords (0.0.5) + rubyzip (2.3.2) + safe_yaml (1.0.5) + sass (3.7.4) + sass-listen (~> 4.0.0) + sass-listen (4.0.0) + rb-fsevent (~> 0.9, >= 0.9.4) + rb-inotify (~> 0.9, >= 0.9.7) + sawyer (0.9.2) + addressable (>= 2.3.5) + faraday (>= 0.17.3, < 3) + simpleidn (0.2.1) + unf (~> 0.1.4) + terminal-table (1.8.0) + unicode-display_width (~> 1.1, >= 1.1.1) + typhoeus (1.4.0) + ethon (>= 0.9.0) + tzinfo (2.0.6) + concurrent-ruby (~> 1.0) + unf (0.1.4) + unf_ext + unf_ext (0.0.8.2) + unicode-display_width (1.8.0) + webrick (1.8.2) + +PLATFORMS + ruby + +DEPENDENCIES + github-pages + jekyll-seo-tag + jekyll-sitemap + just-the-docs + webrick (~> 1.8) + +BUNDLED WITH + 2.1.4 diff --git a/v1.46/LICENSE b/v1.46/LICENSE new file mode 100644 index 000000000..376825af2 --- /dev/null +++ b/v1.46/LICENSE @@ -0,0 +1,13 @@ +Copyright 2020 Treeverse Labs LTD. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/v1.46/README.html b/v1.46/README.html new file mode 100644 index 000000000..1e9fb3d83 --- /dev/null +++ b/v1.46/README.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/architecture/index.html b/v1.46/architecture/index.html new file mode 100644 index 000000000..35603eab3 --- /dev/null +++ b/v1.46/architecture/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/architecture/overview.html b/v1.46/architecture/overview.html new file mode 100644 index 000000000..35603eab3 --- /dev/null +++ b/v1.46/architecture/overview.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/architecture/sizing-guide.html b/v1.46/architecture/sizing-guide.html new file mode 100644 index 000000000..252d1aa93 --- /dev/null +++ b/v1.46/architecture/sizing-guide.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/assets/by-treeverse.png b/v1.46/assets/by-treeverse.png new file mode 100644 index 000000000..8885fa8ff Binary files /dev/null and b/v1.46/assets/by-treeverse.png differ diff --git a/v1.46/assets/css/cookieconsent.css b/v1.46/assets/css/cookieconsent.css new file mode 100644 index 000000000..333f3c728 --- /dev/null +++ b/v1.46/assets/css/cookieconsent.css @@ -0,0 +1,13 @@ +#cc-main { + --cc-btn-primary-bg: #279890; + --cc-btn-primary-border-color: #279890; + --cc-btn-primary-hover-bg: #fff; + --cc-btn-primary-hover-color: #279890; + --cc-btn-primary-hover-border-color: #279890; + --cc-font-family: "GalanoGrotesque-Medium"; + --cc-toggle-on-bg: var(--cc-btn-primary-bg); +} + +h2.cm__title { + margin-top: 0 !important; +} diff --git a/v1.46/assets/css/just-the-docs-dark.css b/v1.46/assets/css/just-the-docs-dark.css new file mode 100644 index 000000000..52ba65f5f --- /dev/null +++ b/v1.46/assets/css/just-the-docs-dark.css @@ -0,0 +1,2690 @@ +@charset "UTF-8"; +@import url("pygments/igorpro.css"); +.highlight, pre.highlight { background: #f9f9f9; color: #383942; } + +.highlight pre { background: #f9f9f9; } + +.highlight .hll { background: #f9f9f9; } + +.highlight .c { color: #9fa0a6; font-style: italic; } + +.highlight .err { color: #fff; background-color: #e05151; } + +.highlight .k { color: #a625a4; } + +.highlight .l { color: #50a04f; } + +.highlight .n { color: #383942; } + +.highlight .o { color: #383942; } + +.highlight .p { color: #383942; } + +.highlight .cm { color: #9fa0a6; font-style: italic; } + +.highlight .cp { color: #9fa0a6; font-style: italic; } + +.highlight .c1 { color: #9fa0a6; font-style: italic; } + +.highlight .cs { color: #9fa0a6; font-style: italic; } + +.highlight .ge { font-style: italic; } + +.highlight .gs { font-weight: 700; } + +.highlight .kc { color: #a625a4; } + +.highlight .kd { color: #a625a4; } + +.highlight .kn { color: #a625a4; } + +.highlight .kp { color: #a625a4; } + +.highlight .kr { color: #a625a4; } + +.highlight .kt { color: #a625a4; } + +.highlight .ld { color: #50a04f; } + +.highlight .m { color: #b66a00; } + +.highlight .s { color: #50a04f; } + +.highlight .na { color: #b66a00; } + +.highlight .nb { color: #ca7601; } + +.highlight .nc { color: #ca7601; } + +.highlight .no { color: #ca7601; } + +.highlight .nd { color: #ca7601; } + +.highlight .ni { color: #ca7601; } + +.highlight .ne { color: #ca7601; } + +.highlight .nf { color: #383942; } + +.highlight .nl { color: #ca7601; } + +.highlight .nn { color: #383942; } + +.highlight .nx { color: #383942; } + +.highlight .py { color: #ca7601; } + +.highlight .nt { color: #e35549; } + +.highlight .nv { color: #ca7601; } + +.highlight .ow { font-weight: 700; } + +.highlight .w { color: #f8f8f2; } + +.highlight .mf { color: #b66a00; } + +.highlight .mh { color: #b66a00; } + +.highlight .mi { color: #b66a00; } + +.highlight .mo { color: #b66a00; } + +.highlight .sb { color: #50a04f; } + +.highlight .sc { color: #50a04f; } + +.highlight .sd { color: #50a04f; } + +.highlight .s2 { color: #50a04f; } + +.highlight .se { color: #50a04f; } + +.highlight .sh { color: #50a04f; } + +.highlight .si { color: #50a04f; } + +.highlight .sx { color: #50a04f; } + +.highlight .sr { color: #0083bb; } + +.highlight .s1 { color: #50a04f; } + +.highlight .ss { color: #0083bb; } + +.highlight .bp { color: #ca7601; } + +.highlight .vc { color: #ca7601; } + +.highlight .vg { color: #ca7601; } + +.highlight .vi { color: #e35549; } + +.highlight .il { color: #b66a00; } + +.highlight .gu { color: #75715e; } + +.highlight .gd { color: #e05151; } + +.highlight .gi { color: #43d089; } + +.highlight .language-json .w + .s2 { color: #e35549; } + +.highlight .language-json .kc { color: #0083bb; } + +.highlight, pre.highlight { background: #31343f; color: #dee2f7; } + +.highlight pre { background: #31343f; } + +.highlight .hll { background: #31343f; } + +.highlight .c { color: #63677e; font-style: italic; } + +.highlight .err { color: #960050; background-color: #1e0010; } + +.highlight .k { color: #e19ef5; } + +.highlight .l { color: #a3eea0; } + +.highlight .n { color: #dee2f7; } + +.highlight .o { color: #dee2f7; } + +.highlight .p { color: #dee2f7; } + +.highlight .cm { color: #63677e; font-style: italic; } + +.highlight .cp { color: #63677e; font-style: italic; } + +.highlight .c1 { color: #63677e; font-style: italic; } + +.highlight .cs { color: #63677e; font-style: italic; } + +.highlight .ge { font-style: italic; } + +.highlight .gs { font-weight: 700; } + +.highlight .kc { color: #e19ef5; } + +.highlight .kd { color: #e19ef5; } + +.highlight .kn { color: #e19ef5; } + +.highlight .kp { color: #e19ef5; } + +.highlight .kr { color: #e19ef5; } + +.highlight .kt { color: #e19ef5; } + +.highlight .ld { color: #a3eea0; } + +.highlight .m { color: #eddc96; } + +.highlight .s { color: #a3eea0; } + +.highlight .na { color: #eddc96; } + +.highlight .nb { color: #fdce68; } + +.highlight .nc { color: #fdce68; } + +.highlight .no { color: #fdce68; } + +.highlight .nd { color: #fdce68; } + +.highlight .ni { color: #fdce68; } + +.highlight .ne { color: #fdce68; } + +.highlight .nf { color: #dee2f7; } + +.highlight .nl { color: #fdce68; } + +.highlight .nn { color: #dee2f7; } + +.highlight .nx { color: #dee2f7; } + +.highlight .py { color: #fdce68; } + +.highlight .nt { color: #f9867b; } + +.highlight .nv { color: #fdce68; } + +.highlight .ow { font-weight: 700; } + +.highlight .w { color: #f8f8f2; } + +.highlight .mf { color: #eddc96; } + +.highlight .mh { color: #eddc96; } + +.highlight .mi { color: #eddc96; } + +.highlight .mo { color: #eddc96; } + +.highlight .sb { color: #a3eea0; } + +.highlight .sc { color: #a3eea0; } + +.highlight .sd { color: #a3eea0; } + +.highlight .s2 { color: #a3eea0; } + +.highlight .se { color: #a3eea0; } + +.highlight .sh { color: #a3eea0; } + +.highlight .si { color: #a3eea0; } + +.highlight .sx { color: #a3eea0; } + +.highlight .sr { color: #7be2f9; } + +.highlight .s1 { color: #a3eea0; } + +.highlight .ss { color: #7be2f9; } + +.highlight .bp { color: #fdce68; } + +.highlight .vc { color: #fdce68; } + +.highlight .vg { color: #fdce68; } + +.highlight .vi { color: #f9867b; } + +.highlight .il { color: #eddc96; } + +.highlight .gu { color: #75715e; } + +.highlight .gd { color: #f92672; } + +.highlight .gi { color: #a6e22e; } + +/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ +/* Document ========================================================================== */ +/** 1. Correct the line height in all browsers. 2. Prevent adjustments of font size after orientation changes in iOS. */ +html { line-height: 1.15; /* 1 */ -webkit-text-size-adjust: 100%; /* 2 */ } + +/* Sections ========================================================================== */ +/** Remove the margin in all browsers. */ +body { margin: 0; } + +/** Render the `main` element consistently in IE. */ +main { display: block; } + +/** Correct the font size and margin on `h1` elements within `section` and `article` contexts in Chrome, Firefox, and Safari. */ +h1 { font-size: 2em; margin: 0.67em 0; } + +/* Grouping content ========================================================================== */ +/** 1. Add the correct box sizing in Firefox. 2. Show the overflow in Edge and IE. */ +hr { box-sizing: content-box; /* 1 */ height: 0; /* 1 */ overflow: visible; /* 2 */ } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +pre { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/* Text-level semantics ========================================================================== */ +/** Remove the gray background on active links in IE 10. */ +a { background-color: transparent; } + +/** 1. Remove the bottom border in Chrome 57- 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. */ +abbr[title] { border-bottom: none; /* 1 */ text-decoration: underline; /* 2 */ text-decoration: underline dotted; /* 2 */ } + +/** Add the correct font weight in Chrome, Edge, and Safari. */ +b, strong { font-weight: bolder; } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +code, kbd, samp { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/** Add the correct font size in all browsers. */ +small { font-size: 80%; } + +/** Prevent `sub` and `sup` elements from affecting the line height in all browsers. */ +sub, sup { font-size: 75%; line-height: 0; position: relative; vertical-align: baseline; } + +sub { bottom: -0.25em; } + +sup { top: -0.5em; } + +/* Embedded content ========================================================================== */ +/** Remove the border on images inside links in IE 10. */ +img { border-style: none; } + +/* Forms ========================================================================== */ +/** 1. Change the font styles in all browsers. 2. Remove the margin in Firefox and Safari. */ +button, input, optgroup, select, textarea { font-family: inherit; /* 1 */ font-size: 100%; /* 1 */ line-height: 1.15; /* 1 */ margin: 0; /* 2 */ } + +/** Show the overflow in IE. 1. Show the overflow in Edge. */ +button, input { /* 1 */ overflow: visible; } + +/** Remove the inheritance of text transform in Edge, Firefox, and IE. 1. Remove the inheritance of text transform in Firefox. */ +button, select { /* 1 */ text-transform: none; } + +/** Correct the inability to style clickable types in iOS and Safari. */ +button, [type="button"], [type="reset"], [type="submit"] { -webkit-appearance: button; } + +/** Remove the inner border and padding in Firefox. */ +button::-moz-focus-inner, [type="button"]::-moz-focus-inner, [type="reset"]::-moz-focus-inner, [type="submit"]::-moz-focus-inner { border-style: none; padding: 0; } + +/** Restore the focus styles unset by the previous rule. */ +button:-moz-focusring, [type="button"]:-moz-focusring, [type="reset"]:-moz-focusring, [type="submit"]:-moz-focusring { outline: 1px dotted ButtonText; } + +/** Correct the padding in Firefox. */ +fieldset { padding: 0.35em 0.75em 0.625em; } + +/** 1. Correct the text wrapping in Edge and IE. 2. Correct the color inheritance from `fieldset` elements in IE. 3. Remove the padding so developers are not caught out when they zero out `fieldset` elements in all browsers. */ +legend { box-sizing: border-box; /* 1 */ color: inherit; /* 2 */ display: table; /* 1 */ max-width: 100%; /* 1 */ padding: 0; /* 3 */ white-space: normal; /* 1 */ } + +/** Add the correct vertical alignment in Chrome, Firefox, and Opera. */ +progress { vertical-align: baseline; } + +/** Remove the default vertical scrollbar in IE 10+. */ +textarea { overflow: auto; } + +/** 1. Add the correct box sizing in IE 10. 2. Remove the padding in IE 10. */ +[type="checkbox"], [type="radio"] { box-sizing: border-box; /* 1 */ padding: 0; /* 2 */ } + +/** Correct the cursor style of increment and decrement buttons in Chrome. */ +[type="number"]::-webkit-inner-spin-button, [type="number"]::-webkit-outer-spin-button { height: auto; } + +/** 1. Correct the odd appearance in Chrome and Safari. 2. Correct the outline style in Safari. */ +[type="search"] { -webkit-appearance: textfield; /* 1 */ outline-offset: -2px; /* 2 */ } + +/** Remove the inner padding in Chrome and Safari on macOS. */ +[type="search"]::-webkit-search-decoration { -webkit-appearance: none; } + +/** 1. Correct the inability to style clickable types in iOS and Safari. 2. Change font properties to `inherit` in Safari. */ +::-webkit-file-upload-button { -webkit-appearance: button; /* 1 */ font: inherit; /* 2 */ } + +/* Interactive ========================================================================== */ +/* Add the correct display in Edge, IE 10+, and Firefox. */ +details { display: block; } + +/* Add the correct display in all browsers. */ +summary { display: list-item; } + +/* Misc ========================================================================== */ +/** Add the correct display in IE 10+. */ +template { display: none; } + +/** Add the correct display in IE 10. */ +[hidden] { display: none; } + +:root { color-scheme: dark; } + +* { box-sizing: border-box; } + +html { font-size: 0.875rem !important; scroll-behavior: smooth; } +@media (min-width: 31.25rem) { html { font-size: 1rem !important; } } + +body { font-family: system-ui, -apple-system, blinkmacsystemfont, "Segoe UI", roboto, "Helvetica Neue", arial, sans-serif; font-size: inherit; line-height: 1.4; color: #e6e1e8; background-color: #27262b; overflow-wrap: break-word; } + +ol, ul, dl, pre, address, blockquote, table, div, hr, form, fieldset, noscript .table-wrapper { margin-top: 0; } + +h1, h2, h3, h4, h5, h6, #toctitle { margin-top: 0; margin-bottom: 1em; font-weight: 500; line-height: 1.25; color: #f5f6fa; } + +p { margin-top: 1em; margin-bottom: 1em; } + +a { color: #2c84fa; text-decoration: none; } + +a:not([class]) { text-decoration: underline; text-decoration-color: #44434d; text-underline-offset: 2px; } +a:not([class]):hover { text-decoration-color: rgba(44, 132, 250, 0.45); } + +code { font-family: "SFMono-Regular", menlo, consolas, monospace; font-size: 0.75em; line-height: 1.4; } + +figure, pre { margin: 0; } + +li { margin: 0.25em 0; } + +img { max-width: 100%; height: auto; } + +hr { height: 1px; padding: 0; margin: 2rem 0; background-color: #44434d; border: 0; } + +blockquote { margin: 10px 0; margin-block-start: 0; margin-inline-start: 0; padding-left: 1rem; border-left: 3px solid #44434d; } + +.side-bar { z-index: 0; display: flex; flex-wrap: wrap; background-color: #27262b; } +@media (min-width: 50rem) { .side-bar { flex-flow: column nowrap; position: fixed; width: 15.5rem; height: 100%; border-right: 1px solid #44434d; align-items: flex-end; } } +@media (min-width: 66.5rem) { .side-bar { width: calc((100% - 66.5rem) / 2 + 16.5rem); min-width: 16.5rem; } } + +@media (min-width: 50rem) { .main { position: relative; max-width: 50rem; margin-left: 15.5rem; } } +@media (min-width: 66.5rem) { .main { margin-left: Max(16.5rem, calc((100% - 66.5rem) / 2 + 16.5rem)); } } + +.main-content-wrap { padding-right: 1rem; padding-left: 1rem; padding-top: 1rem; padding-bottom: 1rem; } +@media (min-width: 50rem) { .main-content-wrap { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 50rem) { .main-content-wrap { padding-top: 2rem; padding-bottom: 2rem; } } + +.main-header { z-index: 0; display: none; background-color: #27262b; } +@media (min-width: 50rem) { .main-header { display: flex; justify-content: space-between; height: 3.75rem; background-color: #27262b; border-bottom: 1px solid #44434d; } } +.main-header.nav-open { display: block; } +@media (min-width: 50rem) { .main-header.nav-open { display: flex; } } + +.site-nav, .site-header, .site-footer { width: 100%; } +@media (min-width: 66.5rem) { .site-nav, .site-header, .site-footer { width: 16.5rem; } } + +.site-nav { display: none; } +.site-nav.nav-open { display: block; } +@media (min-width: 50rem) { .site-nav { display: block; padding-top: 3rem; padding-bottom: 1rem; overflow-y: auto; flex: 1 1 auto; } } + +.site-header { display: flex; min-height: 3.75rem; align-items: center; } +@media (min-width: 50rem) { .site-header { height: 3.75rem; max-height: 3.75rem; border-bottom: 1px solid #44434d; } } + +.site-title { padding-right: 1rem; padding-left: 1rem; flex-grow: 1; display: flex; height: 100%; align-items: center; padding-top: 0.75rem; padding-bottom: 0.75rem; color: #f5f6fa; font-size: 1.125rem !important; } +@media (min-width: 50rem) { .site-title { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-title { font-size: 1.5rem !important; line-height: 1.25; } } +@media (min-width: 50rem) { .site-title { padding-top: 0.5rem; padding-bottom: 0.5rem; } } + +.site-logo { width: 100%; height: 100%; background-image: url("/v1.46/assets/logo.svg"); background-repeat: no-repeat; background-position: left center; background-size: contain; } + +.site-button { display: flex; height: 100%; padding: 1rem; align-items: center; } + +@media (min-width: 50rem) { .site-header .site-button { display: none; } } +.site-title:hover { background-image: linear-gradient(-90deg, #201f23 0%, rgba(32, 31, 35, 0.8) 80%, rgba(32, 31, 35, 0) 100%); } + +.site-button:hover { background-image: linear-gradient(-90deg, #201f23 0%, rgba(32, 31, 35, 0.8) 100%); } + +body { position: relative; padding-bottom: 4rem; overflow-y: scroll; } +@media (min-width: 50rem) { body { position: static; padding-bottom: 0; } } + +.site-footer { padding-right: 1rem; padding-left: 1rem; position: absolute; bottom: 0; left: 0; padding-top: 1rem; padding-bottom: 1rem; color: #959396; font-size: 0.6875rem !important; } +@media (min-width: 50rem) { .site-footer { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-footer { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .site-footer { position: static; justify-self: end; } } + +.icon { width: 1.5rem; height: 1.5rem; color: #2c84fa; } + +.main-content { line-height: 1.6; } +.main-content ol, .main-content ul, .main-content dl, .main-content pre, .main-content address, .main-content blockquote, .main-content .table-wrapper { margin-top: 0.5em; } +.main-content a { overflow: hidden; text-overflow: ellipsis; } +.main-content ul, .main-content ol { padding-left: 1.5em; } +.main-content li .highlight { margin-top: 0.25rem; } +.main-content ol { list-style-type: none; counter-reset: step-counter; } +.main-content ol > li { position: relative; } +.main-content ol > li::before { position: absolute; top: 0.2em; left: -1.6em; color: #959396; content: counter(step-counter); counter-increment: step-counter; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .main-content ol > li::before { font-size: 0.875rem !important; } } +@media (min-width: 31.25rem) { .main-content ol > li::before { top: 0.11em; } } +.main-content ol > li ol { counter-reset: sub-counter; } +.main-content ol > li ol > li::before { content: counter(sub-counter,lower-alpha); counter-increment: sub-counter; } +.main-content ul { list-style: none; } +.main-content ul > li::before { position: absolute; margin-left: -1.4em; color: #959396; content: "•"; } +.main-content .task-list-item::before { content: ""; } +.main-content .task-list-item-checkbox { margin-right: 0.6em; margin-left: -1.4em; } +.main-content hr + * { margin-top: 0; } +.main-content h1:first-of-type { margin-top: 0.5em; } +.main-content dl { display: grid; grid-template: auto / 10em 1fr; } +.main-content dt, .main-content dd { margin: 0.25em 0; } +.main-content dt { grid-column: 1; font-weight: 500; text-align: right; } +.main-content dt::after { content: ":"; } +.main-content dd { grid-column: 2; margin-bottom: 0; margin-left: 1em; } +.main-content dd blockquote:first-child, .main-content dd div:first-child, .main-content dd dl:first-child, .main-content dd dt:first-child, .main-content dd h1:first-child, .main-content dd h2:first-child, .main-content dd h3:first-child, .main-content dd h4:first-child, .main-content dd h5:first-child, .main-content dd h6:first-child, .main-content dd li:first-child, .main-content dd ol:first-child, .main-content dd p:first-child, .main-content dd pre:first-child, .main-content dd table:first-child, .main-content dd ul:first-child, .main-content dd .table-wrapper:first-child { margin-top: 0; } +.main-content dd dl:first-child dt:first-child, .main-content dd dl:first-child dd:nth-child(2), .main-content ol dl:first-child dt:first-child, .main-content ol dl:first-child dd:nth-child(2), .main-content ul dl:first-child dt:first-child, .main-content ul dl:first-child dd:nth-child(2) { margin-top: 0; } +.main-content .anchor-heading { position: absolute; right: -1rem; width: 1.5rem; height: 100%; padding-right: 0.25rem; padding-left: 0.25rem; overflow: visible; } +@media (min-width: 50rem) { .main-content .anchor-heading { right: auto; left: -1.5rem; } } +.main-content .anchor-heading svg { display: inline-block; width: 100%; height: 100%; color: #2c84fa; visibility: hidden; } +.main-content .anchor-heading:hover svg, .main-content .anchor-heading:focus svg, .main-content h1:hover > .anchor-heading svg, .main-content h2:hover > .anchor-heading svg, .main-content h3:hover > .anchor-heading svg, .main-content h4:hover > .anchor-heading svg, .main-content h5:hover > .anchor-heading svg, .main-content h6:hover > .anchor-heading svg { visibility: visible; } +.main-content summary { cursor: pointer; } +.main-content h1, .main-content h2, .main-content h3, .main-content h4, .main-content h5, .main-content h6, .main-content #toctitle { position: relative; margin-top: 1.5em; margin-bottom: 0.25em; } +.main-content h1 + table, .main-content h1 + .table-wrapper, .main-content h1 + .code-example, .main-content h1 + .highlighter-rouge, .main-content h1 + .sectionbody .listingblock, .main-content h2 + table, .main-content h2 + .table-wrapper, .main-content h2 + .code-example, .main-content h2 + .highlighter-rouge, .main-content h2 + .sectionbody .listingblock, .main-content h3 + table, .main-content h3 + .table-wrapper, .main-content h3 + .code-example, .main-content h3 + .highlighter-rouge, .main-content h3 + .sectionbody .listingblock, .main-content h4 + table, .main-content h4 + .table-wrapper, .main-content h4 + .code-example, .main-content h4 + .highlighter-rouge, .main-content h4 + .sectionbody .listingblock, .main-content h5 + table, .main-content h5 + .table-wrapper, .main-content h5 + .code-example, .main-content h5 + .highlighter-rouge, .main-content h5 + .sectionbody .listingblock, .main-content h6 + table, .main-content h6 + .table-wrapper, .main-content h6 + .code-example, .main-content h6 + .highlighter-rouge, .main-content h6 + .sectionbody .listingblock, .main-content #toctitle + table, .main-content #toctitle + .table-wrapper, .main-content #toctitle + .code-example, .main-content #toctitle + .highlighter-rouge, .main-content #toctitle + .sectionbody .listingblock { margin-top: 1em; } +.main-content h1 + p:not(.label), .main-content h2 + p:not(.label), .main-content h3 + p:not(.label), .main-content h4 + p:not(.label), .main-content h5 + p:not(.label), .main-content h6 + p:not(.label), .main-content #toctitle + p:not(.label) { margin-top: 0; } +.main-content > h1:first-child, .main-content > h2:first-child, .main-content > h3:first-child, .main-content > h4:first-child, .main-content > h5:first-child, .main-content > h6:first-child, .main-content > .sect1:first-child > h2, .main-content > .sect2:first-child > h3, .main-content > .sect3:first-child > h4, .main-content > .sect4:first-child > h5, .main-content > .sect5:first-child > h6 { margin-top: 0.5rem; } + +.nav-list { padding: 0; margin-top: 0; margin-bottom: 0; list-style: none; } +.nav-list .nav-list-item { font-size: 0.875rem !important; position: relative; margin: 0; } +@media (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 1rem !important; } } +@media (min-width: 50rem) { .nav-list .nav-list-item { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 0.875rem !important; } } + +.nav-list .nav-list-item .nav-list-link { display: block; min-height: 3rem; padding-top: 0.25rem; padding-bottom: 0.25rem; line-height: 2.5rem; padding-right: 3rem; padding-left: 1rem; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-link { min-height: 2rem; line-height: 1.5rem; padding-right: 2rem; padding-left: 2rem; } } +.nav-list .nav-list-item .nav-list-link.external > svg { width: 1rem; height: 1rem; vertical-align: text-bottom; } +.nav-list .nav-list-item .nav-list-link.active { font-weight: 600; text-decoration: none; } +.nav-list .nav-list-item .nav-list-link:hover, .nav-list .nav-list-item .nav-list-link.active { background-image: linear-gradient(-90deg, #201f23 0%, rgba(32, 31, 35, 0.8) 80%, rgba(32, 31, 35, 0) 100%); } +.nav-list .nav-list-item .nav-list-expander { position: absolute; right: 0; width: 3rem; height: 3rem; padding: 0.75rem; color: #2c84fa; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-expander { width: 2rem; height: 2rem; padding: 0.5rem; } } +.nav-list .nav-list-item .nav-list-expander:hover { background-image: linear-gradient(-90deg, #201f23 0%, rgba(32, 31, 35, 0.8) 100%); } +.nav-list .nav-list-item .nav-list-expander svg { transform: rotate(90deg); } +.nav-list .nav-list-item > .nav-list { display: none; padding-left: 0.75rem; list-style: none; } +.nav-list .nav-list-item > .nav-list .nav-list-item { position: relative; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-link { color: #959396; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-expander { color: #959396; } +.nav-list .nav-list-item.active > .nav-list-expander svg { transform: rotate(-90deg); } +.nav-list .nav-list-item.active > .nav-list { display: block; } + +.nav-category { padding: 0.5rem 1rem; font-weight: 600; text-align: start; text-transform: uppercase; border-bottom: 1px solid #44434d; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .nav-category { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .nav-category { padding: 0.5rem 2rem; margin-top: 1rem; text-align: start; } + .nav-category:first-child { margin-top: 0; } } + +.nav-list.nav-category-list > .nav-list-item { margin: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list { padding: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-link { color: #2c84fa; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-expander { color: #2c84fa; } + +.aux-nav { height: 100%; overflow-x: auto; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .aux-nav { font-size: 0.75rem !important; } } +.aux-nav .aux-nav-list { display: flex; height: 100%; padding: 0; margin: 0; list-style: none; } +.aux-nav .aux-nav-list-item { display: inline-block; height: 100%; padding: 0; margin: 0; } +@media (min-width: 50rem) { .aux-nav { padding-right: 1rem; } } + +@media (min-width: 50rem) { .breadcrumb-nav { margin-top: -1rem; } } + +.breadcrumb-nav-list { padding-left: 0; margin-bottom: 0.75rem; list-style: none; } + +.breadcrumb-nav-list-item { display: table-cell; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .breadcrumb-nav-list-item { font-size: 0.75rem !important; } } +.breadcrumb-nav-list-item::before { display: none; } +.breadcrumb-nav-list-item::after { display: inline-block; margin-right: 0.5rem; margin-left: 0.5rem; color: #959396; content: "/"; } +.breadcrumb-nav-list-item:last-child::after { content: ""; } + +h1, .text-alpha { font-size: 2rem !important; line-height: 1.25; font-weight: 300; } +@media (min-width: 31.25rem) { h1, .text-alpha { font-size: 2.25rem !important; } } + +h2, .text-beta, #toctitle { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { h2, .text-beta, #toctitle { font-size: 1.5rem !important; line-height: 1.25; } } + +h3, .text-gamma { font-size: 1rem !important; } +@media (min-width: 31.25rem) { h3, .text-gamma { font-size: 1.125rem !important; } } + +h4, .text-delta { font-size: 0.6875rem !important; font-weight: 400; text-transform: uppercase; letter-spacing: 0.1em; } +@media (min-width: 31.25rem) { h4, .text-delta { font-size: 0.75rem !important; } } + +h4 code { text-transform: none; } + +h5, .text-epsilon { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { h5, .text-epsilon { font-size: 0.875rem !important; } } + +h6, .text-zeta { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { h6, .text-zeta { font-size: 0.75rem !important; } } + +.text-small { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .text-small { font-size: 0.75rem !important; } } + +.text-mono { font-family: "SFMono-Regular", menlo, consolas, monospace !important; } + +.text-left { text-align: left !important; } + +.text-center { text-align: center !important; } + +.text-right { text-align: right !important; } + +.label:not(g), .label-blue:not(g) { display: inline-block; padding: 0.16em 0.56em; margin-right: 0.5rem; margin-left: 0.5rem; color: #fff; text-transform: uppercase; vertical-align: middle; background-color: #2869e6; font-size: 0.6875rem !important; border-radius: 12px; } +@media (min-width: 31.25rem) { .label:not(g), .label-blue:not(g) { font-size: 0.75rem !important; } } + +.label-green:not(g) { background-color: #009c7b; } + +.label-purple:not(g) { background-color: #5e41d0; } + +.label-red:not(g) { background-color: #e94c4c; } + +.label-yellow:not(g) { color: #44434d; background-color: #f7d12e; } + +.btn { display: inline-block; box-sizing: border-box; padding: 0.3em 1em; margin: 0; font-family: inherit; font-size: inherit; font-weight: 500; line-height: 1.5; color: #2c84fa; text-decoration: none; vertical-align: baseline; cursor: pointer; background-color: #302d36; border-width: 0; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); appearance: none; } +.btn:focus { text-decoration: none; outline: none; box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:focus:hover, .btn.selected:focus { box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:hover, .btn.zeroclipboard-is-hover { color: #227efa; } +.btn:hover, .btn:active, .btn.zeroclipboard-is-hover, .btn.zeroclipboard-is-active { text-decoration: none; background-color: #2e2b33; } +.btn:active, .btn.selected, .btn.zeroclipboard-is-active { background-color: #29262e; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn.selected:hover { background-color: #cfcfcf; } +.btn:disabled, .btn:disabled:hover, .btn.disabled, .btn.disabled:hover { color: rgba(102, 102, 102, 0.5); cursor: default; background-color: rgba(229, 229, 229, 0.5); background-image: none; box-shadow: none; } + +.btn-outline { color: #2c84fa; background: transparent; box-shadow: inset 0 0 0 2px #e6e1e8; } +.btn-outline:hover, .btn-outline:active, .btn-outline.zeroclipboard-is-hover, .btn-outline.zeroclipboard-is-active { color: #1878fa; text-decoration: none; background-color: transparent; box-shadow: inset 0 0 0 3px #e6e1e8; } +.btn-outline:focus { text-decoration: none; outline: none; box-shadow: inset 0 0 0 2px #5c5962, 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn-outline:focus:hover, .btn-outline.selected:focus { box-shadow: inset 0 0 0 2px #5c5962; } + +.btn-primary { color: #fff; background-color: #2448a7; background-image: linear-gradient(#2b55c4, #2448a7); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-primary:hover, .btn-primary.zeroclipboard-is-hover { color: #fff; background-color: #22459e; background-image: linear-gradient(#2850b7, #22459e); } +.btn-primary:active, .btn-primary.selected, .btn-primary.zeroclipboard-is-active { background-color: #21439a; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-primary.selected:hover { background-color: #1d3a85; } + +.btn-purple { color: #fff; background-color: #5739ce; background-image: linear-gradient(#6f55d5, #5739ce); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-purple:hover, .btn-purple.zeroclipboard-is-hover { color: #fff; background-color: #5132cb; background-image: linear-gradient(#6549d2, #5132cb); } +.btn-purple:active, .btn-purple.selected, .btn-purple.zeroclipboard-is-active { background-color: #4f31c6; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-purple.selected:hover { background-color: #472cb2; } + +.btn-blue { color: #fff; background-color: #227efa; background-image: linear-gradient(#4593fb, #227efa); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-blue:hover, .btn-blue.zeroclipboard-is-hover { color: #fff; background-color: #1878fa; background-image: linear-gradient(#368afa, #1878fa); } +.btn-blue:active, .btn-blue.selected, .btn-blue.zeroclipboard-is-active { background-color: #1375f9; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-blue.selected:hover { background-color: #0669ed; } + +.btn-green { color: #fff; background-color: #10ac7d; background-image: linear-gradient(#13cc95, #10ac7d); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-green:hover, .btn-green.zeroclipboard-is-hover { color: #fff; background-color: #0fa276; background-image: linear-gradient(#12be8b, #0fa276); } +.btn-green:active, .btn-green.selected, .btn-green.zeroclipboard-is-active { background-color: #0f9e73; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-green.selected:hover { background-color: #0d8662; } + +.btn-reset { background: none; border: none; margin: 0; text-align: inherit; font: inherit; border-radius: 0; appearance: none; } + +.search { position: relative; z-index: 2; flex-grow: 1; height: 4rem; padding: 0.5rem; transition: padding linear 200ms; } +@media (min-width: 50rem) { .search { position: relative !important; width: auto !important; height: 100% !important; padding: 0; transition: none; } } + +.search-input-wrap { position: relative; z-index: 1; height: 3rem; overflow: hidden; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); transition: height linear 200ms; } +@media (min-width: 50rem) { .search-input-wrap { position: absolute; width: 100%; max-width: 33.5rem; height: 100% !important; border-radius: 0; box-shadow: none; transition: width ease 400ms; } } + +.search-input { position: absolute; width: 100%; height: 100%; padding: 0.5rem 1rem 0.5rem 2.5rem; font-size: 1rem; color: #e6e1e8; background-color: #302d36; border-top: 0; border-right: 0; border-bottom: 0; border-left: 0; border-radius: 0; } +@media (min-width: 50rem) { .search-input { padding: 0.5rem 1rem 0.5rem 3.5rem; font-size: 0.875rem; background-color: #27262b; transition: padding-left linear 200ms; } } +.search-input:focus { outline: 0; } +.search-input:focus + .search-label .search-icon { color: #2c84fa; } + +.search-label { position: absolute; display: flex; height: 100%; padding-left: 1rem; } +@media (min-width: 50rem) { .search-label { padding-left: 2rem; transition: padding-left linear 200ms; } } +.search-label .search-icon { width: 1.2rem; height: 1.2rem; align-self: center; color: #959396; } + +.search-results { position: absolute; left: 0; display: none; width: 100%; max-height: calc(100% - 4rem); overflow-y: auto; background-color: #302d36; border-bottom-right-radius: 4px; border-bottom-left-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } +@media (min-width: 50rem) { .search-results { top: 100%; width: 33.5rem; max-height: calc(100vh - 200%) !important; } } + +.search-results-list { padding-left: 0; margin-bottom: 0.25rem; list-style: none; font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .search-results-list { font-size: 1rem !important; } } +@media (min-width: 50rem) { .search-results-list { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-results-list { font-size: 0.875rem !important; } } + +.search-results-list-item { padding: 0; margin: 0; } + +.search-result { display: block; padding: 0.25rem 0.75rem; } +.search-result:hover, .search-result.active { background-color: #201f23; } + +.search-result-title { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; } +@media (min-width: 31.25rem) { .search-result-title { display: inline-block; width: 40%; padding-right: 0.5rem; vertical-align: top; } } + +.search-result-doc { display: flex; align-items: center; word-wrap: break-word; } +.search-result-doc.search-result-doc-parent { opacity: 0.5; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.875rem !important; } } +@media (min-width: 50rem) { .search-result-doc.search-result-doc-parent { font-size: 0.6875rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.75rem !important; } } + +.search-result-doc .search-result-icon { width: 1rem; height: 1rem; margin-right: 0.5rem; color: #2c84fa; flex-shrink: 0; } +.search-result-doc .search-result-doc-title { overflow: auto; } + +.search-result-section { margin-left: 1.5rem; word-wrap: break-word; } + +.search-result-rel-url { display: block; margin-left: 1.5rem; overflow: hidden; color: #959396; text-overflow: ellipsis; white-space: nowrap; font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .search-result-rel-url { font-size: 0.625rem !important; } } + +.search-result-previews { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; padding-left: 1rem; margin-left: 0.5rem; color: #959396; word-wrap: break-word; border-left: 1px solid; border-left-color: #44434d; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .search-result-previews { font-size: 0.75rem !important; } } +@media (min-width: 31.25rem) { .search-result-previews { display: inline-block; width: 60%; padding-left: 0.5rem; margin-left: 0; vertical-align: top; } } + +.search-result-preview + .search-result-preview { margin-top: 0.25rem; } + +.search-result-highlight { font-weight: bold; } + +.search-no-result { padding: 0.5rem 0.75rem; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-no-result { font-size: 0.875rem !important; } } + +.search-button { position: fixed; right: 1rem; bottom: 1rem; display: flex; width: 3.5rem; height: 3.5rem; background-color: #302d36; border: 1px solid rgba(44, 132, 250, 0.3); border-radius: 1.75rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); align-items: center; justify-content: center; } + +.search-overlay { position: fixed; top: 0; left: 0; z-index: 1; width: 0; height: 0; background-color: rgba(0, 0, 0, 0.3); opacity: 0; transition: opacity ease 400ms, width 0s 400ms, height 0s 400ms; } + +.search-active .search { position: fixed; top: 0; left: 0; width: 100%; height: 100%; padding: 0; } +.search-active .search-input-wrap { height: 4rem; border-radius: 0; } +@media (min-width: 50rem) { .search-active .search-input-wrap { width: 33.5rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } } +.search-active .search-input { background-color: #302d36; } +@media (min-width: 50rem) { .search-active .search-input { padding-left: 2.3rem; } } +@media (min-width: 50rem) { .search-active .search-label { padding-left: 0.6rem; } } +.search-active .search-results { display: block; } +.search-active .search-overlay { width: 100%; height: 100%; opacity: 1; transition: opacity ease 400ms, width 0s, height 0s; } +@media (min-width: 50rem) { .search-active .main { position: fixed; right: 0; left: 0; } } +.search-active .main-header { padding-top: 4rem; } +@media (min-width: 50rem) { .search-active .main-header { padding-top: 0; } } + +.table-wrapper { display: block; width: 100%; max-width: 100%; margin-bottom: 1.5rem; overflow-x: auto; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } + +table { display: table; min-width: 100%; border-collapse: separate; } + +th, td { font-size: 0.75rem !important; min-width: 7.5rem; padding: 0.5rem 0.75rem; background-color: #302d36; border-bottom: 1px solid rgba(68, 67, 77, 0.5); border-left: 1px solid #44434d; } +@media (min-width: 31.25rem) { th, td { font-size: 0.875rem !important; } } +th:first-of-type, td:first-of-type { border-left: 0; } + +tbody tr:last-of-type th, tbody tr:last-of-type td { border-bottom: 0; } +tbody tr:last-of-type td { padding-bottom: 0.75rem; } + +thead th { border-bottom: 1px solid #44434d; } + +:not(pre, figure) > code { padding: 0.2em 0.15em; font-weight: 400; background-color: #31343f; border: 1px solid #44434d; border-radius: 4px; } + +a:visited code { border-color: #44434d; } + +div.highlighter-rouge, div.listingblock > div.content, figure.highlight { margin-top: 0; margin-bottom: 0.75rem; background-color: #31343f; border-radius: 4px; box-shadow: none; -webkit-overflow-scrolling: touch; position: relative; padding: 0; } +div.highlighter-rouge > button, div.listingblock > div.content > button, figure.highlight > button { width: 0.75rem; opacity: 0; position: absolute; top: 0; right: 0; border: 0.75rem solid #31343f; background-color: #31343f; color: #e6e1e8; box-sizing: content-box; } +div.highlighter-rouge > button svg, div.listingblock > div.content > button svg, figure.highlight > button svg { fill: #e6e1e8; } +div.highlighter-rouge > button:active, div.listingblock > div.content > button:active, figure.highlight > button:active { text-decoration: none; outline: none; opacity: 1; } +div.highlighter-rouge > button:focus, div.listingblock > div.content > button:focus, figure.highlight > button:focus { opacity: 1; } +div.highlighter-rouge:hover > button, div.listingblock > div.content:hover > button, figure.highlight:hover > button { cursor: copy; opacity: 1; } + +div.highlighter-rouge div.highlight { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.highlighter-rouge pre.highlight, div.highlighter-rouge code { padding: 0; margin: 0; border: 0; } + +div.listingblock { margin-top: 0; margin-bottom: 0.75rem; } +div.listingblock div.content { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.listingblock div.content > pre, div.listingblock code { padding: 0; margin: 0; border: 0; } + +figure.highlight pre, figure.highlight :not(pre) > code { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } + +.highlight .table-wrapper { padding: 0.75rem 0; margin: 0; border: 0; box-shadow: none; } +.highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.6875rem !important; min-width: 0; padding: 0; background-color: #31343f; border: 0; } +@media (min-width: 31.25rem) { .highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.75rem !important; } } +.highlight .table-wrapper td.gl { width: 1em; padding-right: 0.75rem; padding-left: 0.75rem; } +.highlight .table-wrapper pre { margin: 0; line-height: 2; } + +.code-example, .listingblock > .title { padding: 0.75rem; margin-bottom: 0.75rem; overflow: auto; border: 1px solid #44434d; border-radius: 4px; } +.code-example + .highlighter-rouge, .code-example + .sectionbody .listingblock, .code-example + .content, .code-example + figure.highlight, .listingblock > .title + .highlighter-rouge, .listingblock > .title + .sectionbody .listingblock, .listingblock > .title + .content, .listingblock > .title + figure.highlight { position: relative; margin-top: -1rem; border-right: 1px solid #44434d; border-bottom: 1px solid #44434d; border-left: 1px solid #44434d; border-top-left-radius: 0; border-top-right-radius: 0; } + +code.language-mermaid { padding: 0; background-color: inherit; border: 0; } + +.highlight, pre.highlight { background: #31343f; color: #dee2f7; } + +.highlight pre { background: #31343f; } + +.text-grey-dk-000 { color: #959396 !important; } + +.text-grey-dk-100 { color: #5c5962 !important; } + +.text-grey-dk-200 { color: #44434d !important; } + +.text-grey-dk-250 { color: #302d36 !important; } + +.text-grey-dk-300 { color: #27262b !important; } + +.text-grey-lt-000 { color: #f5f6fa !important; } + +.text-grey-lt-100 { color: #eeebee !important; } + +.text-grey-lt-200 { color: #ecebed !important; } + +.text-grey-lt-300 { color: #e6e1e8 !important; } + +.text-blue-000 { color: #2c84fa !important; } + +.text-blue-100 { color: #2869e6 !important; } + +.text-blue-200 { color: #264caf !important; } + +.text-blue-300 { color: #183385 !important; } + +.text-green-000 { color: #41d693 !important; } + +.text-green-100 { color: #11b584 !important; } + +.text-green-200 { color: #009c7b !important; } + +.text-green-300 { color: #026e57 !important; } + +.text-purple-000 { color: #7253ed !important; } + +.text-purple-100 { color: #5e41d0 !important; } + +.text-purple-200 { color: #4e26af !important; } + +.text-purple-300 { color: #381885 !important; } + +.text-yellow-000 { color: #ffeb82 !important; } + +.text-yellow-100 { color: #fadf50 !important; } + +.text-yellow-200 { color: #f7d12e !important; } + +.text-yellow-300 { color: #e7af06 !important; } + +.text-red-000 { color: #f77e7e !important; } + +.text-red-100 { color: #f96e65 !important; } + +.text-red-200 { color: #e94c4c !important; } + +.text-red-300 { color: #dd2e2e !important; } + +.bg-grey-dk-000 { background-color: #959396 !important; } + +.bg-grey-dk-100 { background-color: #5c5962 !important; } + +.bg-grey-dk-200 { background-color: #44434d !important; } + +.bg-grey-dk-250 { background-color: #302d36 !important; } + +.bg-grey-dk-300 { background-color: #27262b !important; } + +.bg-grey-lt-000 { background-color: #f5f6fa !important; } + +.bg-grey-lt-100 { background-color: #eeebee !important; } + +.bg-grey-lt-200 { background-color: #ecebed !important; } + +.bg-grey-lt-300 { background-color: #e6e1e8 !important; } + +.bg-blue-000 { background-color: #2c84fa !important; } + +.bg-blue-100 { background-color: #2869e6 !important; } + +.bg-blue-200 { background-color: #264caf !important; } + +.bg-blue-300 { background-color: #183385 !important; } + +.bg-green-000 { background-color: #41d693 !important; } + +.bg-green-100 { background-color: #11b584 !important; } + +.bg-green-200 { background-color: #009c7b !important; } + +.bg-green-300 { background-color: #026e57 !important; } + +.bg-purple-000 { background-color: #7253ed !important; } + +.bg-purple-100 { background-color: #5e41d0 !important; } + +.bg-purple-200 { background-color: #4e26af !important; } + +.bg-purple-300 { background-color: #381885 !important; } + +.bg-yellow-000 { background-color: #ffeb82 !important; } + +.bg-yellow-100 { background-color: #fadf50 !important; } + +.bg-yellow-200 { background-color: #f7d12e !important; } + +.bg-yellow-300 { background-color: #e7af06 !important; } + +.bg-red-000 { background-color: #f77e7e !important; } + +.bg-red-100 { background-color: #f96e65 !important; } + +.bg-red-200 { background-color: #e94c4c !important; } + +.bg-red-300 { background-color: #dd2e2e !important; } + +.d-block { display: block !important; } + +.d-flex { display: flex !important; } + +.d-inline { display: inline !important; } + +.d-inline-block { display: inline-block !important; } + +.d-none { display: none !important; } + +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +.float-left { float: left !important; } + +.float-right { float: right !important; } + +.flex-justify-start { justify-content: flex-start !important; } + +.flex-justify-end { justify-content: flex-end !important; } + +.flex-justify-between { justify-content: space-between !important; } + +.flex-justify-around { justify-content: space-around !important; } + +.v-align-baseline { vertical-align: baseline !important; } + +.v-align-bottom { vertical-align: bottom !important; } + +.v-align-middle { vertical-align: middle !important; } + +.v-align-text-bottom { vertical-align: text-bottom !important; } + +.v-align-text-top { vertical-align: text-top !important; } + +.v-align-top { vertical-align: top !important; } + +.fs-1 { font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .fs-1 { font-size: 0.625rem !important; } } + +.fs-2 { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .fs-2 { font-size: 0.75rem !important; } } + +.fs-3 { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .fs-3 { font-size: 0.875rem !important; } } + +.fs-4 { font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .fs-4 { font-size: 1rem !important; } } + +.fs-5 { font-size: 1rem !important; } +@media (min-width: 31.25rem) { .fs-5 { font-size: 1.125rem !important; } } + +.fs-6 { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { .fs-6 { font-size: 1.5rem !important; line-height: 1.25; } } + +.fs-7 { font-size: 1.5rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-7 { font-size: 2rem !important; } } + +.fs-8 { font-size: 2rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-8 { font-size: 2.25rem !important; } } + +.fs-9 { font-size: 2.25rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-9 { font-size: 2.625rem !important; } } + +.fs-10 { font-size: 2.625rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-10 { font-size: 3rem !important; } } + +.fw-300 { font-weight: 300 !important; } + +.fw-400 { font-weight: 400 !important; } + +.fw-500 { font-weight: 500 !important; } + +.fw-700 { font-weight: 700 !important; } + +.lh-0 { line-height: 0 !important; } + +.lh-default { line-height: 1.4; } + +.lh-tight { line-height: 1.25; } + +.ls-5 { letter-spacing: 0.05em !important; } + +.ls-10 { letter-spacing: 0.1em !important; } + +.ls-0 { letter-spacing: 0 !important; } + +.text-uppercase { text-transform: uppercase !important; } + +.list-style-none { padding: 0 !important; margin: 0 !important; list-style: none !important; } +.list-style-none li::before { display: none !important; } + +.mx-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-0 { margin: 0 !important; } + +.mt-0 { margin-top: 0 !important; } + +.mr-0 { margin-right: 0 !important; } + +.mb-0 { margin-bottom: 0 !important; } + +.ml-0 { margin-left: 0 !important; } + +.mx-0 { margin-right: 0 !important; margin-left: 0 !important; } + +.my-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + +.mxn-0 { margin-right: -0 !important; margin-left: -0 !important; } + +.mx-0-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-1 { margin: 0.25rem !important; } + +.mt-1 { margin-top: 0.25rem !important; } + +.mr-1 { margin-right: 0.25rem !important; } + +.mb-1 { margin-bottom: 0.25rem !important; } + +.ml-1 { margin-left: 0.25rem !important; } + +.mx-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + +.my-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + +.mxn-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } + +.mx-1-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-2 { margin: 0.5rem !important; } + +.mt-2 { margin-top: 0.5rem !important; } + +.mr-2 { margin-right: 0.5rem !important; } + +.mb-2 { margin-bottom: 0.5rem !important; } + +.ml-2 { margin-left: 0.5rem !important; } + +.mx-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + +.my-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + +.mxn-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } + +.mx-2-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-3 { margin: 0.75rem !important; } + +.mt-3 { margin-top: 0.75rem !important; } + +.mr-3 { margin-right: 0.75rem !important; } + +.mb-3 { margin-bottom: 0.75rem !important; } + +.ml-3 { margin-left: 0.75rem !important; } + +.mx-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + +.my-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + +.mxn-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } + +.mx-3-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-4 { margin: 1rem !important; } + +.mt-4 { margin-top: 1rem !important; } + +.mr-4 { margin-right: 1rem !important; } + +.mb-4 { margin-bottom: 1rem !important; } + +.ml-4 { margin-left: 1rem !important; } + +.mx-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + +.my-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + +.mxn-4 { margin-right: -1rem !important; margin-left: -1rem !important; } + +.mx-4-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-5 { margin: 1.5rem !important; } + +.mt-5 { margin-top: 1.5rem !important; } + +.mr-5 { margin-right: 1.5rem !important; } + +.mb-5 { margin-bottom: 1.5rem !important; } + +.ml-5 { margin-left: 1.5rem !important; } + +.mx-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + +.my-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + +.mxn-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } + +.mx-5-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-6 { margin: 2rem !important; } + +.mt-6 { margin-top: 2rem !important; } + +.mr-6 { margin-right: 2rem !important; } + +.mb-6 { margin-bottom: 2rem !important; } + +.ml-6 { margin-left: 2rem !important; } + +.mx-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + +.my-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + +.mxn-6 { margin-right: -2rem !important; margin-left: -2rem !important; } + +.mx-6-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-7 { margin: 2.5rem !important; } + +.mt-7 { margin-top: 2.5rem !important; } + +.mr-7 { margin-right: 2.5rem !important; } + +.mb-7 { margin-bottom: 2.5rem !important; } + +.ml-7 { margin-left: 2.5rem !important; } + +.mx-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + +.my-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + +.mxn-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } + +.mx-7-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-8 { margin: 3rem !important; } + +.mt-8 { margin-top: 3rem !important; } + +.mr-8 { margin-right: 3rem !important; } + +.mb-8 { margin-bottom: 3rem !important; } + +.ml-8 { margin-left: 3rem !important; } + +.mx-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + +.my-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + +.mxn-8 { margin-right: -3rem !important; margin-left: -3rem !important; } + +.mx-8-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-9 { margin: 3.5rem !important; } + +.mt-9 { margin-top: 3.5rem !important; } + +.mr-9 { margin-right: 3.5rem !important; } + +.mb-9 { margin-bottom: 3.5rem !important; } + +.ml-9 { margin-left: 3.5rem !important; } + +.mx-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + +.my-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + +.mxn-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } + +.mx-9-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-10 { margin: 4rem !important; } + +.mt-10 { margin-top: 4rem !important; } + +.mr-10 { margin-right: 4rem !important; } + +.mb-10 { margin-bottom: 4rem !important; } + +.ml-10 { margin-left: 4rem !important; } + +.mx-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + +.my-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + +.mxn-10 { margin-right: -4rem !important; margin-left: -4rem !important; } + +.mx-10-auto { margin-right: auto !important; margin-left: auto !important; } + +@media (min-width: 20rem) { .m-xs-0 { margin: 0 !important; } + .mt-xs-0 { margin-top: 0 !important; } + .mr-xs-0 { margin-right: 0 !important; } + .mb-xs-0 { margin-bottom: 0 !important; } + .ml-xs-0 { margin-left: 0 !important; } + .mx-xs-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xs-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xs-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 20rem) { .m-xs-1 { margin: 0.25rem !important; } + .mt-xs-1 { margin-top: 0.25rem !important; } + .mr-xs-1 { margin-right: 0.25rem !important; } + .mb-xs-1 { margin-bottom: 0.25rem !important; } + .ml-xs-1 { margin-left: 0.25rem !important; } + .mx-xs-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xs-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xs-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 20rem) { .m-xs-2 { margin: 0.5rem !important; } + .mt-xs-2 { margin-top: 0.5rem !important; } + .mr-xs-2 { margin-right: 0.5rem !important; } + .mb-xs-2 { margin-bottom: 0.5rem !important; } + .ml-xs-2 { margin-left: 0.5rem !important; } + .mx-xs-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xs-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xs-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 20rem) { .m-xs-3 { margin: 0.75rem !important; } + .mt-xs-3 { margin-top: 0.75rem !important; } + .mr-xs-3 { margin-right: 0.75rem !important; } + .mb-xs-3 { margin-bottom: 0.75rem !important; } + .ml-xs-3 { margin-left: 0.75rem !important; } + .mx-xs-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xs-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xs-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 20rem) { .m-xs-4 { margin: 1rem !important; } + .mt-xs-4 { margin-top: 1rem !important; } + .mr-xs-4 { margin-right: 1rem !important; } + .mb-xs-4 { margin-bottom: 1rem !important; } + .ml-xs-4 { margin-left: 1rem !important; } + .mx-xs-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xs-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xs-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 20rem) { .m-xs-5 { margin: 1.5rem !important; } + .mt-xs-5 { margin-top: 1.5rem !important; } + .mr-xs-5 { margin-right: 1.5rem !important; } + .mb-xs-5 { margin-bottom: 1.5rem !important; } + .ml-xs-5 { margin-left: 1.5rem !important; } + .mx-xs-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xs-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xs-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 20rem) { .m-xs-6 { margin: 2rem !important; } + .mt-xs-6 { margin-top: 2rem !important; } + .mr-xs-6 { margin-right: 2rem !important; } + .mb-xs-6 { margin-bottom: 2rem !important; } + .ml-xs-6 { margin-left: 2rem !important; } + .mx-xs-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xs-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xs-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 20rem) { .m-xs-7 { margin: 2.5rem !important; } + .mt-xs-7 { margin-top: 2.5rem !important; } + .mr-xs-7 { margin-right: 2.5rem !important; } + .mb-xs-7 { margin-bottom: 2.5rem !important; } + .ml-xs-7 { margin-left: 2.5rem !important; } + .mx-xs-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xs-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xs-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 20rem) { .m-xs-8 { margin: 3rem !important; } + .mt-xs-8 { margin-top: 3rem !important; } + .mr-xs-8 { margin-right: 3rem !important; } + .mb-xs-8 { margin-bottom: 3rem !important; } + .ml-xs-8 { margin-left: 3rem !important; } + .mx-xs-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xs-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xs-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 20rem) { .m-xs-9 { margin: 3.5rem !important; } + .mt-xs-9 { margin-top: 3.5rem !important; } + .mr-xs-9 { margin-right: 3.5rem !important; } + .mb-xs-9 { margin-bottom: 3.5rem !important; } + .ml-xs-9 { margin-left: 3.5rem !important; } + .mx-xs-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xs-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xs-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 20rem) { .m-xs-10 { margin: 4rem !important; } + .mt-xs-10 { margin-top: 4rem !important; } + .mr-xs-10 { margin-right: 4rem !important; } + .mb-xs-10 { margin-bottom: 4rem !important; } + .ml-xs-10 { margin-left: 4rem !important; } + .mx-xs-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xs-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xs-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 31.25rem) { .m-sm-0 { margin: 0 !important; } + .mt-sm-0 { margin-top: 0 !important; } + .mr-sm-0 { margin-right: 0 !important; } + .mb-sm-0 { margin-bottom: 0 !important; } + .ml-sm-0 { margin-left: 0 !important; } + .mx-sm-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-sm-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-sm-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 31.25rem) { .m-sm-1 { margin: 0.25rem !important; } + .mt-sm-1 { margin-top: 0.25rem !important; } + .mr-sm-1 { margin-right: 0.25rem !important; } + .mb-sm-1 { margin-bottom: 0.25rem !important; } + .ml-sm-1 { margin-left: 0.25rem !important; } + .mx-sm-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-sm-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-sm-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 31.25rem) { .m-sm-2 { margin: 0.5rem !important; } + .mt-sm-2 { margin-top: 0.5rem !important; } + .mr-sm-2 { margin-right: 0.5rem !important; } + .mb-sm-2 { margin-bottom: 0.5rem !important; } + .ml-sm-2 { margin-left: 0.5rem !important; } + .mx-sm-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-sm-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-sm-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-3 { margin: 0.75rem !important; } + .mt-sm-3 { margin-top: 0.75rem !important; } + .mr-sm-3 { margin-right: 0.75rem !important; } + .mb-sm-3 { margin-bottom: 0.75rem !important; } + .ml-sm-3 { margin-left: 0.75rem !important; } + .mx-sm-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-sm-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-sm-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 31.25rem) { .m-sm-4 { margin: 1rem !important; } + .mt-sm-4 { margin-top: 1rem !important; } + .mr-sm-4 { margin-right: 1rem !important; } + .mb-sm-4 { margin-bottom: 1rem !important; } + .ml-sm-4 { margin-left: 1rem !important; } + .mx-sm-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-sm-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-sm-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 31.25rem) { .m-sm-5 { margin: 1.5rem !important; } + .mt-sm-5 { margin-top: 1.5rem !important; } + .mr-sm-5 { margin-right: 1.5rem !important; } + .mb-sm-5 { margin-bottom: 1.5rem !important; } + .ml-sm-5 { margin-left: 1.5rem !important; } + .mx-sm-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-sm-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-sm-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-6 { margin: 2rem !important; } + .mt-sm-6 { margin-top: 2rem !important; } + .mr-sm-6 { margin-right: 2rem !important; } + .mb-sm-6 { margin-bottom: 2rem !important; } + .ml-sm-6 { margin-left: 2rem !important; } + .mx-sm-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-sm-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-sm-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 31.25rem) { .m-sm-7 { margin: 2.5rem !important; } + .mt-sm-7 { margin-top: 2.5rem !important; } + .mr-sm-7 { margin-right: 2.5rem !important; } + .mb-sm-7 { margin-bottom: 2.5rem !important; } + .ml-sm-7 { margin-left: 2.5rem !important; } + .mx-sm-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-sm-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-sm-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-8 { margin: 3rem !important; } + .mt-sm-8 { margin-top: 3rem !important; } + .mr-sm-8 { margin-right: 3rem !important; } + .mb-sm-8 { margin-bottom: 3rem !important; } + .ml-sm-8 { margin-left: 3rem !important; } + .mx-sm-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-sm-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-sm-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 31.25rem) { .m-sm-9 { margin: 3.5rem !important; } + .mt-sm-9 { margin-top: 3.5rem !important; } + .mr-sm-9 { margin-right: 3.5rem !important; } + .mb-sm-9 { margin-bottom: 3.5rem !important; } + .ml-sm-9 { margin-left: 3.5rem !important; } + .mx-sm-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-sm-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-sm-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-10 { margin: 4rem !important; } + .mt-sm-10 { margin-top: 4rem !important; } + .mr-sm-10 { margin-right: 4rem !important; } + .mb-sm-10 { margin-bottom: 4rem !important; } + .ml-sm-10 { margin-left: 4rem !important; } + .mx-sm-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-sm-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-sm-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 50rem) { .m-md-0 { margin: 0 !important; } + .mt-md-0 { margin-top: 0 !important; } + .mr-md-0 { margin-right: 0 !important; } + .mb-md-0 { margin-bottom: 0 !important; } + .ml-md-0 { margin-left: 0 !important; } + .mx-md-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-md-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-md-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 50rem) { .m-md-1 { margin: 0.25rem !important; } + .mt-md-1 { margin-top: 0.25rem !important; } + .mr-md-1 { margin-right: 0.25rem !important; } + .mb-md-1 { margin-bottom: 0.25rem !important; } + .ml-md-1 { margin-left: 0.25rem !important; } + .mx-md-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-md-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-md-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 50rem) { .m-md-2 { margin: 0.5rem !important; } + .mt-md-2 { margin-top: 0.5rem !important; } + .mr-md-2 { margin-right: 0.5rem !important; } + .mb-md-2 { margin-bottom: 0.5rem !important; } + .ml-md-2 { margin-left: 0.5rem !important; } + .mx-md-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-md-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-md-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 50rem) { .m-md-3 { margin: 0.75rem !important; } + .mt-md-3 { margin-top: 0.75rem !important; } + .mr-md-3 { margin-right: 0.75rem !important; } + .mb-md-3 { margin-bottom: 0.75rem !important; } + .ml-md-3 { margin-left: 0.75rem !important; } + .mx-md-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-md-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-md-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 50rem) { .m-md-4 { margin: 1rem !important; } + .mt-md-4 { margin-top: 1rem !important; } + .mr-md-4 { margin-right: 1rem !important; } + .mb-md-4 { margin-bottom: 1rem !important; } + .ml-md-4 { margin-left: 1rem !important; } + .mx-md-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-md-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-md-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 50rem) { .m-md-5 { margin: 1.5rem !important; } + .mt-md-5 { margin-top: 1.5rem !important; } + .mr-md-5 { margin-right: 1.5rem !important; } + .mb-md-5 { margin-bottom: 1.5rem !important; } + .ml-md-5 { margin-left: 1.5rem !important; } + .mx-md-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-md-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-md-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 50rem) { .m-md-6 { margin: 2rem !important; } + .mt-md-6 { margin-top: 2rem !important; } + .mr-md-6 { margin-right: 2rem !important; } + .mb-md-6 { margin-bottom: 2rem !important; } + .ml-md-6 { margin-left: 2rem !important; } + .mx-md-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-md-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-md-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 50rem) { .m-md-7 { margin: 2.5rem !important; } + .mt-md-7 { margin-top: 2.5rem !important; } + .mr-md-7 { margin-right: 2.5rem !important; } + .mb-md-7 { margin-bottom: 2.5rem !important; } + .ml-md-7 { margin-left: 2.5rem !important; } + .mx-md-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-md-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-md-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 50rem) { .m-md-8 { margin: 3rem !important; } + .mt-md-8 { margin-top: 3rem !important; } + .mr-md-8 { margin-right: 3rem !important; } + .mb-md-8 { margin-bottom: 3rem !important; } + .ml-md-8 { margin-left: 3rem !important; } + .mx-md-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-md-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-md-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 50rem) { .m-md-9 { margin: 3.5rem !important; } + .mt-md-9 { margin-top: 3.5rem !important; } + .mr-md-9 { margin-right: 3.5rem !important; } + .mb-md-9 { margin-bottom: 3.5rem !important; } + .ml-md-9 { margin-left: 3.5rem !important; } + .mx-md-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-md-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-md-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 50rem) { .m-md-10 { margin: 4rem !important; } + .mt-md-10 { margin-top: 4rem !important; } + .mr-md-10 { margin-right: 4rem !important; } + .mb-md-10 { margin-bottom: 4rem !important; } + .ml-md-10 { margin-left: 4rem !important; } + .mx-md-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-md-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-md-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 66.5rem) { .m-lg-0 { margin: 0 !important; } + .mt-lg-0 { margin-top: 0 !important; } + .mr-lg-0 { margin-right: 0 !important; } + .mb-lg-0 { margin-bottom: 0 !important; } + .ml-lg-0 { margin-left: 0 !important; } + .mx-lg-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-lg-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-lg-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 66.5rem) { .m-lg-1 { margin: 0.25rem !important; } + .mt-lg-1 { margin-top: 0.25rem !important; } + .mr-lg-1 { margin-right: 0.25rem !important; } + .mb-lg-1 { margin-bottom: 0.25rem !important; } + .ml-lg-1 { margin-left: 0.25rem !important; } + .mx-lg-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-lg-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-lg-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 66.5rem) { .m-lg-2 { margin: 0.5rem !important; } + .mt-lg-2 { margin-top: 0.5rem !important; } + .mr-lg-2 { margin-right: 0.5rem !important; } + .mb-lg-2 { margin-bottom: 0.5rem !important; } + .ml-lg-2 { margin-left: 0.5rem !important; } + .mx-lg-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-lg-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-lg-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-3 { margin: 0.75rem !important; } + .mt-lg-3 { margin-top: 0.75rem !important; } + .mr-lg-3 { margin-right: 0.75rem !important; } + .mb-lg-3 { margin-bottom: 0.75rem !important; } + .ml-lg-3 { margin-left: 0.75rem !important; } + .mx-lg-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-lg-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-lg-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 66.5rem) { .m-lg-4 { margin: 1rem !important; } + .mt-lg-4 { margin-top: 1rem !important; } + .mr-lg-4 { margin-right: 1rem !important; } + .mb-lg-4 { margin-bottom: 1rem !important; } + .ml-lg-4 { margin-left: 1rem !important; } + .mx-lg-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-lg-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-lg-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 66.5rem) { .m-lg-5 { margin: 1.5rem !important; } + .mt-lg-5 { margin-top: 1.5rem !important; } + .mr-lg-5 { margin-right: 1.5rem !important; } + .mb-lg-5 { margin-bottom: 1.5rem !important; } + .ml-lg-5 { margin-left: 1.5rem !important; } + .mx-lg-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-lg-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-lg-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-6 { margin: 2rem !important; } + .mt-lg-6 { margin-top: 2rem !important; } + .mr-lg-6 { margin-right: 2rem !important; } + .mb-lg-6 { margin-bottom: 2rem !important; } + .ml-lg-6 { margin-left: 2rem !important; } + .mx-lg-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-lg-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-lg-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 66.5rem) { .m-lg-7 { margin: 2.5rem !important; } + .mt-lg-7 { margin-top: 2.5rem !important; } + .mr-lg-7 { margin-right: 2.5rem !important; } + .mb-lg-7 { margin-bottom: 2.5rem !important; } + .ml-lg-7 { margin-left: 2.5rem !important; } + .mx-lg-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-lg-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-lg-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-8 { margin: 3rem !important; } + .mt-lg-8 { margin-top: 3rem !important; } + .mr-lg-8 { margin-right: 3rem !important; } + .mb-lg-8 { margin-bottom: 3rem !important; } + .ml-lg-8 { margin-left: 3rem !important; } + .mx-lg-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-lg-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-lg-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 66.5rem) { .m-lg-9 { margin: 3.5rem !important; } + .mt-lg-9 { margin-top: 3.5rem !important; } + .mr-lg-9 { margin-right: 3.5rem !important; } + .mb-lg-9 { margin-bottom: 3.5rem !important; } + .ml-lg-9 { margin-left: 3.5rem !important; } + .mx-lg-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-lg-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-lg-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-10 { margin: 4rem !important; } + .mt-lg-10 { margin-top: 4rem !important; } + .mr-lg-10 { margin-right: 4rem !important; } + .mb-lg-10 { margin-bottom: 4rem !important; } + .ml-lg-10 { margin-left: 4rem !important; } + .mx-lg-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-lg-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-lg-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 87.5rem) { .m-xl-0 { margin: 0 !important; } + .mt-xl-0 { margin-top: 0 !important; } + .mr-xl-0 { margin-right: 0 !important; } + .mb-xl-0 { margin-bottom: 0 !important; } + .ml-xl-0 { margin-left: 0 !important; } + .mx-xl-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xl-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xl-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 87.5rem) { .m-xl-1 { margin: 0.25rem !important; } + .mt-xl-1 { margin-top: 0.25rem !important; } + .mr-xl-1 { margin-right: 0.25rem !important; } + .mb-xl-1 { margin-bottom: 0.25rem !important; } + .ml-xl-1 { margin-left: 0.25rem !important; } + .mx-xl-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xl-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xl-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 87.5rem) { .m-xl-2 { margin: 0.5rem !important; } + .mt-xl-2 { margin-top: 0.5rem !important; } + .mr-xl-2 { margin-right: 0.5rem !important; } + .mb-xl-2 { margin-bottom: 0.5rem !important; } + .ml-xl-2 { margin-left: 0.5rem !important; } + .mx-xl-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xl-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xl-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-3 { margin: 0.75rem !important; } + .mt-xl-3 { margin-top: 0.75rem !important; } + .mr-xl-3 { margin-right: 0.75rem !important; } + .mb-xl-3 { margin-bottom: 0.75rem !important; } + .ml-xl-3 { margin-left: 0.75rem !important; } + .mx-xl-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xl-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xl-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 87.5rem) { .m-xl-4 { margin: 1rem !important; } + .mt-xl-4 { margin-top: 1rem !important; } + .mr-xl-4 { margin-right: 1rem !important; } + .mb-xl-4 { margin-bottom: 1rem !important; } + .ml-xl-4 { margin-left: 1rem !important; } + .mx-xl-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xl-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xl-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 87.5rem) { .m-xl-5 { margin: 1.5rem !important; } + .mt-xl-5 { margin-top: 1.5rem !important; } + .mr-xl-5 { margin-right: 1.5rem !important; } + .mb-xl-5 { margin-bottom: 1.5rem !important; } + .ml-xl-5 { margin-left: 1.5rem !important; } + .mx-xl-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xl-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xl-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-6 { margin: 2rem !important; } + .mt-xl-6 { margin-top: 2rem !important; } + .mr-xl-6 { margin-right: 2rem !important; } + .mb-xl-6 { margin-bottom: 2rem !important; } + .ml-xl-6 { margin-left: 2rem !important; } + .mx-xl-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xl-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xl-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 87.5rem) { .m-xl-7 { margin: 2.5rem !important; } + .mt-xl-7 { margin-top: 2.5rem !important; } + .mr-xl-7 { margin-right: 2.5rem !important; } + .mb-xl-7 { margin-bottom: 2.5rem !important; } + .ml-xl-7 { margin-left: 2.5rem !important; } + .mx-xl-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xl-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xl-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-8 { margin: 3rem !important; } + .mt-xl-8 { margin-top: 3rem !important; } + .mr-xl-8 { margin-right: 3rem !important; } + .mb-xl-8 { margin-bottom: 3rem !important; } + .ml-xl-8 { margin-left: 3rem !important; } + .mx-xl-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xl-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xl-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 87.5rem) { .m-xl-9 { margin: 3.5rem !important; } + .mt-xl-9 { margin-top: 3.5rem !important; } + .mr-xl-9 { margin-right: 3.5rem !important; } + .mb-xl-9 { margin-bottom: 3.5rem !important; } + .ml-xl-9 { margin-left: 3.5rem !important; } + .mx-xl-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xl-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xl-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-10 { margin: 4rem !important; } + .mt-xl-10 { margin-top: 4rem !important; } + .mr-xl-10 { margin-right: 4rem !important; } + .mb-xl-10 { margin-bottom: 4rem !important; } + .ml-xl-10 { margin-left: 4rem !important; } + .mx-xl-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xl-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xl-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +.p-0 { padding: 0 !important; } + +.pt-0 { padding-top: 0 !important; } + +.pr-0 { padding-right: 0 !important; } + +.pb-0 { padding-bottom: 0 !important; } + +.pl-0 { padding-left: 0 !important; } + +.px-0 { padding-right: 0 !important; padding-left: 0 !important; } + +.py-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + +.p-1 { padding: 0.25rem !important; } + +.pt-1 { padding-top: 0.25rem !important; } + +.pr-1 { padding-right: 0.25rem !important; } + +.pb-1 { padding-bottom: 0.25rem !important; } + +.pl-1 { padding-left: 0.25rem !important; } + +.px-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + +.py-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + +.p-2 { padding: 0.5rem !important; } + +.pt-2 { padding-top: 0.5rem !important; } + +.pr-2 { padding-right: 0.5rem !important; } + +.pb-2 { padding-bottom: 0.5rem !important; } + +.pl-2 { padding-left: 0.5rem !important; } + +.px-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + +.py-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + +.p-3 { padding: 0.75rem !important; } + +.pt-3 { padding-top: 0.75rem !important; } + +.pr-3 { padding-right: 0.75rem !important; } + +.pb-3 { padding-bottom: 0.75rem !important; } + +.pl-3 { padding-left: 0.75rem !important; } + +.px-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + +.py-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + +.p-4 { padding: 1rem !important; } + +.pt-4 { padding-top: 1rem !important; } + +.pr-4 { padding-right: 1rem !important; } + +.pb-4 { padding-bottom: 1rem !important; } + +.pl-4 { padding-left: 1rem !important; } + +.px-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + +.py-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + +.p-5 { padding: 1.5rem !important; } + +.pt-5 { padding-top: 1.5rem !important; } + +.pr-5 { padding-right: 1.5rem !important; } + +.pb-5 { padding-bottom: 1.5rem !important; } + +.pl-5 { padding-left: 1.5rem !important; } + +.px-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + +.py-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + +.p-6 { padding: 2rem !important; } + +.pt-6 { padding-top: 2rem !important; } + +.pr-6 { padding-right: 2rem !important; } + +.pb-6 { padding-bottom: 2rem !important; } + +.pl-6 { padding-left: 2rem !important; } + +.px-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + +.py-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + +.p-7 { padding: 2.5rem !important; } + +.pt-7 { padding-top: 2.5rem !important; } + +.pr-7 { padding-right: 2.5rem !important; } + +.pb-7 { padding-bottom: 2.5rem !important; } + +.pl-7 { padding-left: 2.5rem !important; } + +.px-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + +.py-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + +.p-8 { padding: 3rem !important; } + +.pt-8 { padding-top: 3rem !important; } + +.pr-8 { padding-right: 3rem !important; } + +.pb-8 { padding-bottom: 3rem !important; } + +.pl-8 { padding-left: 3rem !important; } + +.px-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + +.py-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + +.p-9 { padding: 3.5rem !important; } + +.pt-9 { padding-top: 3.5rem !important; } + +.pr-9 { padding-right: 3.5rem !important; } + +.pb-9 { padding-bottom: 3.5rem !important; } + +.pl-9 { padding-left: 3.5rem !important; } + +.px-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + +.py-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + +.p-10 { padding: 4rem !important; } + +.pt-10 { padding-top: 4rem !important; } + +.pr-10 { padding-right: 4rem !important; } + +.pb-10 { padding-bottom: 4rem !important; } + +.pl-10 { padding-left: 4rem !important; } + +.px-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + +.py-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } + +@media (min-width: 20rem) { .p-xs-0 { padding: 0 !important; } + .pt-xs-0 { padding-top: 0 !important; } + .pr-xs-0 { padding-right: 0 !important; } + .pb-xs-0 { padding-bottom: 0 !important; } + .pl-xs-0 { padding-left: 0 !important; } + .px-xs-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xs-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xs-1 { padding: 0.25rem !important; } + .pt-xs-1 { padding-top: 0.25rem !important; } + .pr-xs-1 { padding-right: 0.25rem !important; } + .pb-xs-1 { padding-bottom: 0.25rem !important; } + .pl-xs-1 { padding-left: 0.25rem !important; } + .px-xs-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xs-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xs-2 { padding: 0.5rem !important; } + .pt-xs-2 { padding-top: 0.5rem !important; } + .pr-xs-2 { padding-right: 0.5rem !important; } + .pb-xs-2 { padding-bottom: 0.5rem !important; } + .pl-xs-2 { padding-left: 0.5rem !important; } + .px-xs-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xs-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xs-3 { padding: 0.75rem !important; } + .pt-xs-3 { padding-top: 0.75rem !important; } + .pr-xs-3 { padding-right: 0.75rem !important; } + .pb-xs-3 { padding-bottom: 0.75rem !important; } + .pl-xs-3 { padding-left: 0.75rem !important; } + .px-xs-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xs-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xs-4 { padding: 1rem !important; } + .pt-xs-4 { padding-top: 1rem !important; } + .pr-xs-4 { padding-right: 1rem !important; } + .pb-xs-4 { padding-bottom: 1rem !important; } + .pl-xs-4 { padding-left: 1rem !important; } + .px-xs-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xs-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xs-5 { padding: 1.5rem !important; } + .pt-xs-5 { padding-top: 1.5rem !important; } + .pr-xs-5 { padding-right: 1.5rem !important; } + .pb-xs-5 { padding-bottom: 1.5rem !important; } + .pl-xs-5 { padding-left: 1.5rem !important; } + .px-xs-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xs-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xs-6 { padding: 2rem !important; } + .pt-xs-6 { padding-top: 2rem !important; } + .pr-xs-6 { padding-right: 2rem !important; } + .pb-xs-6 { padding-bottom: 2rem !important; } + .pl-xs-6 { padding-left: 2rem !important; } + .px-xs-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xs-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xs-7 { padding: 2.5rem !important; } + .pt-xs-7 { padding-top: 2.5rem !important; } + .pr-xs-7 { padding-right: 2.5rem !important; } + .pb-xs-7 { padding-bottom: 2.5rem !important; } + .pl-xs-7 { padding-left: 2.5rem !important; } + .px-xs-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xs-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xs-8 { padding: 3rem !important; } + .pt-xs-8 { padding-top: 3rem !important; } + .pr-xs-8 { padding-right: 3rem !important; } + .pb-xs-8 { padding-bottom: 3rem !important; } + .pl-xs-8 { padding-left: 3rem !important; } + .px-xs-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xs-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xs-9 { padding: 3.5rem !important; } + .pt-xs-9 { padding-top: 3.5rem !important; } + .pr-xs-9 { padding-right: 3.5rem !important; } + .pb-xs-9 { padding-bottom: 3.5rem !important; } + .pl-xs-9 { padding-left: 3.5rem !important; } + .px-xs-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xs-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xs-10 { padding: 4rem !important; } + .pt-xs-10 { padding-top: 4rem !important; } + .pr-xs-10 { padding-right: 4rem !important; } + .pb-xs-10 { padding-bottom: 4rem !important; } + .pl-xs-10 { padding-left: 4rem !important; } + .px-xs-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xs-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 31.25rem) { .p-sm-0 { padding: 0 !important; } + .pt-sm-0 { padding-top: 0 !important; } + .pr-sm-0 { padding-right: 0 !important; } + .pb-sm-0 { padding-bottom: 0 !important; } + .pl-sm-0 { padding-left: 0 !important; } + .px-sm-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-sm-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-sm-1 { padding: 0.25rem !important; } + .pt-sm-1 { padding-top: 0.25rem !important; } + .pr-sm-1 { padding-right: 0.25rem !important; } + .pb-sm-1 { padding-bottom: 0.25rem !important; } + .pl-sm-1 { padding-left: 0.25rem !important; } + .px-sm-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-sm-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-sm-2 { padding: 0.5rem !important; } + .pt-sm-2 { padding-top: 0.5rem !important; } + .pr-sm-2 { padding-right: 0.5rem !important; } + .pb-sm-2 { padding-bottom: 0.5rem !important; } + .pl-sm-2 { padding-left: 0.5rem !important; } + .px-sm-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-sm-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-sm-3 { padding: 0.75rem !important; } + .pt-sm-3 { padding-top: 0.75rem !important; } + .pr-sm-3 { padding-right: 0.75rem !important; } + .pb-sm-3 { padding-bottom: 0.75rem !important; } + .pl-sm-3 { padding-left: 0.75rem !important; } + .px-sm-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-sm-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-sm-4 { padding: 1rem !important; } + .pt-sm-4 { padding-top: 1rem !important; } + .pr-sm-4 { padding-right: 1rem !important; } + .pb-sm-4 { padding-bottom: 1rem !important; } + .pl-sm-4 { padding-left: 1rem !important; } + .px-sm-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-sm-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-sm-5 { padding: 1.5rem !important; } + .pt-sm-5 { padding-top: 1.5rem !important; } + .pr-sm-5 { padding-right: 1.5rem !important; } + .pb-sm-5 { padding-bottom: 1.5rem !important; } + .pl-sm-5 { padding-left: 1.5rem !important; } + .px-sm-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-sm-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-sm-6 { padding: 2rem !important; } + .pt-sm-6 { padding-top: 2rem !important; } + .pr-sm-6 { padding-right: 2rem !important; } + .pb-sm-6 { padding-bottom: 2rem !important; } + .pl-sm-6 { padding-left: 2rem !important; } + .px-sm-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-sm-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-sm-7 { padding: 2.5rem !important; } + .pt-sm-7 { padding-top: 2.5rem !important; } + .pr-sm-7 { padding-right: 2.5rem !important; } + .pb-sm-7 { padding-bottom: 2.5rem !important; } + .pl-sm-7 { padding-left: 2.5rem !important; } + .px-sm-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-sm-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-sm-8 { padding: 3rem !important; } + .pt-sm-8 { padding-top: 3rem !important; } + .pr-sm-8 { padding-right: 3rem !important; } + .pb-sm-8 { padding-bottom: 3rem !important; } + .pl-sm-8 { padding-left: 3rem !important; } + .px-sm-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-sm-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-sm-9 { padding: 3.5rem !important; } + .pt-sm-9 { padding-top: 3.5rem !important; } + .pr-sm-9 { padding-right: 3.5rem !important; } + .pb-sm-9 { padding-bottom: 3.5rem !important; } + .pl-sm-9 { padding-left: 3.5rem !important; } + .px-sm-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-sm-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-sm-10 { padding: 4rem !important; } + .pt-sm-10 { padding-top: 4rem !important; } + .pr-sm-10 { padding-right: 4rem !important; } + .pb-sm-10 { padding-bottom: 4rem !important; } + .pl-sm-10 { padding-left: 4rem !important; } + .px-sm-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-sm-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 50rem) { .p-md-0 { padding: 0 !important; } + .pt-md-0 { padding-top: 0 !important; } + .pr-md-0 { padding-right: 0 !important; } + .pb-md-0 { padding-bottom: 0 !important; } + .pl-md-0 { padding-left: 0 !important; } + .px-md-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-md-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-md-1 { padding: 0.25rem !important; } + .pt-md-1 { padding-top: 0.25rem !important; } + .pr-md-1 { padding-right: 0.25rem !important; } + .pb-md-1 { padding-bottom: 0.25rem !important; } + .pl-md-1 { padding-left: 0.25rem !important; } + .px-md-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-md-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-md-2 { padding: 0.5rem !important; } + .pt-md-2 { padding-top: 0.5rem !important; } + .pr-md-2 { padding-right: 0.5rem !important; } + .pb-md-2 { padding-bottom: 0.5rem !important; } + .pl-md-2 { padding-left: 0.5rem !important; } + .px-md-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-md-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-md-3 { padding: 0.75rem !important; } + .pt-md-3 { padding-top: 0.75rem !important; } + .pr-md-3 { padding-right: 0.75rem !important; } + .pb-md-3 { padding-bottom: 0.75rem !important; } + .pl-md-3 { padding-left: 0.75rem !important; } + .px-md-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-md-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-md-4 { padding: 1rem !important; } + .pt-md-4 { padding-top: 1rem !important; } + .pr-md-4 { padding-right: 1rem !important; } + .pb-md-4 { padding-bottom: 1rem !important; } + .pl-md-4 { padding-left: 1rem !important; } + .px-md-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-md-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-md-5 { padding: 1.5rem !important; } + .pt-md-5 { padding-top: 1.5rem !important; } + .pr-md-5 { padding-right: 1.5rem !important; } + .pb-md-5 { padding-bottom: 1.5rem !important; } + .pl-md-5 { padding-left: 1.5rem !important; } + .px-md-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-md-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-md-6 { padding: 2rem !important; } + .pt-md-6 { padding-top: 2rem !important; } + .pr-md-6 { padding-right: 2rem !important; } + .pb-md-6 { padding-bottom: 2rem !important; } + .pl-md-6 { padding-left: 2rem !important; } + .px-md-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-md-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-md-7 { padding: 2.5rem !important; } + .pt-md-7 { padding-top: 2.5rem !important; } + .pr-md-7 { padding-right: 2.5rem !important; } + .pb-md-7 { padding-bottom: 2.5rem !important; } + .pl-md-7 { padding-left: 2.5rem !important; } + .px-md-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-md-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-md-8 { padding: 3rem !important; } + .pt-md-8 { padding-top: 3rem !important; } + .pr-md-8 { padding-right: 3rem !important; } + .pb-md-8 { padding-bottom: 3rem !important; } + .pl-md-8 { padding-left: 3rem !important; } + .px-md-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-md-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-md-9 { padding: 3.5rem !important; } + .pt-md-9 { padding-top: 3.5rem !important; } + .pr-md-9 { padding-right: 3.5rem !important; } + .pb-md-9 { padding-bottom: 3.5rem !important; } + .pl-md-9 { padding-left: 3.5rem !important; } + .px-md-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-md-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-md-10 { padding: 4rem !important; } + .pt-md-10 { padding-top: 4rem !important; } + .pr-md-10 { padding-right: 4rem !important; } + .pb-md-10 { padding-bottom: 4rem !important; } + .pl-md-10 { padding-left: 4rem !important; } + .px-md-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-md-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 66.5rem) { .p-lg-0 { padding: 0 !important; } + .pt-lg-0 { padding-top: 0 !important; } + .pr-lg-0 { padding-right: 0 !important; } + .pb-lg-0 { padding-bottom: 0 !important; } + .pl-lg-0 { padding-left: 0 !important; } + .px-lg-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-lg-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-lg-1 { padding: 0.25rem !important; } + .pt-lg-1 { padding-top: 0.25rem !important; } + .pr-lg-1 { padding-right: 0.25rem !important; } + .pb-lg-1 { padding-bottom: 0.25rem !important; } + .pl-lg-1 { padding-left: 0.25rem !important; } + .px-lg-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-lg-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-lg-2 { padding: 0.5rem !important; } + .pt-lg-2 { padding-top: 0.5rem !important; } + .pr-lg-2 { padding-right: 0.5rem !important; } + .pb-lg-2 { padding-bottom: 0.5rem !important; } + .pl-lg-2 { padding-left: 0.5rem !important; } + .px-lg-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-lg-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-lg-3 { padding: 0.75rem !important; } + .pt-lg-3 { padding-top: 0.75rem !important; } + .pr-lg-3 { padding-right: 0.75rem !important; } + .pb-lg-3 { padding-bottom: 0.75rem !important; } + .pl-lg-3 { padding-left: 0.75rem !important; } + .px-lg-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-lg-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-lg-4 { padding: 1rem !important; } + .pt-lg-4 { padding-top: 1rem !important; } + .pr-lg-4 { padding-right: 1rem !important; } + .pb-lg-4 { padding-bottom: 1rem !important; } + .pl-lg-4 { padding-left: 1rem !important; } + .px-lg-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-lg-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-lg-5 { padding: 1.5rem !important; } + .pt-lg-5 { padding-top: 1.5rem !important; } + .pr-lg-5 { padding-right: 1.5rem !important; } + .pb-lg-5 { padding-bottom: 1.5rem !important; } + .pl-lg-5 { padding-left: 1.5rem !important; } + .px-lg-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-lg-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-lg-6 { padding: 2rem !important; } + .pt-lg-6 { padding-top: 2rem !important; } + .pr-lg-6 { padding-right: 2rem !important; } + .pb-lg-6 { padding-bottom: 2rem !important; } + .pl-lg-6 { padding-left: 2rem !important; } + .px-lg-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-lg-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-lg-7 { padding: 2.5rem !important; } + .pt-lg-7 { padding-top: 2.5rem !important; } + .pr-lg-7 { padding-right: 2.5rem !important; } + .pb-lg-7 { padding-bottom: 2.5rem !important; } + .pl-lg-7 { padding-left: 2.5rem !important; } + .px-lg-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-lg-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-lg-8 { padding: 3rem !important; } + .pt-lg-8 { padding-top: 3rem !important; } + .pr-lg-8 { padding-right: 3rem !important; } + .pb-lg-8 { padding-bottom: 3rem !important; } + .pl-lg-8 { padding-left: 3rem !important; } + .px-lg-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-lg-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-lg-9 { padding: 3.5rem !important; } + .pt-lg-9 { padding-top: 3.5rem !important; } + .pr-lg-9 { padding-right: 3.5rem !important; } + .pb-lg-9 { padding-bottom: 3.5rem !important; } + .pl-lg-9 { padding-left: 3.5rem !important; } + .px-lg-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-lg-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-lg-10 { padding: 4rem !important; } + .pt-lg-10 { padding-top: 4rem !important; } + .pr-lg-10 { padding-right: 4rem !important; } + .pb-lg-10 { padding-bottom: 4rem !important; } + .pl-lg-10 { padding-left: 4rem !important; } + .px-lg-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-lg-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 87.5rem) { .p-xl-0 { padding: 0 !important; } + .pt-xl-0 { padding-top: 0 !important; } + .pr-xl-0 { padding-right: 0 !important; } + .pb-xl-0 { padding-bottom: 0 !important; } + .pl-xl-0 { padding-left: 0 !important; } + .px-xl-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xl-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xl-1 { padding: 0.25rem !important; } + .pt-xl-1 { padding-top: 0.25rem !important; } + .pr-xl-1 { padding-right: 0.25rem !important; } + .pb-xl-1 { padding-bottom: 0.25rem !important; } + .pl-xl-1 { padding-left: 0.25rem !important; } + .px-xl-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xl-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xl-2 { padding: 0.5rem !important; } + .pt-xl-2 { padding-top: 0.5rem !important; } + .pr-xl-2 { padding-right: 0.5rem !important; } + .pb-xl-2 { padding-bottom: 0.5rem !important; } + .pl-xl-2 { padding-left: 0.5rem !important; } + .px-xl-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xl-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xl-3 { padding: 0.75rem !important; } + .pt-xl-3 { padding-top: 0.75rem !important; } + .pr-xl-3 { padding-right: 0.75rem !important; } + .pb-xl-3 { padding-bottom: 0.75rem !important; } + .pl-xl-3 { padding-left: 0.75rem !important; } + .px-xl-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xl-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xl-4 { padding: 1rem !important; } + .pt-xl-4 { padding-top: 1rem !important; } + .pr-xl-4 { padding-right: 1rem !important; } + .pb-xl-4 { padding-bottom: 1rem !important; } + .pl-xl-4 { padding-left: 1rem !important; } + .px-xl-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xl-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xl-5 { padding: 1.5rem !important; } + .pt-xl-5 { padding-top: 1.5rem !important; } + .pr-xl-5 { padding-right: 1.5rem !important; } + .pb-xl-5 { padding-bottom: 1.5rem !important; } + .pl-xl-5 { padding-left: 1.5rem !important; } + .px-xl-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xl-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xl-6 { padding: 2rem !important; } + .pt-xl-6 { padding-top: 2rem !important; } + .pr-xl-6 { padding-right: 2rem !important; } + .pb-xl-6 { padding-bottom: 2rem !important; } + .pl-xl-6 { padding-left: 2rem !important; } + .px-xl-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xl-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xl-7 { padding: 2.5rem !important; } + .pt-xl-7 { padding-top: 2.5rem !important; } + .pr-xl-7 { padding-right: 2.5rem !important; } + .pb-xl-7 { padding-bottom: 2.5rem !important; } + .pl-xl-7 { padding-left: 2.5rem !important; } + .px-xl-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xl-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xl-8 { padding: 3rem !important; } + .pt-xl-8 { padding-top: 3rem !important; } + .pr-xl-8 { padding-right: 3rem !important; } + .pb-xl-8 { padding-bottom: 3rem !important; } + .pl-xl-8 { padding-left: 3rem !important; } + .px-xl-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xl-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xl-9 { padding: 3.5rem !important; } + .pt-xl-9 { padding-top: 3.5rem !important; } + .pr-xl-9 { padding-right: 3.5rem !important; } + .pb-xl-9 { padding-bottom: 3.5rem !important; } + .pl-xl-9 { padding-left: 3.5rem !important; } + .px-xl-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xl-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xl-10 { padding: 4rem !important; } + .pt-xl-10 { padding-top: 4rem !important; } + .pr-xl-10 { padding-right: 4rem !important; } + .pb-xl-10 { padding-bottom: 4rem !important; } + .pl-xl-10 { padding-left: 4rem !important; } + .px-xl-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xl-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media print { .site-footer, .site-button, #edit-this-page, #back-to-top, .site-nav, .main-header { display: none !important; } + .side-bar { width: 100%; height: auto; border-right: 0 !important; } + .site-header { border-bottom: 1px solid #44434d; } + .site-title { font-size: 1rem !important; font-weight: 700 !important; } + .text-small { font-size: 8pt !important; } + pre.highlight { border: 1px solid #44434d; } + .main { max-width: none; margin-left: 0; } } +a.skip-to-main { left: -999px; position: absolute; top: auto; width: 1px; height: 1px; overflow: hidden; z-index: -999; } + +a.skip-to-main:focus, a.skip-to-main:active { color: #2c84fa; background-color: #27262b; left: auto; top: auto; width: 30%; height: auto; overflow: auto; margin: 10px 35%; padding: 5px; border-radius: 15px; border: 4px solid #264caf; text-align: center; font-size: 1.2em; z-index: 999; } + +div.opaque { background-color: #27262b; } + +p.note, blockquote.note { background: #2ce5b5; border-left: 4px solid #f5f7f9; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note > .note-title, blockquote.note > .note-title { color: #f5f7f9; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.note-title, blockquote.note-title { background: #2ce5b5; border-left: 4px solid #f5f7f9; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note-title > p:first-child, blockquote.note-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #f5f7f9; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.note { margin-left: 0; margin-right: 0; } +blockquote.note > p:first-child { margin-top: 0; } +blockquote.note > p:last-child { margin-bottom: 0; } + +blockquote.note-title { margin-left: 0; margin-right: 0; } +blockquote.note-title > p:nth-child(2) { margin-top: 0; } +blockquote.note-title > p:last-child { margin-bottom: 0; } + +p.warning, blockquote.warning { background: rgba(221, 46, 46, 0.2); border-left: 4px solid #f77e7e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning::before, blockquote.warning::before { color: #f77e7e; content: "⚠️ Warning ⚠️"; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } +p.warning > .warning-title, blockquote.warning > .warning-title { color: #f77e7e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.warning-title, blockquote.warning-title { background: rgba(221, 46, 46, 0.2); border-left: 4px solid #f77e7e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning-title > p:first-child, blockquote.warning-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #f77e7e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.warning { margin-left: 0; margin-right: 0; } +blockquote.warning > p:first-child { margin-top: 0; } +blockquote.warning > p:last-child { margin-bottom: 0; } + +blockquote.warning-title { margin-left: 0; margin-right: 0; } +blockquote.warning-title > p:nth-child(2) { margin-top: 0; } +blockquote.warning-title > p:last-child { margin-bottom: 0; } + +p.tip, blockquote.tip { background: rgba(2, 110, 87, 0.2); border-left: 4px solid #41d693; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip > .tip-title, blockquote.tip > .tip-title { color: #41d693; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.tip-title, blockquote.tip-title { background: rgba(2, 110, 87, 0.2); border-left: 4px solid #41d693; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip-title > p:first-child, blockquote.tip-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #41d693; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.tip { margin-left: 0; margin-right: 0; } +blockquote.tip > p:first-child { margin-top: 0; } +blockquote.tip > p:last-child { margin-bottom: 0; } + +blockquote.tip-title { margin-left: 0; margin-right: 0; } +blockquote.tip-title > p:nth-child(2) { margin-top: 0; } +blockquote.tip-title > p:last-child { margin-bottom: 0; } + +p.fubar, blockquote.fubar { background: rgba(56, 24, 133, 0.2); border-left: 4px solid #7253ed; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar > .fubar-title, blockquote.fubar > .fubar-title { color: #7253ed; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.fubar-title, blockquote.fubar-title { background: rgba(56, 24, 133, 0.2); border-left: 4px solid #7253ed; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar-title > p:first-child, blockquote.fubar-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #7253ed; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.fubar { margin-left: 0; margin-right: 0; } +blockquote.fubar > p:first-child { margin-top: 0; } +blockquote.fubar > p:last-child { margin-bottom: 0; } + +blockquote.fubar-title { margin-left: 0; margin-right: 0; } +blockquote.fubar-title > p:nth-child(2) { margin-top: 0; } +blockquote.fubar-title > p:last-child { margin-bottom: 0; } + +@font-face { font-family: GalanoGrotesque-ExtraLight; src: local(GalanoGrotesque-ExtraLight), url("../fonts/Rene Bieder - Galano Grotesque ExtraLight.otf"); } +@font-face { font-family: GalanoGrotesque; src: local(GalanoGrotesque-Regular), url("../fonts/Rene Bieder - Galano Grotesque.otf"); } +@font-face { font-family: GalanoGrotesque-Bold; src: local(GalanoGrotesque-Bold), url("../fonts/Rene Bieder - Galano Grotesque Bold.otf"); } +@font-face { font-family: GalanoGrotesque-Medium; src: local(Rene-Bieder-Galano-Grotesque-Medium), url("../fonts/Rene Bieder - Galano Grotesque Medium.otf"); } +html { scroll-padding-top: 90px; } + +body { font-family: GalanoGrotesque; color: #000000; } +body .body-wrapper, body footer { display: flex; } +body .body-wrapper .main, body .body-wrapper .footer-main, body footer .main, body footer .footer-main { width: 75%; max-width: 75%; margin-left: 0; float: right; position: relative; } +body .body-wrapper .main .main-content-wrap, body .body-wrapper .footer-main .main-content-wrap, body footer .main .main-content-wrap, body footer .footer-main .main-content-wrap { max-width: 1300px; padding: 80px; } +body .body-wrapper .side-bar, body .body-wrapper .footer-sidebar, body footer .side-bar, body footer .footer-sidebar { width: 25%; max-width: 25%; position: relative; height: auto; background-color: #f5f7f9; } +body a { color: #279890; } +body p, body li { font-family: GalanoGrotesque; font-size: 1em; } +body h1 { margin-top: 40px !important; font-family: GalanoGrotesque-Medium; font-size: 42px !important; color: #000000; } +body h2 { margin-top: 40px !important; font-size: 28px !important; font-family: GalanoGrotesque-Medium; } +body h2#table-of-contents { font-family: GalanoGrotesque-Medium; color: #ffffff; font-size: 14px !important; } +body h3 { margin-top: 25px !important; font-size: 18px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body h4 { margin-top: 25px !important; font-size: 16px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body .text-delta { text-transform: none; width: 100%; display: block; background-color: #279890; color: #ffffff; border-radius: 2px 2px 0 0; font-size: 14px !important; padding: 10px 15px 8px 15px; } +body .text-delta + ul { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body .text-delta + ul li { text-decoration: underline; } +body .text-delta + ul li::before { content: none; } +body .text-delta + ul li a:hover { background-image: none; } +body .text-delta + ul li a { white-space: normal; } +body #markdown-toc { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body #markdown-toc li::before { display: none; } +body #markdown-toc li a { color: #279890; font-size: 1em; background-image: none; white-space: normal; } +body #markdown-toc li a:hover { background-image: none; color: #279890; } +body #markdown-toc li ol, body #markdown-toc li ul { border: none; } +body #markdown-toc li ol li a, body #markdown-toc li ul li a { padding-bottom: 0; } +body div.highlighter-rouge pre.highlight { padding: 10px; } +body .highlight code { background-color: transparent; } +body .btn { padding: 6px 14px; color: #2f3433; font-family: GalanoGrotesque; border-radius: 4px; background: #ffffff; border: solid 2px #2f3433; box-shadow: none; } +body .btn:hover { border-color: #279890; color: #279890; background: #ffffff; } +body .btn-green { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; background-color: #279890; border: 2px solid #279890; } +body .btn-green:hover { background-color: #ffffff; color: #279890; } +body .main-content h1:hover a.anchor-heading, body .main-content h2:hover a.anchor-heading, body .main-content h3:hover a.anchor-heading, body .main-content h4:hover a.anchor-heading, body .main-content h5:hover a.anchor-heading, body .main-content h6:hover a.anchor-heading { left: -1.7rem; background-image: url("../../assets/icons/anchor.svg"); width: 24px; height: 24px; background-repeat: no-repeat; background-size: contain; background-position: center center; } +body .main-content h1:hover a.anchor-heading svg, body .main-content h2:hover a.anchor-heading svg, body .main-content h3:hover a.anchor-heading svg, body .main-content h4:hover a.anchor-heading svg, body .main-content h5:hover a.anchor-heading svg, body .main-content h6:hover a.anchor-heading svg { display: none; } +body .main-content h1:hover a.anchor-heading:hover, body .main-content h2:hover a.anchor-heading:hover, body .main-content h3:hover a.anchor-heading:hover, body .main-content h4:hover a.anchor-heading:hover, body .main-content h5:hover a.anchor-heading:hover, body .main-content h6:hover a.anchor-heading:hover { background-image: url("../../assets/icons/anchor green.svg"); } +body .main-content .toc-block { display: table-cell; } + +.main .main-header, .side-bar .site-header { background-color: white; height: 80px; max-height: 80px; min-height: 80px; border: 0; position: fixed; z-index: 2; } + +.main .main-header { width: 75%; } +.main .main-header nav.aux-nav { width: 100%; padding: 0; max-width: 1080px; } +.main .main-header nav.aux-nav .aux-nav-list { justify-content: flex-end; padding: 20px 80px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #000000; margin: 0 4px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button:hover { background-image: none; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item:hover a, .main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.active a { color: #279890; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; margin: 0 0 0 20px; padding: 18px; border-radius: 6px; background-color: #f24886; border: 2px solid #f24886; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button:hover { background-color: #b62d55; border: 2px solid #b62d55; color: #ffffff; } + +.side-bar:after { content: ' '; position: fixed; width: 100vw; top: 80px; box-shadow: 0 0 6px 2px rgba(0, 0, 0, 0.23); left: 0; right: 0; z-index: 0; } + +.side-bar .site-header { width: 25%; justify-content: flex-end; top: 0; } +.side-bar .site-header a.site-title { max-width: 290px; padding: 0; } +.side-bar .site-header a.site-title .site-logo { width: 150px; height: inherit; } +.side-bar .site-header a.site-title:hover { background-image: none; background-color: transparent; } + +.site-logo { width: 135px; height: 30px; } + +.side-bar { padding: 80px 0; border: 0; display: block; z-index: 1; } +.side-bar nav.site-nav, .side-bar nav.aux-nav { padding: 0; width: 100%; display: initial; } +.side-bar nav.site-nav .nav-list, .side-bar nav.site-nav .aux-nav-list, .side-bar nav.aux-nav .nav-list, .side-bar nav.aux-nav .aux-nav-list { width: 100%; padding: 0; margin-top: 30px; } +.side-bar nav.site-nav .nav-list .nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item { padding: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .nav-list .nav-list-item:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active:hover { background-color: transparent; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li { padding-right: 0; padding-left: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.header, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .nav-list .nav-list-item.header, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header { padding: 12px 40px 12px calc(100% - 290px); color: #279890; font-family: GalanoGrotesque-Bold; font-size: 18px !important; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:hover { background-color: transparent; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type) { padding-top: 36px; border-top: 1px solid #dadada; margin-top: 20px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link { padding: 12px 40px 12px calc(100% - 290px); font-size: 14px !important; color: #576489; font-family: GalanoGrotesque; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover { background-color: #ffffff; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active { font-family: GalanoGrotesque-Medium !important; color: #279890 !important; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander { right: 40px; width: 22px; height: 22px; padding: 0; margin: 14px 0 0 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg { color: #000000; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover { background-color: none; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list { margin-top: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item { margin-left: 20px; font-size: 13px !important; letter-spacing: 0.05px; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item { margin-left: 30px; color: #949494; font-size: 12px !important; } +.side-bar nav.site-nav .nav-version, .side-bar nav.aux-nav .nav-version { display: flex; align-items: center; justify-content: center; padding: 12px 40px 12px calc(100% - 290px); text-transform: none; } +.side-bar nav.site-nav .nav-version select, .side-bar nav.aux-nav .nav-version select { height: 2em; margin: 0 0 0 10px; flex-grow: 1; } +.side-bar nav.site-nav .search, .side-bar nav.aux-nav .search { margin-top: 35px; margin-left: calc(100% - 290px); margin-right: 40px; max-width: 250px !important; max-height: 40px !important; width: 250px !important; z-index: 1; } +.side-bar nav.site-nav .search .search-input, .side-bar nav.aux-nav .search .search-input { background-color: #f5f7f9; border: 1px solid #dadada; border-radius: 6px; color: #6e7171; font-family: GalanoGrotesque-Medium; } +.side-bar nav.site-nav .search .search-input:focus, .side-bar nav.site-nav .search .search-input:active, .side-bar nav.aux-nav .search .search-input:focus, .side-bar nav.aux-nav .search .search-input:active { background-color: #f5f7f9; color: #6e7171; } +.side-bar nav.site-nav .search .search-input .search-icon, .side-bar nav.aux-nav .search .search-input .search-icon { color: #949494; } +.side-bar nav.site-nav .search .search-input:focus + .search-label .search-icon, .side-bar nav.aux-nav .search .search-input:focus + .search-label .search-icon { color: #949494; } +.side-bar nav.site-nav .mobile-menu, .side-bar nav.aux-nav .mobile-menu { display: none; } + +.label { text-transform: none !important; margin-left: 0 !important; margin-right: 0 !important; } + +.search-active .main { position: inherit; right: inherit; left: inherit; } + +.search-overlay { display: none; } + +.search-result-doc .search-result-icon { color: #279890; } + +footer { height: 360px; } +footer .footer-main { padding: 80px; } +footer .footer-main ul { list-style: none; padding: 0; margin: 0; } +footer .footer-main ul li { display: inline-block; padding: 0; margin: 0; } +footer .footer-main .row { display: flex; } +footer .footer-main .row .left { width: 70%; } +footer .footer-main .row .right { width: 30%; display: flex; justify-content: flex-end; } +footer .footer-main .row .site-title { padding: 0; margin-bottom: 30px; } +footer .footer-main .row .site-title .site-logo { height: 45px; } +footer .footer-main .row .footer-links li { margin-right: 40px; } +footer .footer-main .row .footer-links li a { padding: 0; color: #000000; font-size: 18px; } +footer .footer-main .row .footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a, footer .footer-main .row .bottom-footer-links li a { padding: 0 15px; font-size: 16px; color: #949494; } +footer .footer-main .row .footer-social li a:last-of-type, footer .footer-main .row .bottom-footer-links li a:last-of-type { padding-right: 0; } +footer .footer-main .row .footer-social li a:hover, footer .footer-main .row .bottom-footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a img.hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.no-hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.hover { display: block; } +footer .footer-main .top-border { padding-top: 30px; margin-top: 30px; border-top: 1px solid #dadada; } +footer .footer-main a:hover { background-color: transparent !important; background-image: none !important; } +footer .footer-sidebar { z-index: -1; } + +@media (max-width: 799px) { body { padding-bottom: 0; } + body .body-wrapper { display: block; } + body .body-wrapper .side-bar { width: 100%; max-width: 100%; padding: 0; } + body .body-wrapper .side-bar .site-header { width: 100%; flex-direction: row-reverse; justify-content: flex-end; } + body .body-wrapper .side-bar .site-header .site-button { padding: 11px 30px 0; width: 24px; height: 24px; background-repeat: no-repeat; background-size: auto; background-position: center center; background-image: url("../../assets/icons/menu.svg"); } + body .body-wrapper .side-bar .site-header .site-button:hover, body .body-wrapper .side-bar .site-header .site-button:active, body .body-wrapper .side-bar .site-header .site-button:focus { background-color: transparent; } + body .body-wrapper .side-bar .site-header .site-button.nav-open { background-image: url("../../assets/icons/close.svg"); } + body .body-wrapper .side-bar .site-header .site-button svg { display: none; } + body .body-wrapper .side-bar nav { display: none; overflow: scroll; background: white; } + body .body-wrapper .side-bar nav.nav-open { display: block; position: fixed; top: 0; height: 100vh; } + body .body-wrapper .side-bar nav.nav-open .search { position: relative; margin-top: 120px; margin-left: 30px; margin-right: 30px; padding: 0; } + body .body-wrapper .side-bar nav.nav-open .search .search-results { display: block !important; max-height: 60vh !important; } + body .body-wrapper .side-bar nav.nav-open .search .search-input { background-color: #ffffff; } + body .body-wrapper .side-bar nav.nav-open .search .search-input-wrap { box-shadow: none; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item { padding-left: 30px !important; padding-right: 30px !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a { padding-left: 0 !important; padding-right: 0 !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a.active { background: unset; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list { margin-top: 0; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list a.nav-list-link, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list a.nav-list-link { text-indent: 0; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu { display: block; border-top: 1px solid #dadada; margin-top: 18px; padding-bottom: 80px; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav { display: block; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list { flex-direction: column; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list a.site-button { font-size: 18px !important; color: #000000; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list .button .site-button { border-radius: 6px; border: solid 2px #279890; background-color: #ffffff; color: #279890; padding: 10px 20px !important; display: inline-block; margin-top: 10px; } + body .body-wrapper .main { width: 100%; max-width: 100%; float: none; } + body .body-wrapper .main .main-header { display: none; } + body .body-wrapper .main .main-content a { white-space: normal; } + body .body-wrapper .main .main-content-wrap { padding: 80px 30px; } + body footer { border-top: 1px solid #dadada; height: auto; } + body footer .footer-sidebar { display: none; } + body footer .footer-main { width: 100%; max-width: 100%; padding: 60px 30px; float: none; } + body footer .footer-main .row { flex-direction: column; } + body footer .footer-main .row .right, body footer .footer-main .row .left { width: 100%; display: block; } + body footer .footer-main .row .site-title { margin-bottom: 40px; } + body footer .footer-main .row .footer-links { display: flex; flex-wrap: wrap; margin-bottom: 20px; } + body footer .footer-main .row .footer-links li { margin: 0 0 30px; width: 50%; } + body footer .footer-main .row .footer-social { width: 100%; display: flex; justify-content: space-between; } + body footer .footer-main .row .footer-social li a { padding: 0; } + body footer .footer-main .row .footer-social li a img { width: 34px; height: 34px; } + body footer .footer-main .row .bottom-footer-links li { display: block; margin-top: 30px; } + body footer .footer-main .row .bottom-footer-links li a { padding: 0; } + body footer .footer-main .row .top-border { padding-top: 40px; margin-top: 40px; } } +.copy-code-container { position: absolute; right: 0.3em; } + +.copy-code-button { align-items: center; justify-content: center; border: none; cursor: pointer; font-size: 1rem; color: #000000; padding: 0.4em 0.5em; background-color: transparent; outline: none; } +.copy-code-button:hover { color: #279890 !important; } + +.highlighter-rouge { position: relative; } + +.language-shell.highlighter-rouge pre.highlight::before, .language-sh.highlighter-rouge pre.highlight::before, .language-bash.highlighter-rouge pre.highlight::before { content: "$ "; color: #000000; font-weight: 400; font-family: monospace; font-size: 0.75em; } + +th, td { background-color: transparent; } + +.main-content ul.ui-tabs-nav > li::before { content: ""; } + +.tabs { background: transparent; } +.tabs.ui-widget .copy-code-button { font-family: "Font Awesome 6 Free" !important; } + +.tabs .ui-widget-header { background: transparent; border: none; border-bottom: 1px solid #d3d3d3; -moz-border-radius: 0px; -webkit-border-radius: 0px; border-radius: 0px; } + +.tabs .ui-tabs-nav .ui-state-default { background: transparent; border: none; } + +.tabs .ui-tabs-nav .ui-state-active { border: none; } + +.tabs .ui-tabs-nav .ui-state-default a { color: #c0c0c0; } + +.tabs .ui-tabs-nav .ui-state-active a { color: #279890; } + +/*overridden*/ +div.highlighter-rouge { border-radius: 4px; } +div.highlighter-rouge:hover > button { cursor: pointer; opacity: 0.5; } + +.highlighter-rouge button.copy-code-button { color: #000000; } + +/* Tooltip container */ +.tooltip { position: relative; display: inline-block; border-bottom: 0; } +.tooltip > a, .tooltip span { text-decoration: underline; text-decoration-style: dotted; background-image: none; } +.tooltip .tooltiptext { visibility: hidden; text-decoration: none; background-color: #2ce5b5; color: #000000; padding: 10px 20px; border-radius: 6px; position: absolute; z-index: 1; width: 400px; top: 100%; left: 50%; margin-left: -200px; } +.tooltip:hover .tooltiptext { visibility: visible; } +.tooltip .tooltiptext::after { content: " "; position: absolute; color: #2ce5b5; bottom: 100%; /* At the top of the tooltip */ left: 50%; margin-left: -5px; border-width: 5px; border-style: solid; border-color: transparent transparent #2ce5b5 transparent; } + +.ui-dialog-titlebar { display: none; } + +.feedback-container { position: sticky; top: 180px; z-index: 9999; } +@media (max-width: 1024px) { .feedback-container { display: none; } } +.feedback-container .feedback-buttons { position: absolute; right: 70px; margin-bottom: 20px; top: -55px; } +.feedback-container .feedback-buttons .tooltiptext { font-size: small; width: 150px; margin-left: -75px; text-align: center; } +.feedback-container .feedback-buttons .page-helpful-btn { background: none !important; border-radius: 50%; border: solid #279890 2px; color: #279890; height: 40px; width: 40px; margin: 0 3px; cursor: pointer; } +.feedback-container .feedback-buttons .page-helpful-btn:focus-visible { outline: none; } + +#is-helpful-ty { display: none; } +#is-helpful-ty a { color: #279890 !important; } + +.badge { padding: 2px 4px; background: #ffddec; text-transform: uppercase; text-align: center; border-radius: 6px; font-size: x-small; } + +.quickstart-steps { padding: 10px; } +.quickstart-steps img { width: 50px; height: 50px; } + +.row { display: flex; flex-direction: row; } + +.col { display: flex; flex-direction: column; justify-content: center; margin-bottom: 0.3em; } +.col.step-num { margin-right: 20px; } +.col.step-num > img { width: 40px; } + +img.quickstart { box-shadow: 3px 3px 1px #ccc; border-width: thin; border-color: black; border-style: ridge; } + +.breadcrumb-nav { margin-top: 1rem; } diff --git a/v1.46/assets/css/just-the-docs-default.css b/v1.46/assets/css/just-the-docs-default.css new file mode 100644 index 000000000..333b8f650 --- /dev/null +++ b/v1.46/assets/css/just-the-docs-default.css @@ -0,0 +1,2562 @@ +@charset "UTF-8"; +@import url("pygments/igorpro.css"); +.highlight, pre.highlight { background: #f9f9f9; color: #383942; } + +.highlight pre { background: #f9f9f9; } + +.highlight .hll { background: #f9f9f9; } + +.highlight .c { color: #9fa0a6; font-style: italic; } + +.highlight .err { color: #fff; background-color: #e05151; } + +.highlight .k { color: #a625a4; } + +.highlight .l { color: #50a04f; } + +.highlight .n { color: #383942; } + +.highlight .o { color: #383942; } + +.highlight .p { color: #383942; } + +.highlight .cm { color: #9fa0a6; font-style: italic; } + +.highlight .cp { color: #9fa0a6; font-style: italic; } + +.highlight .c1 { color: #9fa0a6; font-style: italic; } + +.highlight .cs { color: #9fa0a6; font-style: italic; } + +.highlight .ge { font-style: italic; } + +.highlight .gs { font-weight: 700; } + +.highlight .kc { color: #a625a4; } + +.highlight .kd { color: #a625a4; } + +.highlight .kn { color: #a625a4; } + +.highlight .kp { color: #a625a4; } + +.highlight .kr { color: #a625a4; } + +.highlight .kt { color: #a625a4; } + +.highlight .ld { color: #50a04f; } + +.highlight .m { color: #b66a00; } + +.highlight .s { color: #50a04f; } + +.highlight .na { color: #b66a00; } + +.highlight .nb { color: #ca7601; } + +.highlight .nc { color: #ca7601; } + +.highlight .no { color: #ca7601; } + +.highlight .nd { color: #ca7601; } + +.highlight .ni { color: #ca7601; } + +.highlight .ne { color: #ca7601; } + +.highlight .nf { color: #383942; } + +.highlight .nl { color: #ca7601; } + +.highlight .nn { color: #383942; } + +.highlight .nx { color: #383942; } + +.highlight .py { color: #ca7601; } + +.highlight .nt { color: #e35549; } + +.highlight .nv { color: #ca7601; } + +.highlight .ow { font-weight: 700; } + +.highlight .w { color: #f8f8f2; } + +.highlight .mf { color: #b66a00; } + +.highlight .mh { color: #b66a00; } + +.highlight .mi { color: #b66a00; } + +.highlight .mo { color: #b66a00; } + +.highlight .sb { color: #50a04f; } + +.highlight .sc { color: #50a04f; } + +.highlight .sd { color: #50a04f; } + +.highlight .s2 { color: #50a04f; } + +.highlight .se { color: #50a04f; } + +.highlight .sh { color: #50a04f; } + +.highlight .si { color: #50a04f; } + +.highlight .sx { color: #50a04f; } + +.highlight .sr { color: #0083bb; } + +.highlight .s1 { color: #50a04f; } + +.highlight .ss { color: #0083bb; } + +.highlight .bp { color: #ca7601; } + +.highlight .vc { color: #ca7601; } + +.highlight .vg { color: #ca7601; } + +.highlight .vi { color: #e35549; } + +.highlight .il { color: #b66a00; } + +.highlight .gu { color: #75715e; } + +.highlight .gd { color: #e05151; } + +.highlight .gi { color: #43d089; } + +.highlight .language-json .w + .s2 { color: #e35549; } + +.highlight .language-json .kc { color: #0083bb; } + +/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ +/* Document ========================================================================== */ +/** 1. Correct the line height in all browsers. 2. Prevent adjustments of font size after orientation changes in iOS. */ +html { line-height: 1.15; /* 1 */ -webkit-text-size-adjust: 100%; /* 2 */ } + +/* Sections ========================================================================== */ +/** Remove the margin in all browsers. */ +body { margin: 0; } + +/** Render the `main` element consistently in IE. */ +main { display: block; } + +/** Correct the font size and margin on `h1` elements within `section` and `article` contexts in Chrome, Firefox, and Safari. */ +h1 { font-size: 2em; margin: 0.67em 0; } + +/* Grouping content ========================================================================== */ +/** 1. Add the correct box sizing in Firefox. 2. Show the overflow in Edge and IE. */ +hr { box-sizing: content-box; /* 1 */ height: 0; /* 1 */ overflow: visible; /* 2 */ } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +pre { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/* Text-level semantics ========================================================================== */ +/** Remove the gray background on active links in IE 10. */ +a { background-color: transparent; } + +/** 1. Remove the bottom border in Chrome 57- 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. */ +abbr[title] { border-bottom: none; /* 1 */ text-decoration: underline; /* 2 */ text-decoration: underline dotted; /* 2 */ } + +/** Add the correct font weight in Chrome, Edge, and Safari. */ +b, strong { font-weight: bolder; } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +code, kbd, samp { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/** Add the correct font size in all browsers. */ +small { font-size: 80%; } + +/** Prevent `sub` and `sup` elements from affecting the line height in all browsers. */ +sub, sup { font-size: 75%; line-height: 0; position: relative; vertical-align: baseline; } + +sub { bottom: -0.25em; } + +sup { top: -0.5em; } + +/* Embedded content ========================================================================== */ +/** Remove the border on images inside links in IE 10. */ +img { border-style: none; } + +/* Forms ========================================================================== */ +/** 1. Change the font styles in all browsers. 2. Remove the margin in Firefox and Safari. */ +button, input, optgroup, select, textarea { font-family: inherit; /* 1 */ font-size: 100%; /* 1 */ line-height: 1.15; /* 1 */ margin: 0; /* 2 */ } + +/** Show the overflow in IE. 1. Show the overflow in Edge. */ +button, input { /* 1 */ overflow: visible; } + +/** Remove the inheritance of text transform in Edge, Firefox, and IE. 1. Remove the inheritance of text transform in Firefox. */ +button, select { /* 1 */ text-transform: none; } + +/** Correct the inability to style clickable types in iOS and Safari. */ +button, [type="button"], [type="reset"], [type="submit"] { -webkit-appearance: button; } + +/** Remove the inner border and padding in Firefox. */ +button::-moz-focus-inner, [type="button"]::-moz-focus-inner, [type="reset"]::-moz-focus-inner, [type="submit"]::-moz-focus-inner { border-style: none; padding: 0; } + +/** Restore the focus styles unset by the previous rule. */ +button:-moz-focusring, [type="button"]:-moz-focusring, [type="reset"]:-moz-focusring, [type="submit"]:-moz-focusring { outline: 1px dotted ButtonText; } + +/** Correct the padding in Firefox. */ +fieldset { padding: 0.35em 0.75em 0.625em; } + +/** 1. Correct the text wrapping in Edge and IE. 2. Correct the color inheritance from `fieldset` elements in IE. 3. Remove the padding so developers are not caught out when they zero out `fieldset` elements in all browsers. */ +legend { box-sizing: border-box; /* 1 */ color: inherit; /* 2 */ display: table; /* 1 */ max-width: 100%; /* 1 */ padding: 0; /* 3 */ white-space: normal; /* 1 */ } + +/** Add the correct vertical alignment in Chrome, Firefox, and Opera. */ +progress { vertical-align: baseline; } + +/** Remove the default vertical scrollbar in IE 10+. */ +textarea { overflow: auto; } + +/** 1. Add the correct box sizing in IE 10. 2. Remove the padding in IE 10. */ +[type="checkbox"], [type="radio"] { box-sizing: border-box; /* 1 */ padding: 0; /* 2 */ } + +/** Correct the cursor style of increment and decrement buttons in Chrome. */ +[type="number"]::-webkit-inner-spin-button, [type="number"]::-webkit-outer-spin-button { height: auto; } + +/** 1. Correct the odd appearance in Chrome and Safari. 2. Correct the outline style in Safari. */ +[type="search"] { -webkit-appearance: textfield; /* 1 */ outline-offset: -2px; /* 2 */ } + +/** Remove the inner padding in Chrome and Safari on macOS. */ +[type="search"]::-webkit-search-decoration { -webkit-appearance: none; } + +/** 1. Correct the inability to style clickable types in iOS and Safari. 2. Change font properties to `inherit` in Safari. */ +::-webkit-file-upload-button { -webkit-appearance: button; /* 1 */ font: inherit; /* 2 */ } + +/* Interactive ========================================================================== */ +/* Add the correct display in Edge, IE 10+, and Firefox. */ +details { display: block; } + +/* Add the correct display in all browsers. */ +summary { display: list-item; } + +/* Misc ========================================================================== */ +/** Add the correct display in IE 10+. */ +template { display: none; } + +/** Add the correct display in IE 10. */ +[hidden] { display: none; } + +:root { color-scheme: light; } + +* { box-sizing: border-box; } + +html { font-size: 0.875rem !important; scroll-behavior: smooth; } +@media (min-width: 31.25rem) { html { font-size: 1rem !important; } } + +body { font-family: system-ui, -apple-system, blinkmacsystemfont, "Segoe UI", roboto, "Helvetica Neue", arial, sans-serif; font-size: inherit; line-height: 1.4; color: #5c5962; background-color: #fff; overflow-wrap: break-word; } + +ol, ul, dl, pre, address, blockquote, table, div, hr, form, fieldset, noscript .table-wrapper { margin-top: 0; } + +h1, h2, h3, h4, h5, h6, #toctitle { margin-top: 0; margin-bottom: 1em; font-weight: 500; line-height: 1.25; color: #27262b; } + +p { margin-top: 1em; margin-bottom: 1em; } + +a { color: #7253ed; text-decoration: none; } + +a:not([class]) { text-decoration: underline; text-decoration-color: #eeebee; text-underline-offset: 2px; } +a:not([class]):hover { text-decoration-color: rgba(114, 83, 237, 0.45); } + +code { font-family: "SFMono-Regular", menlo, consolas, monospace; font-size: 0.75em; line-height: 1.4; } + +figure, pre { margin: 0; } + +li { margin: 0.25em 0; } + +img { max-width: 100%; height: auto; } + +hr { height: 1px; padding: 0; margin: 2rem 0; background-color: #eeebee; border: 0; } + +blockquote { margin: 10px 0; margin-block-start: 0; margin-inline-start: 0; padding-left: 1rem; border-left: 3px solid #eeebee; } + +.side-bar { z-index: 0; display: flex; flex-wrap: wrap; background-color: #f5f6fa; } +@media (min-width: 50rem) { .side-bar { flex-flow: column nowrap; position: fixed; width: 15.5rem; height: 100%; border-right: 1px solid #eeebee; align-items: flex-end; } } +@media (min-width: 66.5rem) { .side-bar { width: calc((100% - 66.5rem) / 2 + 16.5rem); min-width: 16.5rem; } } + +@media (min-width: 50rem) { .main { position: relative; max-width: 50rem; margin-left: 15.5rem; } } +@media (min-width: 66.5rem) { .main { margin-left: Max(16.5rem, calc((100% - 66.5rem) / 2 + 16.5rem)); } } + +.main-content-wrap { padding-right: 1rem; padding-left: 1rem; padding-top: 1rem; padding-bottom: 1rem; } +@media (min-width: 50rem) { .main-content-wrap { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 50rem) { .main-content-wrap { padding-top: 2rem; padding-bottom: 2rem; } } + +.main-header { z-index: 0; display: none; background-color: #f5f6fa; } +@media (min-width: 50rem) { .main-header { display: flex; justify-content: space-between; height: 3.75rem; background-color: #fff; border-bottom: 1px solid #eeebee; } } +.main-header.nav-open { display: block; } +@media (min-width: 50rem) { .main-header.nav-open { display: flex; } } + +.site-nav, .site-header, .site-footer { width: 100%; } +@media (min-width: 66.5rem) { .site-nav, .site-header, .site-footer { width: 16.5rem; } } + +.site-nav { display: none; } +.site-nav.nav-open { display: block; } +@media (min-width: 50rem) { .site-nav { display: block; padding-top: 3rem; padding-bottom: 1rem; overflow-y: auto; flex: 1 1 auto; } } + +.site-header { display: flex; min-height: 3.75rem; align-items: center; } +@media (min-width: 50rem) { .site-header { height: 3.75rem; max-height: 3.75rem; border-bottom: 1px solid #eeebee; } } + +.site-title { padding-right: 1rem; padding-left: 1rem; flex-grow: 1; display: flex; height: 100%; align-items: center; padding-top: 0.75rem; padding-bottom: 0.75rem; color: #27262b; font-size: 1.125rem !important; } +@media (min-width: 50rem) { .site-title { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-title { font-size: 1.5rem !important; line-height: 1.25; } } +@media (min-width: 50rem) { .site-title { padding-top: 0.5rem; padding-bottom: 0.5rem; } } + +.site-logo { width: 100%; height: 100%; background-image: url("/v1.46/assets/logo.svg"); background-repeat: no-repeat; background-position: left center; background-size: contain; } + +.site-button { display: flex; height: 100%; padding: 1rem; align-items: center; } + +@media (min-width: 50rem) { .site-header .site-button { display: none; } } +.site-title:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 80%, rgba(235, 237, 245, 0) 100%); } + +.site-button:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 100%); } + +body { position: relative; padding-bottom: 4rem; overflow-y: scroll; } +@media (min-width: 50rem) { body { position: static; padding-bottom: 0; } } + +.site-footer { padding-right: 1rem; padding-left: 1rem; position: absolute; bottom: 0; left: 0; padding-top: 1rem; padding-bottom: 1rem; color: #959396; font-size: 0.6875rem !important; } +@media (min-width: 50rem) { .site-footer { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-footer { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .site-footer { position: static; justify-self: end; } } + +.icon { width: 1.5rem; height: 1.5rem; color: #7253ed; } + +.main-content { line-height: 1.6; } +.main-content ol, .main-content ul, .main-content dl, .main-content pre, .main-content address, .main-content blockquote, .main-content .table-wrapper { margin-top: 0.5em; } +.main-content a { overflow: hidden; text-overflow: ellipsis; } +.main-content ul, .main-content ol { padding-left: 1.5em; } +.main-content li .highlight { margin-top: 0.25rem; } +.main-content ol { list-style-type: none; counter-reset: step-counter; } +.main-content ol > li { position: relative; } +.main-content ol > li::before { position: absolute; top: 0.2em; left: -1.6em; color: #959396; content: counter(step-counter); counter-increment: step-counter; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .main-content ol > li::before { font-size: 0.875rem !important; } } +@media (min-width: 31.25rem) { .main-content ol > li::before { top: 0.11em; } } +.main-content ol > li ol { counter-reset: sub-counter; } +.main-content ol > li ol > li::before { content: counter(sub-counter,lower-alpha); counter-increment: sub-counter; } +.main-content ul { list-style: none; } +.main-content ul > li::before { position: absolute; margin-left: -1.4em; color: #959396; content: "•"; } +.main-content .task-list-item::before { content: ""; } +.main-content .task-list-item-checkbox { margin-right: 0.6em; margin-left: -1.4em; } +.main-content hr + * { margin-top: 0; } +.main-content h1:first-of-type { margin-top: 0.5em; } +.main-content dl { display: grid; grid-template: auto / 10em 1fr; } +.main-content dt, .main-content dd { margin: 0.25em 0; } +.main-content dt { grid-column: 1; font-weight: 500; text-align: right; } +.main-content dt::after { content: ":"; } +.main-content dd { grid-column: 2; margin-bottom: 0; margin-left: 1em; } +.main-content dd blockquote:first-child, .main-content dd div:first-child, .main-content dd dl:first-child, .main-content dd dt:first-child, .main-content dd h1:first-child, .main-content dd h2:first-child, .main-content dd h3:first-child, .main-content dd h4:first-child, .main-content dd h5:first-child, .main-content dd h6:first-child, .main-content dd li:first-child, .main-content dd ol:first-child, .main-content dd p:first-child, .main-content dd pre:first-child, .main-content dd table:first-child, .main-content dd ul:first-child, .main-content dd .table-wrapper:first-child { margin-top: 0; } +.main-content dd dl:first-child dt:first-child, .main-content dd dl:first-child dd:nth-child(2), .main-content ol dl:first-child dt:first-child, .main-content ol dl:first-child dd:nth-child(2), .main-content ul dl:first-child dt:first-child, .main-content ul dl:first-child dd:nth-child(2) { margin-top: 0; } +.main-content .anchor-heading { position: absolute; right: -1rem; width: 1.5rem; height: 100%; padding-right: 0.25rem; padding-left: 0.25rem; overflow: visible; } +@media (min-width: 50rem) { .main-content .anchor-heading { right: auto; left: -1.5rem; } } +.main-content .anchor-heading svg { display: inline-block; width: 100%; height: 100%; color: #7253ed; visibility: hidden; } +.main-content .anchor-heading:hover svg, .main-content .anchor-heading:focus svg, .main-content h1:hover > .anchor-heading svg, .main-content h2:hover > .anchor-heading svg, .main-content h3:hover > .anchor-heading svg, .main-content h4:hover > .anchor-heading svg, .main-content h5:hover > .anchor-heading svg, .main-content h6:hover > .anchor-heading svg { visibility: visible; } +.main-content summary { cursor: pointer; } +.main-content h1, .main-content h2, .main-content h3, .main-content h4, .main-content h5, .main-content h6, .main-content #toctitle { position: relative; margin-top: 1.5em; margin-bottom: 0.25em; } +.main-content h1 + table, .main-content h1 + .table-wrapper, .main-content h1 + .code-example, .main-content h1 + .highlighter-rouge, .main-content h1 + .sectionbody .listingblock, .main-content h2 + table, .main-content h2 + .table-wrapper, .main-content h2 + .code-example, .main-content h2 + .highlighter-rouge, .main-content h2 + .sectionbody .listingblock, .main-content h3 + table, .main-content h3 + .table-wrapper, .main-content h3 + .code-example, .main-content h3 + .highlighter-rouge, .main-content h3 + .sectionbody .listingblock, .main-content h4 + table, .main-content h4 + .table-wrapper, .main-content h4 + .code-example, .main-content h4 + .highlighter-rouge, .main-content h4 + .sectionbody .listingblock, .main-content h5 + table, .main-content h5 + .table-wrapper, .main-content h5 + .code-example, .main-content h5 + .highlighter-rouge, .main-content h5 + .sectionbody .listingblock, .main-content h6 + table, .main-content h6 + .table-wrapper, .main-content h6 + .code-example, .main-content h6 + .highlighter-rouge, .main-content h6 + .sectionbody .listingblock, .main-content #toctitle + table, .main-content #toctitle + .table-wrapper, .main-content #toctitle + .code-example, .main-content #toctitle + .highlighter-rouge, .main-content #toctitle + .sectionbody .listingblock { margin-top: 1em; } +.main-content h1 + p:not(.label), .main-content h2 + p:not(.label), .main-content h3 + p:not(.label), .main-content h4 + p:not(.label), .main-content h5 + p:not(.label), .main-content h6 + p:not(.label), .main-content #toctitle + p:not(.label) { margin-top: 0; } +.main-content > h1:first-child, .main-content > h2:first-child, .main-content > h3:first-child, .main-content > h4:first-child, .main-content > h5:first-child, .main-content > h6:first-child, .main-content > .sect1:first-child > h2, .main-content > .sect2:first-child > h3, .main-content > .sect3:first-child > h4, .main-content > .sect4:first-child > h5, .main-content > .sect5:first-child > h6 { margin-top: 0.5rem; } + +.nav-list { padding: 0; margin-top: 0; margin-bottom: 0; list-style: none; } +.nav-list .nav-list-item { font-size: 0.875rem !important; position: relative; margin: 0; } +@media (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 1rem !important; } } +@media (min-width: 50rem) { .nav-list .nav-list-item { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 0.875rem !important; } } + +.nav-list .nav-list-item .nav-list-link { display: block; min-height: 3rem; padding-top: 0.25rem; padding-bottom: 0.25rem; line-height: 2.5rem; padding-right: 3rem; padding-left: 1rem; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-link { min-height: 2rem; line-height: 1.5rem; padding-right: 2rem; padding-left: 2rem; } } +.nav-list .nav-list-item .nav-list-link.external > svg { width: 1rem; height: 1rem; vertical-align: text-bottom; } +.nav-list .nav-list-item .nav-list-link.active { font-weight: 600; text-decoration: none; } +.nav-list .nav-list-item .nav-list-link:hover, .nav-list .nav-list-item .nav-list-link.active { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 80%, rgba(235, 237, 245, 0) 100%); } +.nav-list .nav-list-item .nav-list-expander { position: absolute; right: 0; width: 3rem; height: 3rem; padding: 0.75rem; color: #7253ed; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-expander { width: 2rem; height: 2rem; padding: 0.5rem; } } +.nav-list .nav-list-item .nav-list-expander:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 100%); } +.nav-list .nav-list-item .nav-list-expander svg { transform: rotate(90deg); } +.nav-list .nav-list-item > .nav-list { display: none; padding-left: 0.75rem; list-style: none; } +.nav-list .nav-list-item > .nav-list .nav-list-item { position: relative; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-link { color: #5c5962; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-expander { color: #5c5962; } +.nav-list .nav-list-item.active > .nav-list-expander svg { transform: rotate(-90deg); } +.nav-list .nav-list-item.active > .nav-list { display: block; } + +.nav-category { padding: 0.5rem 1rem; font-weight: 600; text-align: start; text-transform: uppercase; border-bottom: 1px solid #eeebee; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .nav-category { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .nav-category { padding: 0.5rem 2rem; margin-top: 1rem; text-align: start; } + .nav-category:first-child { margin-top: 0; } } + +.nav-list.nav-category-list > .nav-list-item { margin: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list { padding: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-link { color: #7253ed; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-expander { color: #7253ed; } + +.aux-nav { height: 100%; overflow-x: auto; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .aux-nav { font-size: 0.75rem !important; } } +.aux-nav .aux-nav-list { display: flex; height: 100%; padding: 0; margin: 0; list-style: none; } +.aux-nav .aux-nav-list-item { display: inline-block; height: 100%; padding: 0; margin: 0; } +@media (min-width: 50rem) { .aux-nav { padding-right: 1rem; } } + +@media (min-width: 50rem) { .breadcrumb-nav { margin-top: -1rem; } } + +.breadcrumb-nav-list { padding-left: 0; margin-bottom: 0.75rem; list-style: none; } + +.breadcrumb-nav-list-item { display: table-cell; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .breadcrumb-nav-list-item { font-size: 0.75rem !important; } } +.breadcrumb-nav-list-item::before { display: none; } +.breadcrumb-nav-list-item::after { display: inline-block; margin-right: 0.5rem; margin-left: 0.5rem; color: #959396; content: "/"; } +.breadcrumb-nav-list-item:last-child::after { content: ""; } + +h1, .text-alpha { font-size: 2rem !important; line-height: 1.25; font-weight: 300; } +@media (min-width: 31.25rem) { h1, .text-alpha { font-size: 2.25rem !important; } } + +h2, .text-beta, #toctitle { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { h2, .text-beta, #toctitle { font-size: 1.5rem !important; line-height: 1.25; } } + +h3, .text-gamma { font-size: 1rem !important; } +@media (min-width: 31.25rem) { h3, .text-gamma { font-size: 1.125rem !important; } } + +h4, .text-delta { font-size: 0.6875rem !important; font-weight: 400; text-transform: uppercase; letter-spacing: 0.1em; } +@media (min-width: 31.25rem) { h4, .text-delta { font-size: 0.75rem !important; } } + +h4 code { text-transform: none; } + +h5, .text-epsilon { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { h5, .text-epsilon { font-size: 0.875rem !important; } } + +h6, .text-zeta { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { h6, .text-zeta { font-size: 0.75rem !important; } } + +.text-small { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .text-small { font-size: 0.75rem !important; } } + +.text-mono { font-family: "SFMono-Regular", menlo, consolas, monospace !important; } + +.text-left { text-align: left !important; } + +.text-center { text-align: center !important; } + +.text-right { text-align: right !important; } + +.label:not(g), .label-blue:not(g) { display: inline-block; padding: 0.16em 0.56em; margin-right: 0.5rem; margin-left: 0.5rem; color: #fff; text-transform: uppercase; vertical-align: middle; background-color: #2869e6; font-size: 0.6875rem !important; border-radius: 12px; } +@media (min-width: 31.25rem) { .label:not(g), .label-blue:not(g) { font-size: 0.75rem !important; } } + +.label-green:not(g) { background-color: #009c7b; } + +.label-purple:not(g) { background-color: #5e41d0; } + +.label-red:not(g) { background-color: #e94c4c; } + +.label-yellow:not(g) { color: #44434d; background-color: #f7d12e; } + +.btn { display: inline-block; box-sizing: border-box; padding: 0.3em 1em; margin: 0; font-family: inherit; font-size: inherit; font-weight: 500; line-height: 1.5; color: #7253ed; text-decoration: none; vertical-align: baseline; cursor: pointer; background-color: #f7f7f7; border-width: 0; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); appearance: none; } +.btn:focus { text-decoration: none; outline: none; box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:focus:hover, .btn.selected:focus { box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:hover, .btn.zeroclipboard-is-hover { color: #6a4aec; } +.btn:hover, .btn:active, .btn.zeroclipboard-is-hover, .btn.zeroclipboard-is-active { text-decoration: none; background-color: #f4f4f4; } +.btn:active, .btn.selected, .btn.zeroclipboard-is-active { background-color: #efefef; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn.selected:hover { background-color: #cfcfcf; } +.btn:disabled, .btn:disabled:hover, .btn.disabled, .btn.disabled:hover { color: rgba(102, 102, 102, 0.5); cursor: default; background-color: rgba(229, 229, 229, 0.5); background-image: none; box-shadow: none; } + +.btn-outline { color: #7253ed; background: transparent; box-shadow: inset 0 0 0 2px #e6e1e8; } +.btn-outline:hover, .btn-outline:active, .btn-outline.zeroclipboard-is-hover, .btn-outline.zeroclipboard-is-active { color: #6341eb; text-decoration: none; background-color: transparent; box-shadow: inset 0 0 0 3px #e6e1e8; } +.btn-outline:focus { text-decoration: none; outline: none; box-shadow: inset 0 0 0 2px #5c5962, 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn-outline:focus:hover, .btn-outline.selected:focus { box-shadow: inset 0 0 0 2px #5c5962; } + +.btn-primary { color: #fff; background-color: #5739ce; background-image: linear-gradient(#6f55d5, #5739ce); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-primary:hover, .btn-primary.zeroclipboard-is-hover { color: #fff; background-color: #5132cb; background-image: linear-gradient(#6549d2, #5132cb); } +.btn-primary:active, .btn-primary.selected, .btn-primary.zeroclipboard-is-active { background-color: #4f31c6; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-primary.selected:hover { background-color: #472cb2; } + +.btn-purple { color: #fff; background-color: #5739ce; background-image: linear-gradient(#6f55d5, #5739ce); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-purple:hover, .btn-purple.zeroclipboard-is-hover { color: #fff; background-color: #5132cb; background-image: linear-gradient(#6549d2, #5132cb); } +.btn-purple:active, .btn-purple.selected, .btn-purple.zeroclipboard-is-active { background-color: #4f31c6; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-purple.selected:hover { background-color: #472cb2; } + +.btn-blue { color: #fff; background-color: #227efa; background-image: linear-gradient(#4593fb, #227efa); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-blue:hover, .btn-blue.zeroclipboard-is-hover { color: #fff; background-color: #1878fa; background-image: linear-gradient(#368afa, #1878fa); } +.btn-blue:active, .btn-blue.selected, .btn-blue.zeroclipboard-is-active { background-color: #1375f9; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-blue.selected:hover { background-color: #0669ed; } + +.btn-green { color: #fff; background-color: #10ac7d; background-image: linear-gradient(#13cc95, #10ac7d); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-green:hover, .btn-green.zeroclipboard-is-hover { color: #fff; background-color: #0fa276; background-image: linear-gradient(#12be8b, #0fa276); } +.btn-green:active, .btn-green.selected, .btn-green.zeroclipboard-is-active { background-color: #0f9e73; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-green.selected:hover { background-color: #0d8662; } + +.btn-reset { background: none; border: none; margin: 0; text-align: inherit; font: inherit; border-radius: 0; appearance: none; } + +.search { position: relative; z-index: 2; flex-grow: 1; height: 4rem; padding: 0.5rem; transition: padding linear 200ms; } +@media (min-width: 50rem) { .search { position: relative !important; width: auto !important; height: 100% !important; padding: 0; transition: none; } } + +.search-input-wrap { position: relative; z-index: 1; height: 3rem; overflow: hidden; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); transition: height linear 200ms; } +@media (min-width: 50rem) { .search-input-wrap { position: absolute; width: 100%; max-width: 33.5rem; height: 100% !important; border-radius: 0; box-shadow: none; transition: width ease 400ms; } } + +.search-input { position: absolute; width: 100%; height: 100%; padding: 0.5rem 1rem 0.5rem 2.5rem; font-size: 1rem; color: #5c5962; background-color: #fff; border-top: 0; border-right: 0; border-bottom: 0; border-left: 0; border-radius: 0; } +@media (min-width: 50rem) { .search-input { padding: 0.5rem 1rem 0.5rem 3.5rem; font-size: 0.875rem; background-color: #fff; transition: padding-left linear 200ms; } } +.search-input:focus { outline: 0; } +.search-input:focus + .search-label .search-icon { color: #7253ed; } + +.search-label { position: absolute; display: flex; height: 100%; padding-left: 1rem; } +@media (min-width: 50rem) { .search-label { padding-left: 2rem; transition: padding-left linear 200ms; } } +.search-label .search-icon { width: 1.2rem; height: 1.2rem; align-self: center; color: #959396; } + +.search-results { position: absolute; left: 0; display: none; width: 100%; max-height: calc(100% - 4rem); overflow-y: auto; background-color: #fff; border-bottom-right-radius: 4px; border-bottom-left-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } +@media (min-width: 50rem) { .search-results { top: 100%; width: 33.5rem; max-height: calc(100vh - 200%) !important; } } + +.search-results-list { padding-left: 0; margin-bottom: 0.25rem; list-style: none; font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .search-results-list { font-size: 1rem !important; } } +@media (min-width: 50rem) { .search-results-list { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-results-list { font-size: 0.875rem !important; } } + +.search-results-list-item { padding: 0; margin: 0; } + +.search-result { display: block; padding: 0.25rem 0.75rem; } +.search-result:hover, .search-result.active { background-color: #ebedf5; } + +.search-result-title { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; } +@media (min-width: 31.25rem) { .search-result-title { display: inline-block; width: 40%; padding-right: 0.5rem; vertical-align: top; } } + +.search-result-doc { display: flex; align-items: center; word-wrap: break-word; } +.search-result-doc.search-result-doc-parent { opacity: 0.5; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.875rem !important; } } +@media (min-width: 50rem) { .search-result-doc.search-result-doc-parent { font-size: 0.6875rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.75rem !important; } } + +.search-result-doc .search-result-icon { width: 1rem; height: 1rem; margin-right: 0.5rem; color: #7253ed; flex-shrink: 0; } +.search-result-doc .search-result-doc-title { overflow: auto; } + +.search-result-section { margin-left: 1.5rem; word-wrap: break-word; } + +.search-result-rel-url { display: block; margin-left: 1.5rem; overflow: hidden; color: #959396; text-overflow: ellipsis; white-space: nowrap; font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .search-result-rel-url { font-size: 0.625rem !important; } } + +.search-result-previews { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; padding-left: 1rem; margin-left: 0.5rem; color: #959396; word-wrap: break-word; border-left: 1px solid; border-left-color: #eeebee; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .search-result-previews { font-size: 0.75rem !important; } } +@media (min-width: 31.25rem) { .search-result-previews { display: inline-block; width: 60%; padding-left: 0.5rem; margin-left: 0; vertical-align: top; } } + +.search-result-preview + .search-result-preview { margin-top: 0.25rem; } + +.search-result-highlight { font-weight: bold; } + +.search-no-result { padding: 0.5rem 0.75rem; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-no-result { font-size: 0.875rem !important; } } + +.search-button { position: fixed; right: 1rem; bottom: 1rem; display: flex; width: 3.5rem; height: 3.5rem; background-color: #fff; border: 1px solid rgba(114, 83, 237, 0.3); border-radius: 1.75rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); align-items: center; justify-content: center; } + +.search-overlay { position: fixed; top: 0; left: 0; z-index: 1; width: 0; height: 0; background-color: rgba(0, 0, 0, 0.3); opacity: 0; transition: opacity ease 400ms, width 0s 400ms, height 0s 400ms; } + +.search-active .search { position: fixed; top: 0; left: 0; width: 100%; height: 100%; padding: 0; } +.search-active .search-input-wrap { height: 4rem; border-radius: 0; } +@media (min-width: 50rem) { .search-active .search-input-wrap { width: 33.5rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } } +.search-active .search-input { background-color: #fff; } +@media (min-width: 50rem) { .search-active .search-input { padding-left: 2.3rem; } } +@media (min-width: 50rem) { .search-active .search-label { padding-left: 0.6rem; } } +.search-active .search-results { display: block; } +.search-active .search-overlay { width: 100%; height: 100%; opacity: 1; transition: opacity ease 400ms, width 0s, height 0s; } +@media (min-width: 50rem) { .search-active .main { position: fixed; right: 0; left: 0; } } +.search-active .main-header { padding-top: 4rem; } +@media (min-width: 50rem) { .search-active .main-header { padding-top: 0; } } + +.table-wrapper { display: block; width: 100%; max-width: 100%; margin-bottom: 1.5rem; overflow-x: auto; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } + +table { display: table; min-width: 100%; border-collapse: separate; } + +th, td { font-size: 0.75rem !important; min-width: 7.5rem; padding: 0.5rem 0.75rem; background-color: #fff; border-bottom: 1px solid rgba(238, 235, 238, 0.5); border-left: 1px solid #eeebee; } +@media (min-width: 31.25rem) { th, td { font-size: 0.875rem !important; } } +th:first-of-type, td:first-of-type { border-left: 0; } + +tbody tr:last-of-type th, tbody tr:last-of-type td { border-bottom: 0; } +tbody tr:last-of-type td { padding-bottom: 0.75rem; } + +thead th { border-bottom: 1px solid #eeebee; } + +:not(pre, figure) > code { padding: 0.2em 0.15em; font-weight: 400; background-color: #f5f6fa; border: 1px solid #eeebee; border-radius: 4px; } + +a:visited code { border-color: #eeebee; } + +div.highlighter-rouge, div.listingblock > div.content, figure.highlight { margin-top: 0; margin-bottom: 0.75rem; background-color: #f5f6fa; border-radius: 4px; box-shadow: none; -webkit-overflow-scrolling: touch; position: relative; padding: 0; } +div.highlighter-rouge > button, div.listingblock > div.content > button, figure.highlight > button { width: 0.75rem; opacity: 0; position: absolute; top: 0; right: 0; border: 0.75rem solid #f5f6fa; background-color: #f5f6fa; color: #5c5962; box-sizing: content-box; } +div.highlighter-rouge > button svg, div.listingblock > div.content > button svg, figure.highlight > button svg { fill: #5c5962; } +div.highlighter-rouge > button:active, div.listingblock > div.content > button:active, figure.highlight > button:active { text-decoration: none; outline: none; opacity: 1; } +div.highlighter-rouge > button:focus, div.listingblock > div.content > button:focus, figure.highlight > button:focus { opacity: 1; } +div.highlighter-rouge:hover > button, div.listingblock > div.content:hover > button, figure.highlight:hover > button { cursor: copy; opacity: 1; } + +div.highlighter-rouge div.highlight { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.highlighter-rouge pre.highlight, div.highlighter-rouge code { padding: 0; margin: 0; border: 0; } + +div.listingblock { margin-top: 0; margin-bottom: 0.75rem; } +div.listingblock div.content { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.listingblock div.content > pre, div.listingblock code { padding: 0; margin: 0; border: 0; } + +figure.highlight pre, figure.highlight :not(pre) > code { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } + +.highlight .table-wrapper { padding: 0.75rem 0; margin: 0; border: 0; box-shadow: none; } +.highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.6875rem !important; min-width: 0; padding: 0; background-color: #f5f6fa; border: 0; } +@media (min-width: 31.25rem) { .highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.75rem !important; } } +.highlight .table-wrapper td.gl { width: 1em; padding-right: 0.75rem; padding-left: 0.75rem; } +.highlight .table-wrapper pre { margin: 0; line-height: 2; } + +.code-example, .listingblock > .title { padding: 0.75rem; margin-bottom: 0.75rem; overflow: auto; border: 1px solid #eeebee; border-radius: 4px; } +.code-example + .highlighter-rouge, .code-example + .sectionbody .listingblock, .code-example + .content, .code-example + figure.highlight, .listingblock > .title + .highlighter-rouge, .listingblock > .title + .sectionbody .listingblock, .listingblock > .title + .content, .listingblock > .title + figure.highlight { position: relative; margin-top: -1rem; border-right: 1px solid #eeebee; border-bottom: 1px solid #eeebee; border-left: 1px solid #eeebee; border-top-left-radius: 0; border-top-right-radius: 0; } + +code.language-mermaid { padding: 0; background-color: inherit; border: 0; } + +.highlight, pre.highlight { background: #f5f6fa; color: #5c5962; } + +.highlight pre { background: #f5f6fa; } + +.text-grey-dk-000 { color: #959396 !important; } + +.text-grey-dk-100 { color: #5c5962 !important; } + +.text-grey-dk-200 { color: #44434d !important; } + +.text-grey-dk-250 { color: #302d36 !important; } + +.text-grey-dk-300 { color: #27262b !important; } + +.text-grey-lt-000 { color: #f5f6fa !important; } + +.text-grey-lt-100 { color: #eeebee !important; } + +.text-grey-lt-200 { color: #ecebed !important; } + +.text-grey-lt-300 { color: #e6e1e8 !important; } + +.text-blue-000 { color: #2c84fa !important; } + +.text-blue-100 { color: #2869e6 !important; } + +.text-blue-200 { color: #264caf !important; } + +.text-blue-300 { color: #183385 !important; } + +.text-green-000 { color: #41d693 !important; } + +.text-green-100 { color: #11b584 !important; } + +.text-green-200 { color: #009c7b !important; } + +.text-green-300 { color: #026e57 !important; } + +.text-purple-000 { color: #7253ed !important; } + +.text-purple-100 { color: #5e41d0 !important; } + +.text-purple-200 { color: #4e26af !important; } + +.text-purple-300 { color: #381885 !important; } + +.text-yellow-000 { color: #ffeb82 !important; } + +.text-yellow-100 { color: #fadf50 !important; } + +.text-yellow-200 { color: #f7d12e !important; } + +.text-yellow-300 { color: #e7af06 !important; } + +.text-red-000 { color: #f77e7e !important; } + +.text-red-100 { color: #f96e65 !important; } + +.text-red-200 { color: #e94c4c !important; } + +.text-red-300 { color: #dd2e2e !important; } + +.bg-grey-dk-000 { background-color: #959396 !important; } + +.bg-grey-dk-100 { background-color: #5c5962 !important; } + +.bg-grey-dk-200 { background-color: #44434d !important; } + +.bg-grey-dk-250 { background-color: #302d36 !important; } + +.bg-grey-dk-300 { background-color: #27262b !important; } + +.bg-grey-lt-000 { background-color: #f5f6fa !important; } + +.bg-grey-lt-100 { background-color: #eeebee !important; } + +.bg-grey-lt-200 { background-color: #ecebed !important; } + +.bg-grey-lt-300 { background-color: #e6e1e8 !important; } + +.bg-blue-000 { background-color: #2c84fa !important; } + +.bg-blue-100 { background-color: #2869e6 !important; } + +.bg-blue-200 { background-color: #264caf !important; } + +.bg-blue-300 { background-color: #183385 !important; } + +.bg-green-000 { background-color: #41d693 !important; } + +.bg-green-100 { background-color: #11b584 !important; } + +.bg-green-200 { background-color: #009c7b !important; } + +.bg-green-300 { background-color: #026e57 !important; } + +.bg-purple-000 { background-color: #7253ed !important; } + +.bg-purple-100 { background-color: #5e41d0 !important; } + +.bg-purple-200 { background-color: #4e26af !important; } + +.bg-purple-300 { background-color: #381885 !important; } + +.bg-yellow-000 { background-color: #ffeb82 !important; } + +.bg-yellow-100 { background-color: #fadf50 !important; } + +.bg-yellow-200 { background-color: #f7d12e !important; } + +.bg-yellow-300 { background-color: #e7af06 !important; } + +.bg-red-000 { background-color: #f77e7e !important; } + +.bg-red-100 { background-color: #f96e65 !important; } + +.bg-red-200 { background-color: #e94c4c !important; } + +.bg-red-300 { background-color: #dd2e2e !important; } + +.d-block { display: block !important; } + +.d-flex { display: flex !important; } + +.d-inline { display: inline !important; } + +.d-inline-block { display: inline-block !important; } + +.d-none { display: none !important; } + +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +.float-left { float: left !important; } + +.float-right { float: right !important; } + +.flex-justify-start { justify-content: flex-start !important; } + +.flex-justify-end { justify-content: flex-end !important; } + +.flex-justify-between { justify-content: space-between !important; } + +.flex-justify-around { justify-content: space-around !important; } + +.v-align-baseline { vertical-align: baseline !important; } + +.v-align-bottom { vertical-align: bottom !important; } + +.v-align-middle { vertical-align: middle !important; } + +.v-align-text-bottom { vertical-align: text-bottom !important; } + +.v-align-text-top { vertical-align: text-top !important; } + +.v-align-top { vertical-align: top !important; } + +.fs-1 { font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .fs-1 { font-size: 0.625rem !important; } } + +.fs-2 { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .fs-2 { font-size: 0.75rem !important; } } + +.fs-3 { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .fs-3 { font-size: 0.875rem !important; } } + +.fs-4 { font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .fs-4 { font-size: 1rem !important; } } + +.fs-5 { font-size: 1rem !important; } +@media (min-width: 31.25rem) { .fs-5 { font-size: 1.125rem !important; } } + +.fs-6 { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { .fs-6 { font-size: 1.5rem !important; line-height: 1.25; } } + +.fs-7 { font-size: 1.5rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-7 { font-size: 2rem !important; } } + +.fs-8 { font-size: 2rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-8 { font-size: 2.25rem !important; } } + +.fs-9 { font-size: 2.25rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-9 { font-size: 2.625rem !important; } } + +.fs-10 { font-size: 2.625rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-10 { font-size: 3rem !important; } } + +.fw-300 { font-weight: 300 !important; } + +.fw-400 { font-weight: 400 !important; } + +.fw-500 { font-weight: 500 !important; } + +.fw-700 { font-weight: 700 !important; } + +.lh-0 { line-height: 0 !important; } + +.lh-default { line-height: 1.4; } + +.lh-tight { line-height: 1.25; } + +.ls-5 { letter-spacing: 0.05em !important; } + +.ls-10 { letter-spacing: 0.1em !important; } + +.ls-0 { letter-spacing: 0 !important; } + +.text-uppercase { text-transform: uppercase !important; } + +.list-style-none { padding: 0 !important; margin: 0 !important; list-style: none !important; } +.list-style-none li::before { display: none !important; } + +.mx-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-0 { margin: 0 !important; } + +.mt-0 { margin-top: 0 !important; } + +.mr-0 { margin-right: 0 !important; } + +.mb-0 { margin-bottom: 0 !important; } + +.ml-0 { margin-left: 0 !important; } + +.mx-0 { margin-right: 0 !important; margin-left: 0 !important; } + +.my-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + +.mxn-0 { margin-right: -0 !important; margin-left: -0 !important; } + +.mx-0-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-1 { margin: 0.25rem !important; } + +.mt-1 { margin-top: 0.25rem !important; } + +.mr-1 { margin-right: 0.25rem !important; } + +.mb-1 { margin-bottom: 0.25rem !important; } + +.ml-1 { margin-left: 0.25rem !important; } + +.mx-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + +.my-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + +.mxn-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } + +.mx-1-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-2 { margin: 0.5rem !important; } + +.mt-2 { margin-top: 0.5rem !important; } + +.mr-2 { margin-right: 0.5rem !important; } + +.mb-2 { margin-bottom: 0.5rem !important; } + +.ml-2 { margin-left: 0.5rem !important; } + +.mx-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + +.my-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + +.mxn-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } + +.mx-2-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-3 { margin: 0.75rem !important; } + +.mt-3 { margin-top: 0.75rem !important; } + +.mr-3 { margin-right: 0.75rem !important; } + +.mb-3 { margin-bottom: 0.75rem !important; } + +.ml-3 { margin-left: 0.75rem !important; } + +.mx-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + +.my-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + +.mxn-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } + +.mx-3-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-4 { margin: 1rem !important; } + +.mt-4 { margin-top: 1rem !important; } + +.mr-4 { margin-right: 1rem !important; } + +.mb-4 { margin-bottom: 1rem !important; } + +.ml-4 { margin-left: 1rem !important; } + +.mx-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + +.my-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + +.mxn-4 { margin-right: -1rem !important; margin-left: -1rem !important; } + +.mx-4-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-5 { margin: 1.5rem !important; } + +.mt-5 { margin-top: 1.5rem !important; } + +.mr-5 { margin-right: 1.5rem !important; } + +.mb-5 { margin-bottom: 1.5rem !important; } + +.ml-5 { margin-left: 1.5rem !important; } + +.mx-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + +.my-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + +.mxn-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } + +.mx-5-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-6 { margin: 2rem !important; } + +.mt-6 { margin-top: 2rem !important; } + +.mr-6 { margin-right: 2rem !important; } + +.mb-6 { margin-bottom: 2rem !important; } + +.ml-6 { margin-left: 2rem !important; } + +.mx-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + +.my-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + +.mxn-6 { margin-right: -2rem !important; margin-left: -2rem !important; } + +.mx-6-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-7 { margin: 2.5rem !important; } + +.mt-7 { margin-top: 2.5rem !important; } + +.mr-7 { margin-right: 2.5rem !important; } + +.mb-7 { margin-bottom: 2.5rem !important; } + +.ml-7 { margin-left: 2.5rem !important; } + +.mx-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + +.my-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + +.mxn-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } + +.mx-7-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-8 { margin: 3rem !important; } + +.mt-8 { margin-top: 3rem !important; } + +.mr-8 { margin-right: 3rem !important; } + +.mb-8 { margin-bottom: 3rem !important; } + +.ml-8 { margin-left: 3rem !important; } + +.mx-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + +.my-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + +.mxn-8 { margin-right: -3rem !important; margin-left: -3rem !important; } + +.mx-8-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-9 { margin: 3.5rem !important; } + +.mt-9 { margin-top: 3.5rem !important; } + +.mr-9 { margin-right: 3.5rem !important; } + +.mb-9 { margin-bottom: 3.5rem !important; } + +.ml-9 { margin-left: 3.5rem !important; } + +.mx-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + +.my-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + +.mxn-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } + +.mx-9-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-10 { margin: 4rem !important; } + +.mt-10 { margin-top: 4rem !important; } + +.mr-10 { margin-right: 4rem !important; } + +.mb-10 { margin-bottom: 4rem !important; } + +.ml-10 { margin-left: 4rem !important; } + +.mx-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + +.my-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + +.mxn-10 { margin-right: -4rem !important; margin-left: -4rem !important; } + +.mx-10-auto { margin-right: auto !important; margin-left: auto !important; } + +@media (min-width: 20rem) { .m-xs-0 { margin: 0 !important; } + .mt-xs-0 { margin-top: 0 !important; } + .mr-xs-0 { margin-right: 0 !important; } + .mb-xs-0 { margin-bottom: 0 !important; } + .ml-xs-0 { margin-left: 0 !important; } + .mx-xs-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xs-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xs-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 20rem) { .m-xs-1 { margin: 0.25rem !important; } + .mt-xs-1 { margin-top: 0.25rem !important; } + .mr-xs-1 { margin-right: 0.25rem !important; } + .mb-xs-1 { margin-bottom: 0.25rem !important; } + .ml-xs-1 { margin-left: 0.25rem !important; } + .mx-xs-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xs-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xs-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 20rem) { .m-xs-2 { margin: 0.5rem !important; } + .mt-xs-2 { margin-top: 0.5rem !important; } + .mr-xs-2 { margin-right: 0.5rem !important; } + .mb-xs-2 { margin-bottom: 0.5rem !important; } + .ml-xs-2 { margin-left: 0.5rem !important; } + .mx-xs-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xs-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xs-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 20rem) { .m-xs-3 { margin: 0.75rem !important; } + .mt-xs-3 { margin-top: 0.75rem !important; } + .mr-xs-3 { margin-right: 0.75rem !important; } + .mb-xs-3 { margin-bottom: 0.75rem !important; } + .ml-xs-3 { margin-left: 0.75rem !important; } + .mx-xs-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xs-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xs-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 20rem) { .m-xs-4 { margin: 1rem !important; } + .mt-xs-4 { margin-top: 1rem !important; } + .mr-xs-4 { margin-right: 1rem !important; } + .mb-xs-4 { margin-bottom: 1rem !important; } + .ml-xs-4 { margin-left: 1rem !important; } + .mx-xs-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xs-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xs-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 20rem) { .m-xs-5 { margin: 1.5rem !important; } + .mt-xs-5 { margin-top: 1.5rem !important; } + .mr-xs-5 { margin-right: 1.5rem !important; } + .mb-xs-5 { margin-bottom: 1.5rem !important; } + .ml-xs-5 { margin-left: 1.5rem !important; } + .mx-xs-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xs-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xs-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 20rem) { .m-xs-6 { margin: 2rem !important; } + .mt-xs-6 { margin-top: 2rem !important; } + .mr-xs-6 { margin-right: 2rem !important; } + .mb-xs-6 { margin-bottom: 2rem !important; } + .ml-xs-6 { margin-left: 2rem !important; } + .mx-xs-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xs-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xs-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 20rem) { .m-xs-7 { margin: 2.5rem !important; } + .mt-xs-7 { margin-top: 2.5rem !important; } + .mr-xs-7 { margin-right: 2.5rem !important; } + .mb-xs-7 { margin-bottom: 2.5rem !important; } + .ml-xs-7 { margin-left: 2.5rem !important; } + .mx-xs-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xs-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xs-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 20rem) { .m-xs-8 { margin: 3rem !important; } + .mt-xs-8 { margin-top: 3rem !important; } + .mr-xs-8 { margin-right: 3rem !important; } + .mb-xs-8 { margin-bottom: 3rem !important; } + .ml-xs-8 { margin-left: 3rem !important; } + .mx-xs-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xs-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xs-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 20rem) { .m-xs-9 { margin: 3.5rem !important; } + .mt-xs-9 { margin-top: 3.5rem !important; } + .mr-xs-9 { margin-right: 3.5rem !important; } + .mb-xs-9 { margin-bottom: 3.5rem !important; } + .ml-xs-9 { margin-left: 3.5rem !important; } + .mx-xs-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xs-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xs-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 20rem) { .m-xs-10 { margin: 4rem !important; } + .mt-xs-10 { margin-top: 4rem !important; } + .mr-xs-10 { margin-right: 4rem !important; } + .mb-xs-10 { margin-bottom: 4rem !important; } + .ml-xs-10 { margin-left: 4rem !important; } + .mx-xs-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xs-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xs-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 31.25rem) { .m-sm-0 { margin: 0 !important; } + .mt-sm-0 { margin-top: 0 !important; } + .mr-sm-0 { margin-right: 0 !important; } + .mb-sm-0 { margin-bottom: 0 !important; } + .ml-sm-0 { margin-left: 0 !important; } + .mx-sm-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-sm-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-sm-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 31.25rem) { .m-sm-1 { margin: 0.25rem !important; } + .mt-sm-1 { margin-top: 0.25rem !important; } + .mr-sm-1 { margin-right: 0.25rem !important; } + .mb-sm-1 { margin-bottom: 0.25rem !important; } + .ml-sm-1 { margin-left: 0.25rem !important; } + .mx-sm-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-sm-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-sm-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 31.25rem) { .m-sm-2 { margin: 0.5rem !important; } + .mt-sm-2 { margin-top: 0.5rem !important; } + .mr-sm-2 { margin-right: 0.5rem !important; } + .mb-sm-2 { margin-bottom: 0.5rem !important; } + .ml-sm-2 { margin-left: 0.5rem !important; } + .mx-sm-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-sm-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-sm-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-3 { margin: 0.75rem !important; } + .mt-sm-3 { margin-top: 0.75rem !important; } + .mr-sm-3 { margin-right: 0.75rem !important; } + .mb-sm-3 { margin-bottom: 0.75rem !important; } + .ml-sm-3 { margin-left: 0.75rem !important; } + .mx-sm-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-sm-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-sm-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 31.25rem) { .m-sm-4 { margin: 1rem !important; } + .mt-sm-4 { margin-top: 1rem !important; } + .mr-sm-4 { margin-right: 1rem !important; } + .mb-sm-4 { margin-bottom: 1rem !important; } + .ml-sm-4 { margin-left: 1rem !important; } + .mx-sm-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-sm-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-sm-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 31.25rem) { .m-sm-5 { margin: 1.5rem !important; } + .mt-sm-5 { margin-top: 1.5rem !important; } + .mr-sm-5 { margin-right: 1.5rem !important; } + .mb-sm-5 { margin-bottom: 1.5rem !important; } + .ml-sm-5 { margin-left: 1.5rem !important; } + .mx-sm-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-sm-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-sm-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-6 { margin: 2rem !important; } + .mt-sm-6 { margin-top: 2rem !important; } + .mr-sm-6 { margin-right: 2rem !important; } + .mb-sm-6 { margin-bottom: 2rem !important; } + .ml-sm-6 { margin-left: 2rem !important; } + .mx-sm-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-sm-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-sm-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 31.25rem) { .m-sm-7 { margin: 2.5rem !important; } + .mt-sm-7 { margin-top: 2.5rem !important; } + .mr-sm-7 { margin-right: 2.5rem !important; } + .mb-sm-7 { margin-bottom: 2.5rem !important; } + .ml-sm-7 { margin-left: 2.5rem !important; } + .mx-sm-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-sm-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-sm-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-8 { margin: 3rem !important; } + .mt-sm-8 { margin-top: 3rem !important; } + .mr-sm-8 { margin-right: 3rem !important; } + .mb-sm-8 { margin-bottom: 3rem !important; } + .ml-sm-8 { margin-left: 3rem !important; } + .mx-sm-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-sm-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-sm-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 31.25rem) { .m-sm-9 { margin: 3.5rem !important; } + .mt-sm-9 { margin-top: 3.5rem !important; } + .mr-sm-9 { margin-right: 3.5rem !important; } + .mb-sm-9 { margin-bottom: 3.5rem !important; } + .ml-sm-9 { margin-left: 3.5rem !important; } + .mx-sm-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-sm-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-sm-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-10 { margin: 4rem !important; } + .mt-sm-10 { margin-top: 4rem !important; } + .mr-sm-10 { margin-right: 4rem !important; } + .mb-sm-10 { margin-bottom: 4rem !important; } + .ml-sm-10 { margin-left: 4rem !important; } + .mx-sm-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-sm-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-sm-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 50rem) { .m-md-0 { margin: 0 !important; } + .mt-md-0 { margin-top: 0 !important; } + .mr-md-0 { margin-right: 0 !important; } + .mb-md-0 { margin-bottom: 0 !important; } + .ml-md-0 { margin-left: 0 !important; } + .mx-md-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-md-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-md-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 50rem) { .m-md-1 { margin: 0.25rem !important; } + .mt-md-1 { margin-top: 0.25rem !important; } + .mr-md-1 { margin-right: 0.25rem !important; } + .mb-md-1 { margin-bottom: 0.25rem !important; } + .ml-md-1 { margin-left: 0.25rem !important; } + .mx-md-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-md-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-md-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 50rem) { .m-md-2 { margin: 0.5rem !important; } + .mt-md-2 { margin-top: 0.5rem !important; } + .mr-md-2 { margin-right: 0.5rem !important; } + .mb-md-2 { margin-bottom: 0.5rem !important; } + .ml-md-2 { margin-left: 0.5rem !important; } + .mx-md-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-md-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-md-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 50rem) { .m-md-3 { margin: 0.75rem !important; } + .mt-md-3 { margin-top: 0.75rem !important; } + .mr-md-3 { margin-right: 0.75rem !important; } + .mb-md-3 { margin-bottom: 0.75rem !important; } + .ml-md-3 { margin-left: 0.75rem !important; } + .mx-md-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-md-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-md-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 50rem) { .m-md-4 { margin: 1rem !important; } + .mt-md-4 { margin-top: 1rem !important; } + .mr-md-4 { margin-right: 1rem !important; } + .mb-md-4 { margin-bottom: 1rem !important; } + .ml-md-4 { margin-left: 1rem !important; } + .mx-md-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-md-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-md-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 50rem) { .m-md-5 { margin: 1.5rem !important; } + .mt-md-5 { margin-top: 1.5rem !important; } + .mr-md-5 { margin-right: 1.5rem !important; } + .mb-md-5 { margin-bottom: 1.5rem !important; } + .ml-md-5 { margin-left: 1.5rem !important; } + .mx-md-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-md-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-md-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 50rem) { .m-md-6 { margin: 2rem !important; } + .mt-md-6 { margin-top: 2rem !important; } + .mr-md-6 { margin-right: 2rem !important; } + .mb-md-6 { margin-bottom: 2rem !important; } + .ml-md-6 { margin-left: 2rem !important; } + .mx-md-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-md-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-md-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 50rem) { .m-md-7 { margin: 2.5rem !important; } + .mt-md-7 { margin-top: 2.5rem !important; } + .mr-md-7 { margin-right: 2.5rem !important; } + .mb-md-7 { margin-bottom: 2.5rem !important; } + .ml-md-7 { margin-left: 2.5rem !important; } + .mx-md-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-md-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-md-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 50rem) { .m-md-8 { margin: 3rem !important; } + .mt-md-8 { margin-top: 3rem !important; } + .mr-md-8 { margin-right: 3rem !important; } + .mb-md-8 { margin-bottom: 3rem !important; } + .ml-md-8 { margin-left: 3rem !important; } + .mx-md-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-md-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-md-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 50rem) { .m-md-9 { margin: 3.5rem !important; } + .mt-md-9 { margin-top: 3.5rem !important; } + .mr-md-9 { margin-right: 3.5rem !important; } + .mb-md-9 { margin-bottom: 3.5rem !important; } + .ml-md-9 { margin-left: 3.5rem !important; } + .mx-md-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-md-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-md-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 50rem) { .m-md-10 { margin: 4rem !important; } + .mt-md-10 { margin-top: 4rem !important; } + .mr-md-10 { margin-right: 4rem !important; } + .mb-md-10 { margin-bottom: 4rem !important; } + .ml-md-10 { margin-left: 4rem !important; } + .mx-md-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-md-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-md-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 66.5rem) { .m-lg-0 { margin: 0 !important; } + .mt-lg-0 { margin-top: 0 !important; } + .mr-lg-0 { margin-right: 0 !important; } + .mb-lg-0 { margin-bottom: 0 !important; } + .ml-lg-0 { margin-left: 0 !important; } + .mx-lg-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-lg-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-lg-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 66.5rem) { .m-lg-1 { margin: 0.25rem !important; } + .mt-lg-1 { margin-top: 0.25rem !important; } + .mr-lg-1 { margin-right: 0.25rem !important; } + .mb-lg-1 { margin-bottom: 0.25rem !important; } + .ml-lg-1 { margin-left: 0.25rem !important; } + .mx-lg-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-lg-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-lg-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 66.5rem) { .m-lg-2 { margin: 0.5rem !important; } + .mt-lg-2 { margin-top: 0.5rem !important; } + .mr-lg-2 { margin-right: 0.5rem !important; } + .mb-lg-2 { margin-bottom: 0.5rem !important; } + .ml-lg-2 { margin-left: 0.5rem !important; } + .mx-lg-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-lg-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-lg-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-3 { margin: 0.75rem !important; } + .mt-lg-3 { margin-top: 0.75rem !important; } + .mr-lg-3 { margin-right: 0.75rem !important; } + .mb-lg-3 { margin-bottom: 0.75rem !important; } + .ml-lg-3 { margin-left: 0.75rem !important; } + .mx-lg-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-lg-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-lg-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 66.5rem) { .m-lg-4 { margin: 1rem !important; } + .mt-lg-4 { margin-top: 1rem !important; } + .mr-lg-4 { margin-right: 1rem !important; } + .mb-lg-4 { margin-bottom: 1rem !important; } + .ml-lg-4 { margin-left: 1rem !important; } + .mx-lg-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-lg-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-lg-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 66.5rem) { .m-lg-5 { margin: 1.5rem !important; } + .mt-lg-5 { margin-top: 1.5rem !important; } + .mr-lg-5 { margin-right: 1.5rem !important; } + .mb-lg-5 { margin-bottom: 1.5rem !important; } + .ml-lg-5 { margin-left: 1.5rem !important; } + .mx-lg-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-lg-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-lg-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-6 { margin: 2rem !important; } + .mt-lg-6 { margin-top: 2rem !important; } + .mr-lg-6 { margin-right: 2rem !important; } + .mb-lg-6 { margin-bottom: 2rem !important; } + .ml-lg-6 { margin-left: 2rem !important; } + .mx-lg-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-lg-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-lg-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 66.5rem) { .m-lg-7 { margin: 2.5rem !important; } + .mt-lg-7 { margin-top: 2.5rem !important; } + .mr-lg-7 { margin-right: 2.5rem !important; } + .mb-lg-7 { margin-bottom: 2.5rem !important; } + .ml-lg-7 { margin-left: 2.5rem !important; } + .mx-lg-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-lg-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-lg-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-8 { margin: 3rem !important; } + .mt-lg-8 { margin-top: 3rem !important; } + .mr-lg-8 { margin-right: 3rem !important; } + .mb-lg-8 { margin-bottom: 3rem !important; } + .ml-lg-8 { margin-left: 3rem !important; } + .mx-lg-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-lg-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-lg-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 66.5rem) { .m-lg-9 { margin: 3.5rem !important; } + .mt-lg-9 { margin-top: 3.5rem !important; } + .mr-lg-9 { margin-right: 3.5rem !important; } + .mb-lg-9 { margin-bottom: 3.5rem !important; } + .ml-lg-9 { margin-left: 3.5rem !important; } + .mx-lg-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-lg-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-lg-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-10 { margin: 4rem !important; } + .mt-lg-10 { margin-top: 4rem !important; } + .mr-lg-10 { margin-right: 4rem !important; } + .mb-lg-10 { margin-bottom: 4rem !important; } + .ml-lg-10 { margin-left: 4rem !important; } + .mx-lg-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-lg-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-lg-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 87.5rem) { .m-xl-0 { margin: 0 !important; } + .mt-xl-0 { margin-top: 0 !important; } + .mr-xl-0 { margin-right: 0 !important; } + .mb-xl-0 { margin-bottom: 0 !important; } + .ml-xl-0 { margin-left: 0 !important; } + .mx-xl-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xl-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xl-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 87.5rem) { .m-xl-1 { margin: 0.25rem !important; } + .mt-xl-1 { margin-top: 0.25rem !important; } + .mr-xl-1 { margin-right: 0.25rem !important; } + .mb-xl-1 { margin-bottom: 0.25rem !important; } + .ml-xl-1 { margin-left: 0.25rem !important; } + .mx-xl-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xl-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xl-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 87.5rem) { .m-xl-2 { margin: 0.5rem !important; } + .mt-xl-2 { margin-top: 0.5rem !important; } + .mr-xl-2 { margin-right: 0.5rem !important; } + .mb-xl-2 { margin-bottom: 0.5rem !important; } + .ml-xl-2 { margin-left: 0.5rem !important; } + .mx-xl-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xl-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xl-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-3 { margin: 0.75rem !important; } + .mt-xl-3 { margin-top: 0.75rem !important; } + .mr-xl-3 { margin-right: 0.75rem !important; } + .mb-xl-3 { margin-bottom: 0.75rem !important; } + .ml-xl-3 { margin-left: 0.75rem !important; } + .mx-xl-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xl-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xl-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 87.5rem) { .m-xl-4 { margin: 1rem !important; } + .mt-xl-4 { margin-top: 1rem !important; } + .mr-xl-4 { margin-right: 1rem !important; } + .mb-xl-4 { margin-bottom: 1rem !important; } + .ml-xl-4 { margin-left: 1rem !important; } + .mx-xl-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xl-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xl-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 87.5rem) { .m-xl-5 { margin: 1.5rem !important; } + .mt-xl-5 { margin-top: 1.5rem !important; } + .mr-xl-5 { margin-right: 1.5rem !important; } + .mb-xl-5 { margin-bottom: 1.5rem !important; } + .ml-xl-5 { margin-left: 1.5rem !important; } + .mx-xl-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xl-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xl-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-6 { margin: 2rem !important; } + .mt-xl-6 { margin-top: 2rem !important; } + .mr-xl-6 { margin-right: 2rem !important; } + .mb-xl-6 { margin-bottom: 2rem !important; } + .ml-xl-6 { margin-left: 2rem !important; } + .mx-xl-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xl-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xl-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 87.5rem) { .m-xl-7 { margin: 2.5rem !important; } + .mt-xl-7 { margin-top: 2.5rem !important; } + .mr-xl-7 { margin-right: 2.5rem !important; } + .mb-xl-7 { margin-bottom: 2.5rem !important; } + .ml-xl-7 { margin-left: 2.5rem !important; } + .mx-xl-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xl-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xl-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-8 { margin: 3rem !important; } + .mt-xl-8 { margin-top: 3rem !important; } + .mr-xl-8 { margin-right: 3rem !important; } + .mb-xl-8 { margin-bottom: 3rem !important; } + .ml-xl-8 { margin-left: 3rem !important; } + .mx-xl-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xl-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xl-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 87.5rem) { .m-xl-9 { margin: 3.5rem !important; } + .mt-xl-9 { margin-top: 3.5rem !important; } + .mr-xl-9 { margin-right: 3.5rem !important; } + .mb-xl-9 { margin-bottom: 3.5rem !important; } + .ml-xl-9 { margin-left: 3.5rem !important; } + .mx-xl-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xl-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xl-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-10 { margin: 4rem !important; } + .mt-xl-10 { margin-top: 4rem !important; } + .mr-xl-10 { margin-right: 4rem !important; } + .mb-xl-10 { margin-bottom: 4rem !important; } + .ml-xl-10 { margin-left: 4rem !important; } + .mx-xl-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xl-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xl-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +.p-0 { padding: 0 !important; } + +.pt-0 { padding-top: 0 !important; } + +.pr-0 { padding-right: 0 !important; } + +.pb-0 { padding-bottom: 0 !important; } + +.pl-0 { padding-left: 0 !important; } + +.px-0 { padding-right: 0 !important; padding-left: 0 !important; } + +.py-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + +.p-1 { padding: 0.25rem !important; } + +.pt-1 { padding-top: 0.25rem !important; } + +.pr-1 { padding-right: 0.25rem !important; } + +.pb-1 { padding-bottom: 0.25rem !important; } + +.pl-1 { padding-left: 0.25rem !important; } + +.px-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + +.py-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + +.p-2 { padding: 0.5rem !important; } + +.pt-2 { padding-top: 0.5rem !important; } + +.pr-2 { padding-right: 0.5rem !important; } + +.pb-2 { padding-bottom: 0.5rem !important; } + +.pl-2 { padding-left: 0.5rem !important; } + +.px-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + +.py-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + +.p-3 { padding: 0.75rem !important; } + +.pt-3 { padding-top: 0.75rem !important; } + +.pr-3 { padding-right: 0.75rem !important; } + +.pb-3 { padding-bottom: 0.75rem !important; } + +.pl-3 { padding-left: 0.75rem !important; } + +.px-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + +.py-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + +.p-4 { padding: 1rem !important; } + +.pt-4 { padding-top: 1rem !important; } + +.pr-4 { padding-right: 1rem !important; } + +.pb-4 { padding-bottom: 1rem !important; } + +.pl-4 { padding-left: 1rem !important; } + +.px-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + +.py-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + +.p-5 { padding: 1.5rem !important; } + +.pt-5 { padding-top: 1.5rem !important; } + +.pr-5 { padding-right: 1.5rem !important; } + +.pb-5 { padding-bottom: 1.5rem !important; } + +.pl-5 { padding-left: 1.5rem !important; } + +.px-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + +.py-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + +.p-6 { padding: 2rem !important; } + +.pt-6 { padding-top: 2rem !important; } + +.pr-6 { padding-right: 2rem !important; } + +.pb-6 { padding-bottom: 2rem !important; } + +.pl-6 { padding-left: 2rem !important; } + +.px-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + +.py-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + +.p-7 { padding: 2.5rem !important; } + +.pt-7 { padding-top: 2.5rem !important; } + +.pr-7 { padding-right: 2.5rem !important; } + +.pb-7 { padding-bottom: 2.5rem !important; } + +.pl-7 { padding-left: 2.5rem !important; } + +.px-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + +.py-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + +.p-8 { padding: 3rem !important; } + +.pt-8 { padding-top: 3rem !important; } + +.pr-8 { padding-right: 3rem !important; } + +.pb-8 { padding-bottom: 3rem !important; } + +.pl-8 { padding-left: 3rem !important; } + +.px-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + +.py-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + +.p-9 { padding: 3.5rem !important; } + +.pt-9 { padding-top: 3.5rem !important; } + +.pr-9 { padding-right: 3.5rem !important; } + +.pb-9 { padding-bottom: 3.5rem !important; } + +.pl-9 { padding-left: 3.5rem !important; } + +.px-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + +.py-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + +.p-10 { padding: 4rem !important; } + +.pt-10 { padding-top: 4rem !important; } + +.pr-10 { padding-right: 4rem !important; } + +.pb-10 { padding-bottom: 4rem !important; } + +.pl-10 { padding-left: 4rem !important; } + +.px-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + +.py-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } + +@media (min-width: 20rem) { .p-xs-0 { padding: 0 !important; } + .pt-xs-0 { padding-top: 0 !important; } + .pr-xs-0 { padding-right: 0 !important; } + .pb-xs-0 { padding-bottom: 0 !important; } + .pl-xs-0 { padding-left: 0 !important; } + .px-xs-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xs-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xs-1 { padding: 0.25rem !important; } + .pt-xs-1 { padding-top: 0.25rem !important; } + .pr-xs-1 { padding-right: 0.25rem !important; } + .pb-xs-1 { padding-bottom: 0.25rem !important; } + .pl-xs-1 { padding-left: 0.25rem !important; } + .px-xs-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xs-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xs-2 { padding: 0.5rem !important; } + .pt-xs-2 { padding-top: 0.5rem !important; } + .pr-xs-2 { padding-right: 0.5rem !important; } + .pb-xs-2 { padding-bottom: 0.5rem !important; } + .pl-xs-2 { padding-left: 0.5rem !important; } + .px-xs-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xs-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xs-3 { padding: 0.75rem !important; } + .pt-xs-3 { padding-top: 0.75rem !important; } + .pr-xs-3 { padding-right: 0.75rem !important; } + .pb-xs-3 { padding-bottom: 0.75rem !important; } + .pl-xs-3 { padding-left: 0.75rem !important; } + .px-xs-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xs-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xs-4 { padding: 1rem !important; } + .pt-xs-4 { padding-top: 1rem !important; } + .pr-xs-4 { padding-right: 1rem !important; } + .pb-xs-4 { padding-bottom: 1rem !important; } + .pl-xs-4 { padding-left: 1rem !important; } + .px-xs-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xs-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xs-5 { padding: 1.5rem !important; } + .pt-xs-5 { padding-top: 1.5rem !important; } + .pr-xs-5 { padding-right: 1.5rem !important; } + .pb-xs-5 { padding-bottom: 1.5rem !important; } + .pl-xs-5 { padding-left: 1.5rem !important; } + .px-xs-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xs-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xs-6 { padding: 2rem !important; } + .pt-xs-6 { padding-top: 2rem !important; } + .pr-xs-6 { padding-right: 2rem !important; } + .pb-xs-6 { padding-bottom: 2rem !important; } + .pl-xs-6 { padding-left: 2rem !important; } + .px-xs-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xs-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xs-7 { padding: 2.5rem !important; } + .pt-xs-7 { padding-top: 2.5rem !important; } + .pr-xs-7 { padding-right: 2.5rem !important; } + .pb-xs-7 { padding-bottom: 2.5rem !important; } + .pl-xs-7 { padding-left: 2.5rem !important; } + .px-xs-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xs-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xs-8 { padding: 3rem !important; } + .pt-xs-8 { padding-top: 3rem !important; } + .pr-xs-8 { padding-right: 3rem !important; } + .pb-xs-8 { padding-bottom: 3rem !important; } + .pl-xs-8 { padding-left: 3rem !important; } + .px-xs-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xs-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xs-9 { padding: 3.5rem !important; } + .pt-xs-9 { padding-top: 3.5rem !important; } + .pr-xs-9 { padding-right: 3.5rem !important; } + .pb-xs-9 { padding-bottom: 3.5rem !important; } + .pl-xs-9 { padding-left: 3.5rem !important; } + .px-xs-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xs-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xs-10 { padding: 4rem !important; } + .pt-xs-10 { padding-top: 4rem !important; } + .pr-xs-10 { padding-right: 4rem !important; } + .pb-xs-10 { padding-bottom: 4rem !important; } + .pl-xs-10 { padding-left: 4rem !important; } + .px-xs-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xs-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 31.25rem) { .p-sm-0 { padding: 0 !important; } + .pt-sm-0 { padding-top: 0 !important; } + .pr-sm-0 { padding-right: 0 !important; } + .pb-sm-0 { padding-bottom: 0 !important; } + .pl-sm-0 { padding-left: 0 !important; } + .px-sm-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-sm-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-sm-1 { padding: 0.25rem !important; } + .pt-sm-1 { padding-top: 0.25rem !important; } + .pr-sm-1 { padding-right: 0.25rem !important; } + .pb-sm-1 { padding-bottom: 0.25rem !important; } + .pl-sm-1 { padding-left: 0.25rem !important; } + .px-sm-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-sm-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-sm-2 { padding: 0.5rem !important; } + .pt-sm-2 { padding-top: 0.5rem !important; } + .pr-sm-2 { padding-right: 0.5rem !important; } + .pb-sm-2 { padding-bottom: 0.5rem !important; } + .pl-sm-2 { padding-left: 0.5rem !important; } + .px-sm-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-sm-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-sm-3 { padding: 0.75rem !important; } + .pt-sm-3 { padding-top: 0.75rem !important; } + .pr-sm-3 { padding-right: 0.75rem !important; } + .pb-sm-3 { padding-bottom: 0.75rem !important; } + .pl-sm-3 { padding-left: 0.75rem !important; } + .px-sm-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-sm-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-sm-4 { padding: 1rem !important; } + .pt-sm-4 { padding-top: 1rem !important; } + .pr-sm-4 { padding-right: 1rem !important; } + .pb-sm-4 { padding-bottom: 1rem !important; } + .pl-sm-4 { padding-left: 1rem !important; } + .px-sm-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-sm-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-sm-5 { padding: 1.5rem !important; } + .pt-sm-5 { padding-top: 1.5rem !important; } + .pr-sm-5 { padding-right: 1.5rem !important; } + .pb-sm-5 { padding-bottom: 1.5rem !important; } + .pl-sm-5 { padding-left: 1.5rem !important; } + .px-sm-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-sm-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-sm-6 { padding: 2rem !important; } + .pt-sm-6 { padding-top: 2rem !important; } + .pr-sm-6 { padding-right: 2rem !important; } + .pb-sm-6 { padding-bottom: 2rem !important; } + .pl-sm-6 { padding-left: 2rem !important; } + .px-sm-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-sm-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-sm-7 { padding: 2.5rem !important; } + .pt-sm-7 { padding-top: 2.5rem !important; } + .pr-sm-7 { padding-right: 2.5rem !important; } + .pb-sm-7 { padding-bottom: 2.5rem !important; } + .pl-sm-7 { padding-left: 2.5rem !important; } + .px-sm-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-sm-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-sm-8 { padding: 3rem !important; } + .pt-sm-8 { padding-top: 3rem !important; } + .pr-sm-8 { padding-right: 3rem !important; } + .pb-sm-8 { padding-bottom: 3rem !important; } + .pl-sm-8 { padding-left: 3rem !important; } + .px-sm-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-sm-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-sm-9 { padding: 3.5rem !important; } + .pt-sm-9 { padding-top: 3.5rem !important; } + .pr-sm-9 { padding-right: 3.5rem !important; } + .pb-sm-9 { padding-bottom: 3.5rem !important; } + .pl-sm-9 { padding-left: 3.5rem !important; } + .px-sm-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-sm-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-sm-10 { padding: 4rem !important; } + .pt-sm-10 { padding-top: 4rem !important; } + .pr-sm-10 { padding-right: 4rem !important; } + .pb-sm-10 { padding-bottom: 4rem !important; } + .pl-sm-10 { padding-left: 4rem !important; } + .px-sm-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-sm-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 50rem) { .p-md-0 { padding: 0 !important; } + .pt-md-0 { padding-top: 0 !important; } + .pr-md-0 { padding-right: 0 !important; } + .pb-md-0 { padding-bottom: 0 !important; } + .pl-md-0 { padding-left: 0 !important; } + .px-md-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-md-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-md-1 { padding: 0.25rem !important; } + .pt-md-1 { padding-top: 0.25rem !important; } + .pr-md-1 { padding-right: 0.25rem !important; } + .pb-md-1 { padding-bottom: 0.25rem !important; } + .pl-md-1 { padding-left: 0.25rem !important; } + .px-md-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-md-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-md-2 { padding: 0.5rem !important; } + .pt-md-2 { padding-top: 0.5rem !important; } + .pr-md-2 { padding-right: 0.5rem !important; } + .pb-md-2 { padding-bottom: 0.5rem !important; } + .pl-md-2 { padding-left: 0.5rem !important; } + .px-md-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-md-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-md-3 { padding: 0.75rem !important; } + .pt-md-3 { padding-top: 0.75rem !important; } + .pr-md-3 { padding-right: 0.75rem !important; } + .pb-md-3 { padding-bottom: 0.75rem !important; } + .pl-md-3 { padding-left: 0.75rem !important; } + .px-md-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-md-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-md-4 { padding: 1rem !important; } + .pt-md-4 { padding-top: 1rem !important; } + .pr-md-4 { padding-right: 1rem !important; } + .pb-md-4 { padding-bottom: 1rem !important; } + .pl-md-4 { padding-left: 1rem !important; } + .px-md-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-md-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-md-5 { padding: 1.5rem !important; } + .pt-md-5 { padding-top: 1.5rem !important; } + .pr-md-5 { padding-right: 1.5rem !important; } + .pb-md-5 { padding-bottom: 1.5rem !important; } + .pl-md-5 { padding-left: 1.5rem !important; } + .px-md-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-md-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-md-6 { padding: 2rem !important; } + .pt-md-6 { padding-top: 2rem !important; } + .pr-md-6 { padding-right: 2rem !important; } + .pb-md-6 { padding-bottom: 2rem !important; } + .pl-md-6 { padding-left: 2rem !important; } + .px-md-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-md-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-md-7 { padding: 2.5rem !important; } + .pt-md-7 { padding-top: 2.5rem !important; } + .pr-md-7 { padding-right: 2.5rem !important; } + .pb-md-7 { padding-bottom: 2.5rem !important; } + .pl-md-7 { padding-left: 2.5rem !important; } + .px-md-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-md-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-md-8 { padding: 3rem !important; } + .pt-md-8 { padding-top: 3rem !important; } + .pr-md-8 { padding-right: 3rem !important; } + .pb-md-8 { padding-bottom: 3rem !important; } + .pl-md-8 { padding-left: 3rem !important; } + .px-md-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-md-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-md-9 { padding: 3.5rem !important; } + .pt-md-9 { padding-top: 3.5rem !important; } + .pr-md-9 { padding-right: 3.5rem !important; } + .pb-md-9 { padding-bottom: 3.5rem !important; } + .pl-md-9 { padding-left: 3.5rem !important; } + .px-md-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-md-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-md-10 { padding: 4rem !important; } + .pt-md-10 { padding-top: 4rem !important; } + .pr-md-10 { padding-right: 4rem !important; } + .pb-md-10 { padding-bottom: 4rem !important; } + .pl-md-10 { padding-left: 4rem !important; } + .px-md-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-md-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 66.5rem) { .p-lg-0 { padding: 0 !important; } + .pt-lg-0 { padding-top: 0 !important; } + .pr-lg-0 { padding-right: 0 !important; } + .pb-lg-0 { padding-bottom: 0 !important; } + .pl-lg-0 { padding-left: 0 !important; } + .px-lg-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-lg-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-lg-1 { padding: 0.25rem !important; } + .pt-lg-1 { padding-top: 0.25rem !important; } + .pr-lg-1 { padding-right: 0.25rem !important; } + .pb-lg-1 { padding-bottom: 0.25rem !important; } + .pl-lg-1 { padding-left: 0.25rem !important; } + .px-lg-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-lg-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-lg-2 { padding: 0.5rem !important; } + .pt-lg-2 { padding-top: 0.5rem !important; } + .pr-lg-2 { padding-right: 0.5rem !important; } + .pb-lg-2 { padding-bottom: 0.5rem !important; } + .pl-lg-2 { padding-left: 0.5rem !important; } + .px-lg-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-lg-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-lg-3 { padding: 0.75rem !important; } + .pt-lg-3 { padding-top: 0.75rem !important; } + .pr-lg-3 { padding-right: 0.75rem !important; } + .pb-lg-3 { padding-bottom: 0.75rem !important; } + .pl-lg-3 { padding-left: 0.75rem !important; } + .px-lg-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-lg-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-lg-4 { padding: 1rem !important; } + .pt-lg-4 { padding-top: 1rem !important; } + .pr-lg-4 { padding-right: 1rem !important; } + .pb-lg-4 { padding-bottom: 1rem !important; } + .pl-lg-4 { padding-left: 1rem !important; } + .px-lg-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-lg-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-lg-5 { padding: 1.5rem !important; } + .pt-lg-5 { padding-top: 1.5rem !important; } + .pr-lg-5 { padding-right: 1.5rem !important; } + .pb-lg-5 { padding-bottom: 1.5rem !important; } + .pl-lg-5 { padding-left: 1.5rem !important; } + .px-lg-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-lg-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-lg-6 { padding: 2rem !important; } + .pt-lg-6 { padding-top: 2rem !important; } + .pr-lg-6 { padding-right: 2rem !important; } + .pb-lg-6 { padding-bottom: 2rem !important; } + .pl-lg-6 { padding-left: 2rem !important; } + .px-lg-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-lg-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-lg-7 { padding: 2.5rem !important; } + .pt-lg-7 { padding-top: 2.5rem !important; } + .pr-lg-7 { padding-right: 2.5rem !important; } + .pb-lg-7 { padding-bottom: 2.5rem !important; } + .pl-lg-7 { padding-left: 2.5rem !important; } + .px-lg-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-lg-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-lg-8 { padding: 3rem !important; } + .pt-lg-8 { padding-top: 3rem !important; } + .pr-lg-8 { padding-right: 3rem !important; } + .pb-lg-8 { padding-bottom: 3rem !important; } + .pl-lg-8 { padding-left: 3rem !important; } + .px-lg-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-lg-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-lg-9 { padding: 3.5rem !important; } + .pt-lg-9 { padding-top: 3.5rem !important; } + .pr-lg-9 { padding-right: 3.5rem !important; } + .pb-lg-9 { padding-bottom: 3.5rem !important; } + .pl-lg-9 { padding-left: 3.5rem !important; } + .px-lg-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-lg-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-lg-10 { padding: 4rem !important; } + .pt-lg-10 { padding-top: 4rem !important; } + .pr-lg-10 { padding-right: 4rem !important; } + .pb-lg-10 { padding-bottom: 4rem !important; } + .pl-lg-10 { padding-left: 4rem !important; } + .px-lg-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-lg-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 87.5rem) { .p-xl-0 { padding: 0 !important; } + .pt-xl-0 { padding-top: 0 !important; } + .pr-xl-0 { padding-right: 0 !important; } + .pb-xl-0 { padding-bottom: 0 !important; } + .pl-xl-0 { padding-left: 0 !important; } + .px-xl-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xl-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xl-1 { padding: 0.25rem !important; } + .pt-xl-1 { padding-top: 0.25rem !important; } + .pr-xl-1 { padding-right: 0.25rem !important; } + .pb-xl-1 { padding-bottom: 0.25rem !important; } + .pl-xl-1 { padding-left: 0.25rem !important; } + .px-xl-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xl-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xl-2 { padding: 0.5rem !important; } + .pt-xl-2 { padding-top: 0.5rem !important; } + .pr-xl-2 { padding-right: 0.5rem !important; } + .pb-xl-2 { padding-bottom: 0.5rem !important; } + .pl-xl-2 { padding-left: 0.5rem !important; } + .px-xl-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xl-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xl-3 { padding: 0.75rem !important; } + .pt-xl-3 { padding-top: 0.75rem !important; } + .pr-xl-3 { padding-right: 0.75rem !important; } + .pb-xl-3 { padding-bottom: 0.75rem !important; } + .pl-xl-3 { padding-left: 0.75rem !important; } + .px-xl-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xl-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xl-4 { padding: 1rem !important; } + .pt-xl-4 { padding-top: 1rem !important; } + .pr-xl-4 { padding-right: 1rem !important; } + .pb-xl-4 { padding-bottom: 1rem !important; } + .pl-xl-4 { padding-left: 1rem !important; } + .px-xl-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xl-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xl-5 { padding: 1.5rem !important; } + .pt-xl-5 { padding-top: 1.5rem !important; } + .pr-xl-5 { padding-right: 1.5rem !important; } + .pb-xl-5 { padding-bottom: 1.5rem !important; } + .pl-xl-5 { padding-left: 1.5rem !important; } + .px-xl-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xl-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xl-6 { padding: 2rem !important; } + .pt-xl-6 { padding-top: 2rem !important; } + .pr-xl-6 { padding-right: 2rem !important; } + .pb-xl-6 { padding-bottom: 2rem !important; } + .pl-xl-6 { padding-left: 2rem !important; } + .px-xl-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xl-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xl-7 { padding: 2.5rem !important; } + .pt-xl-7 { padding-top: 2.5rem !important; } + .pr-xl-7 { padding-right: 2.5rem !important; } + .pb-xl-7 { padding-bottom: 2.5rem !important; } + .pl-xl-7 { padding-left: 2.5rem !important; } + .px-xl-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xl-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xl-8 { padding: 3rem !important; } + .pt-xl-8 { padding-top: 3rem !important; } + .pr-xl-8 { padding-right: 3rem !important; } + .pb-xl-8 { padding-bottom: 3rem !important; } + .pl-xl-8 { padding-left: 3rem !important; } + .px-xl-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xl-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xl-9 { padding: 3.5rem !important; } + .pt-xl-9 { padding-top: 3.5rem !important; } + .pr-xl-9 { padding-right: 3.5rem !important; } + .pb-xl-9 { padding-bottom: 3.5rem !important; } + .pl-xl-9 { padding-left: 3.5rem !important; } + .px-xl-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xl-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xl-10 { padding: 4rem !important; } + .pt-xl-10 { padding-top: 4rem !important; } + .pr-xl-10 { padding-right: 4rem !important; } + .pb-xl-10 { padding-bottom: 4rem !important; } + .pl-xl-10 { padding-left: 4rem !important; } + .px-xl-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xl-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media print { .site-footer, .site-button, #edit-this-page, #back-to-top, .site-nav, .main-header { display: none !important; } + .side-bar { width: 100%; height: auto; border-right: 0 !important; } + .site-header { border-bottom: 1px solid #eeebee; } + .site-title { font-size: 1rem !important; font-weight: 700 !important; } + .text-small { font-size: 8pt !important; } + pre.highlight { border: 1px solid #eeebee; } + .main { max-width: none; margin-left: 0; } } +a.skip-to-main { left: -999px; position: absolute; top: auto; width: 1px; height: 1px; overflow: hidden; z-index: -999; } + +a.skip-to-main:focus, a.skip-to-main:active { color: #7253ed; background-color: #fff; left: auto; top: auto; width: 30%; height: auto; overflow: auto; margin: 10px 35%; padding: 5px; border-radius: 15px; border: 4px solid #5e41d0; text-align: center; font-size: 1.2em; z-index: 999; } + +div.opaque { background-color: #fff; } + +p.note, blockquote.note { background: #f5f7f9; border-left: 4px solid #2ce5b5; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note > .note-title, blockquote.note > .note-title { color: #2ce5b5; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.note-title, blockquote.note-title { background: #f5f7f9; border-left: 4px solid #2ce5b5; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note-title > p:first-child, blockquote.note-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #2ce5b5; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.note { margin-left: 0; margin-right: 0; } +blockquote.note > p:first-child { margin-top: 0; } +blockquote.note > p:last-child { margin-bottom: 0; } + +blockquote.note-title { margin-left: 0; margin-right: 0; } +blockquote.note-title > p:nth-child(2) { margin-top: 0; } +blockquote.note-title > p:last-child { margin-bottom: 0; } + +p.warning, blockquote.warning { background: rgba(247, 126, 126, 0.2); border-left: 4px solid #dd2e2e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning::before, blockquote.warning::before { color: #dd2e2e; content: "⚠️ Warning ⚠️"; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } +p.warning > .warning-title, blockquote.warning > .warning-title { color: #dd2e2e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.warning-title, blockquote.warning-title { background: rgba(247, 126, 126, 0.2); border-left: 4px solid #dd2e2e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning-title > p:first-child, blockquote.warning-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #dd2e2e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.warning { margin-left: 0; margin-right: 0; } +blockquote.warning > p:first-child { margin-top: 0; } +blockquote.warning > p:last-child { margin-bottom: 0; } + +blockquote.warning-title { margin-left: 0; margin-right: 0; } +blockquote.warning-title > p:nth-child(2) { margin-top: 0; } +blockquote.warning-title > p:last-child { margin-bottom: 0; } + +p.tip, blockquote.tip { background: rgba(65, 214, 147, 0.2); border-left: 4px solid #026e57; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip > .tip-title, blockquote.tip > .tip-title { color: #026e57; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.tip-title, blockquote.tip-title { background: rgba(65, 214, 147, 0.2); border-left: 4px solid #026e57; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip-title > p:first-child, blockquote.tip-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #026e57; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.tip { margin-left: 0; margin-right: 0; } +blockquote.tip > p:first-child { margin-top: 0; } +blockquote.tip > p:last-child { margin-bottom: 0; } + +blockquote.tip-title { margin-left: 0; margin-right: 0; } +blockquote.tip-title > p:nth-child(2) { margin-top: 0; } +blockquote.tip-title > p:last-child { margin-bottom: 0; } + +p.fubar, blockquote.fubar { background: rgba(114, 83, 237, 0.2); border-left: 4px solid #381885; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar > .fubar-title, blockquote.fubar > .fubar-title { color: #381885; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.fubar-title, blockquote.fubar-title { background: rgba(114, 83, 237, 0.2); border-left: 4px solid #381885; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar-title > p:first-child, blockquote.fubar-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #381885; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.fubar { margin-left: 0; margin-right: 0; } +blockquote.fubar > p:first-child { margin-top: 0; } +blockquote.fubar > p:last-child { margin-bottom: 0; } + +blockquote.fubar-title { margin-left: 0; margin-right: 0; } +blockquote.fubar-title > p:nth-child(2) { margin-top: 0; } +blockquote.fubar-title > p:last-child { margin-bottom: 0; } + +@font-face { font-family: GalanoGrotesque-ExtraLight; src: local(GalanoGrotesque-ExtraLight), url("../fonts/Rene Bieder - Galano Grotesque ExtraLight.otf"); } +@font-face { font-family: GalanoGrotesque; src: local(GalanoGrotesque-Regular), url("../fonts/Rene Bieder - Galano Grotesque.otf"); } +@font-face { font-family: GalanoGrotesque-Bold; src: local(GalanoGrotesque-Bold), url("../fonts/Rene Bieder - Galano Grotesque Bold.otf"); } +@font-face { font-family: GalanoGrotesque-Medium; src: local(Rene-Bieder-Galano-Grotesque-Medium), url("../fonts/Rene Bieder - Galano Grotesque Medium.otf"); } +html { scroll-padding-top: 90px; } + +body { font-family: GalanoGrotesque; color: #000000; } +body .body-wrapper, body footer { display: flex; } +body .body-wrapper .main, body .body-wrapper .footer-main, body footer .main, body footer .footer-main { width: 75%; max-width: 75%; margin-left: 0; float: right; position: relative; } +body .body-wrapper .main .main-content-wrap, body .body-wrapper .footer-main .main-content-wrap, body footer .main .main-content-wrap, body footer .footer-main .main-content-wrap { max-width: 1300px; padding: 80px; } +body .body-wrapper .side-bar, body .body-wrapper .footer-sidebar, body footer .side-bar, body footer .footer-sidebar { width: 25%; max-width: 25%; position: relative; height: auto; background-color: #f5f7f9; } +body a { color: #279890; } +body p, body li { font-family: GalanoGrotesque; font-size: 1em; } +body h1 { margin-top: 40px !important; font-family: GalanoGrotesque-Medium; font-size: 42px !important; color: #000000; } +body h2 { margin-top: 40px !important; font-size: 28px !important; font-family: GalanoGrotesque-Medium; } +body h2#table-of-contents { font-family: GalanoGrotesque-Medium; color: #ffffff; font-size: 14px !important; } +body h3 { margin-top: 25px !important; font-size: 18px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body h4 { margin-top: 25px !important; font-size: 16px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body .text-delta { text-transform: none; width: 100%; display: block; background-color: #279890; color: #ffffff; border-radius: 2px 2px 0 0; font-size: 14px !important; padding: 10px 15px 8px 15px; } +body .text-delta + ul { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body .text-delta + ul li { text-decoration: underline; } +body .text-delta + ul li::before { content: none; } +body .text-delta + ul li a:hover { background-image: none; } +body .text-delta + ul li a { white-space: normal; } +body #markdown-toc { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body #markdown-toc li::before { display: none; } +body #markdown-toc li a { color: #279890; font-size: 1em; background-image: none; white-space: normal; } +body #markdown-toc li a:hover { background-image: none; color: #279890; } +body #markdown-toc li ol, body #markdown-toc li ul { border: none; } +body #markdown-toc li ol li a, body #markdown-toc li ul li a { padding-bottom: 0; } +body div.highlighter-rouge pre.highlight { padding: 10px; } +body .highlight code { background-color: transparent; } +body .btn { padding: 6px 14px; color: #2f3433; font-family: GalanoGrotesque; border-radius: 4px; background: #ffffff; border: solid 2px #2f3433; box-shadow: none; } +body .btn:hover { border-color: #279890; color: #279890; background: #ffffff; } +body .btn-green { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; background-color: #279890; border: 2px solid #279890; } +body .btn-green:hover { background-color: #ffffff; color: #279890; } +body .main-content h1:hover a.anchor-heading, body .main-content h2:hover a.anchor-heading, body .main-content h3:hover a.anchor-heading, body .main-content h4:hover a.anchor-heading, body .main-content h5:hover a.anchor-heading, body .main-content h6:hover a.anchor-heading { left: -1.7rem; background-image: url("../../assets/icons/anchor.svg"); width: 24px; height: 24px; background-repeat: no-repeat; background-size: contain; background-position: center center; } +body .main-content h1:hover a.anchor-heading svg, body .main-content h2:hover a.anchor-heading svg, body .main-content h3:hover a.anchor-heading svg, body .main-content h4:hover a.anchor-heading svg, body .main-content h5:hover a.anchor-heading svg, body .main-content h6:hover a.anchor-heading svg { display: none; } +body .main-content h1:hover a.anchor-heading:hover, body .main-content h2:hover a.anchor-heading:hover, body .main-content h3:hover a.anchor-heading:hover, body .main-content h4:hover a.anchor-heading:hover, body .main-content h5:hover a.anchor-heading:hover, body .main-content h6:hover a.anchor-heading:hover { background-image: url("../../assets/icons/anchor green.svg"); } +body .main-content .toc-block { display: table-cell; } + +.main .main-header, .side-bar .site-header { background-color: white; height: 80px; max-height: 80px; min-height: 80px; border: 0; position: fixed; z-index: 2; } + +.main .main-header { width: 75%; } +.main .main-header nav.aux-nav { width: 100%; padding: 0; max-width: 1080px; } +.main .main-header nav.aux-nav .aux-nav-list { justify-content: flex-end; padding: 20px 80px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #000000; margin: 0 4px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button:hover { background-image: none; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item:hover a, .main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.active a { color: #279890; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; margin: 0 0 0 20px; padding: 18px; border-radius: 6px; background-color: #f24886; border: 2px solid #f24886; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button:hover { background-color: #b62d55; border: 2px solid #b62d55; color: #ffffff; } + +.side-bar:after { content: ' '; position: fixed; width: 100vw; top: 80px; box-shadow: 0 0 6px 2px rgba(0, 0, 0, 0.23); left: 0; right: 0; z-index: 0; } + +.side-bar .site-header { width: 25%; justify-content: flex-end; top: 0; } +.side-bar .site-header a.site-title { max-width: 290px; padding: 0; } +.side-bar .site-header a.site-title .site-logo { width: 150px; height: inherit; } +.side-bar .site-header a.site-title:hover { background-image: none; background-color: transparent; } + +.site-logo { width: 135px; height: 30px; } + +.side-bar { padding: 80px 0; border: 0; display: block; z-index: 1; } +.side-bar nav.site-nav, .side-bar nav.aux-nav { padding: 0; width: 100%; display: initial; } +.side-bar nav.site-nav .nav-list, .side-bar nav.site-nav .aux-nav-list, .side-bar nav.aux-nav .nav-list, .side-bar nav.aux-nav .aux-nav-list { width: 100%; padding: 0; margin-top: 30px; } +.side-bar nav.site-nav .nav-list .nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item { padding: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .nav-list .nav-list-item:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active:hover { background-color: transparent; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li { padding-right: 0; padding-left: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.header, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .nav-list .nav-list-item.header, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header { padding: 12px 40px 12px calc(100% - 290px); color: #279890; font-family: GalanoGrotesque-Bold; font-size: 18px !important; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:hover { background-color: transparent; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type) { padding-top: 36px; border-top: 1px solid #dadada; margin-top: 20px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link { padding: 12px 40px 12px calc(100% - 290px); font-size: 14px !important; color: #576489; font-family: GalanoGrotesque; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover { background-color: #ffffff; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active { font-family: GalanoGrotesque-Medium !important; color: #279890 !important; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander { right: 40px; width: 22px; height: 22px; padding: 0; margin: 14px 0 0 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg { color: #000000; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover { background-color: none; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list { margin-top: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item { margin-left: 20px; font-size: 13px !important; letter-spacing: 0.05px; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item { margin-left: 30px; color: #949494; font-size: 12px !important; } +.side-bar nav.site-nav .nav-version, .side-bar nav.aux-nav .nav-version { display: flex; align-items: center; justify-content: center; padding: 12px 40px 12px calc(100% - 290px); text-transform: none; } +.side-bar nav.site-nav .nav-version select, .side-bar nav.aux-nav .nav-version select { height: 2em; margin: 0 0 0 10px; flex-grow: 1; } +.side-bar nav.site-nav .search, .side-bar nav.aux-nav .search { margin-top: 35px; margin-left: calc(100% - 290px); margin-right: 40px; max-width: 250px !important; max-height: 40px !important; width: 250px !important; z-index: 1; } +.side-bar nav.site-nav .search .search-input, .side-bar nav.aux-nav .search .search-input { background-color: #f5f7f9; border: 1px solid #dadada; border-radius: 6px; color: #6e7171; font-family: GalanoGrotesque-Medium; } +.side-bar nav.site-nav .search .search-input:focus, .side-bar nav.site-nav .search .search-input:active, .side-bar nav.aux-nav .search .search-input:focus, .side-bar nav.aux-nav .search .search-input:active { background-color: #f5f7f9; color: #6e7171; } +.side-bar nav.site-nav .search .search-input .search-icon, .side-bar nav.aux-nav .search .search-input .search-icon { color: #949494; } +.side-bar nav.site-nav .search .search-input:focus + .search-label .search-icon, .side-bar nav.aux-nav .search .search-input:focus + .search-label .search-icon { color: #949494; } +.side-bar nav.site-nav .mobile-menu, .side-bar nav.aux-nav .mobile-menu { display: none; } + +.label { text-transform: none !important; margin-left: 0 !important; margin-right: 0 !important; } + +.search-active .main { position: inherit; right: inherit; left: inherit; } + +.search-overlay { display: none; } + +.search-result-doc .search-result-icon { color: #279890; } + +footer { height: 360px; } +footer .footer-main { padding: 80px; } +footer .footer-main ul { list-style: none; padding: 0; margin: 0; } +footer .footer-main ul li { display: inline-block; padding: 0; margin: 0; } +footer .footer-main .row { display: flex; } +footer .footer-main .row .left { width: 70%; } +footer .footer-main .row .right { width: 30%; display: flex; justify-content: flex-end; } +footer .footer-main .row .site-title { padding: 0; margin-bottom: 30px; } +footer .footer-main .row .site-title .site-logo { height: 45px; } +footer .footer-main .row .footer-links li { margin-right: 40px; } +footer .footer-main .row .footer-links li a { padding: 0; color: #000000; font-size: 18px; } +footer .footer-main .row .footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a, footer .footer-main .row .bottom-footer-links li a { padding: 0 15px; font-size: 16px; color: #949494; } +footer .footer-main .row .footer-social li a:last-of-type, footer .footer-main .row .bottom-footer-links li a:last-of-type { padding-right: 0; } +footer .footer-main .row .footer-social li a:hover, footer .footer-main .row .bottom-footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a img.hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.no-hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.hover { display: block; } +footer .footer-main .top-border { padding-top: 30px; margin-top: 30px; border-top: 1px solid #dadada; } +footer .footer-main a:hover { background-color: transparent !important; background-image: none !important; } +footer .footer-sidebar { z-index: -1; } + +@media (max-width: 799px) { body { padding-bottom: 0; } + body .body-wrapper { display: block; } + body .body-wrapper .side-bar { width: 100%; max-width: 100%; padding: 0; } + body .body-wrapper .side-bar .site-header { width: 100%; flex-direction: row-reverse; justify-content: flex-end; } + body .body-wrapper .side-bar .site-header .site-button { padding: 11px 30px 0; width: 24px; height: 24px; background-repeat: no-repeat; background-size: auto; background-position: center center; background-image: url("../../assets/icons/menu.svg"); } + body .body-wrapper .side-bar .site-header .site-button:hover, body .body-wrapper .side-bar .site-header .site-button:active, body .body-wrapper .side-bar .site-header .site-button:focus { background-color: transparent; } + body .body-wrapper .side-bar .site-header .site-button.nav-open { background-image: url("../../assets/icons/close.svg"); } + body .body-wrapper .side-bar .site-header .site-button svg { display: none; } + body .body-wrapper .side-bar nav { display: none; overflow: scroll; background: white; } + body .body-wrapper .side-bar nav.nav-open { display: block; position: fixed; top: 0; height: 100vh; } + body .body-wrapper .side-bar nav.nav-open .search { position: relative; margin-top: 120px; margin-left: 30px; margin-right: 30px; padding: 0; } + body .body-wrapper .side-bar nav.nav-open .search .search-results { display: block !important; max-height: 60vh !important; } + body .body-wrapper .side-bar nav.nav-open .search .search-input { background-color: #ffffff; } + body .body-wrapper .side-bar nav.nav-open .search .search-input-wrap { box-shadow: none; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item { padding-left: 30px !important; padding-right: 30px !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a { padding-left: 0 !important; padding-right: 0 !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a.active { background: unset; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list { margin-top: 0; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list a.nav-list-link, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list a.nav-list-link { text-indent: 0; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu { display: block; border-top: 1px solid #dadada; margin-top: 18px; padding-bottom: 80px; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav { display: block; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list { flex-direction: column; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list a.site-button { font-size: 18px !important; color: #000000; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list .button .site-button { border-radius: 6px; border: solid 2px #279890; background-color: #ffffff; color: #279890; padding: 10px 20px !important; display: inline-block; margin-top: 10px; } + body .body-wrapper .main { width: 100%; max-width: 100%; float: none; } + body .body-wrapper .main .main-header { display: none; } + body .body-wrapper .main .main-content a { white-space: normal; } + body .body-wrapper .main .main-content-wrap { padding: 80px 30px; } + body footer { border-top: 1px solid #dadada; height: auto; } + body footer .footer-sidebar { display: none; } + body footer .footer-main { width: 100%; max-width: 100%; padding: 60px 30px; float: none; } + body footer .footer-main .row { flex-direction: column; } + body footer .footer-main .row .right, body footer .footer-main .row .left { width: 100%; display: block; } + body footer .footer-main .row .site-title { margin-bottom: 40px; } + body footer .footer-main .row .footer-links { display: flex; flex-wrap: wrap; margin-bottom: 20px; } + body footer .footer-main .row .footer-links li { margin: 0 0 30px; width: 50%; } + body footer .footer-main .row .footer-social { width: 100%; display: flex; justify-content: space-between; } + body footer .footer-main .row .footer-social li a { padding: 0; } + body footer .footer-main .row .footer-social li a img { width: 34px; height: 34px; } + body footer .footer-main .row .bottom-footer-links li { display: block; margin-top: 30px; } + body footer .footer-main .row .bottom-footer-links li a { padding: 0; } + body footer .footer-main .row .top-border { padding-top: 40px; margin-top: 40px; } } +.copy-code-container { position: absolute; right: 0.3em; } + +.copy-code-button { align-items: center; justify-content: center; border: none; cursor: pointer; font-size: 1rem; color: #000000; padding: 0.4em 0.5em; background-color: transparent; outline: none; } +.copy-code-button:hover { color: #279890 !important; } + +.highlighter-rouge { position: relative; } + +.language-shell.highlighter-rouge pre.highlight::before, .language-sh.highlighter-rouge pre.highlight::before, .language-bash.highlighter-rouge pre.highlight::before { content: "$ "; color: #000000; font-weight: 400; font-family: monospace; font-size: 0.75em; } + +th, td { background-color: transparent; } + +.main-content ul.ui-tabs-nav > li::before { content: ""; } + +.tabs { background: transparent; } +.tabs.ui-widget .copy-code-button { font-family: "Font Awesome 6 Free" !important; } + +.tabs .ui-widget-header { background: transparent; border: none; border-bottom: 1px solid #d3d3d3; -moz-border-radius: 0px; -webkit-border-radius: 0px; border-radius: 0px; } + +.tabs .ui-tabs-nav .ui-state-default { background: transparent; border: none; } + +.tabs .ui-tabs-nav .ui-state-active { border: none; } + +.tabs .ui-tabs-nav .ui-state-default a { color: #c0c0c0; } + +.tabs .ui-tabs-nav .ui-state-active a { color: #279890; } + +/*overridden*/ +div.highlighter-rouge { border-radius: 4px; } +div.highlighter-rouge:hover > button { cursor: pointer; opacity: 0.5; } + +.highlighter-rouge button.copy-code-button { color: #000000; } + +/* Tooltip container */ +.tooltip { position: relative; display: inline-block; border-bottom: 0; } +.tooltip > a, .tooltip span { text-decoration: underline; text-decoration-style: dotted; background-image: none; } +.tooltip .tooltiptext { visibility: hidden; text-decoration: none; background-color: #2ce5b5; color: #000000; padding: 10px 20px; border-radius: 6px; position: absolute; z-index: 1; width: 400px; top: 100%; left: 50%; margin-left: -200px; } +.tooltip:hover .tooltiptext { visibility: visible; } +.tooltip .tooltiptext::after { content: " "; position: absolute; color: #2ce5b5; bottom: 100%; /* At the top of the tooltip */ left: 50%; margin-left: -5px; border-width: 5px; border-style: solid; border-color: transparent transparent #2ce5b5 transparent; } + +.ui-dialog-titlebar { display: none; } + +.feedback-container { position: sticky; top: 180px; z-index: 9999; } +@media (max-width: 1024px) { .feedback-container { display: none; } } +.feedback-container .feedback-buttons { position: absolute; right: 70px; margin-bottom: 20px; top: -55px; } +.feedback-container .feedback-buttons .tooltiptext { font-size: small; width: 150px; margin-left: -75px; text-align: center; } +.feedback-container .feedback-buttons .page-helpful-btn { background: none !important; border-radius: 50%; border: solid #279890 2px; color: #279890; height: 40px; width: 40px; margin: 0 3px; cursor: pointer; } +.feedback-container .feedback-buttons .page-helpful-btn:focus-visible { outline: none; } + +#is-helpful-ty { display: none; } +#is-helpful-ty a { color: #279890 !important; } + +.badge { padding: 2px 4px; background: #ffddec; text-transform: uppercase; text-align: center; border-radius: 6px; font-size: x-small; } + +.quickstart-steps { padding: 10px; } +.quickstart-steps img { width: 50px; height: 50px; } + +.row { display: flex; flex-direction: row; } + +.col { display: flex; flex-direction: column; justify-content: center; margin-bottom: 0.3em; } +.col.step-num { margin-right: 20px; } +.col.step-num > img { width: 40px; } + +img.quickstart { box-shadow: 3px 3px 1px #ccc; border-width: thin; border-color: black; border-style: ridge; } + +.breadcrumb-nav { margin-top: 1rem; } diff --git a/v1.46/assets/css/just-the-docs-light.css b/v1.46/assets/css/just-the-docs-light.css new file mode 100644 index 000000000..333b8f650 --- /dev/null +++ b/v1.46/assets/css/just-the-docs-light.css @@ -0,0 +1,2562 @@ +@charset "UTF-8"; +@import url("pygments/igorpro.css"); +.highlight, pre.highlight { background: #f9f9f9; color: #383942; } + +.highlight pre { background: #f9f9f9; } + +.highlight .hll { background: #f9f9f9; } + +.highlight .c { color: #9fa0a6; font-style: italic; } + +.highlight .err { color: #fff; background-color: #e05151; } + +.highlight .k { color: #a625a4; } + +.highlight .l { color: #50a04f; } + +.highlight .n { color: #383942; } + +.highlight .o { color: #383942; } + +.highlight .p { color: #383942; } + +.highlight .cm { color: #9fa0a6; font-style: italic; } + +.highlight .cp { color: #9fa0a6; font-style: italic; } + +.highlight .c1 { color: #9fa0a6; font-style: italic; } + +.highlight .cs { color: #9fa0a6; font-style: italic; } + +.highlight .ge { font-style: italic; } + +.highlight .gs { font-weight: 700; } + +.highlight .kc { color: #a625a4; } + +.highlight .kd { color: #a625a4; } + +.highlight .kn { color: #a625a4; } + +.highlight .kp { color: #a625a4; } + +.highlight .kr { color: #a625a4; } + +.highlight .kt { color: #a625a4; } + +.highlight .ld { color: #50a04f; } + +.highlight .m { color: #b66a00; } + +.highlight .s { color: #50a04f; } + +.highlight .na { color: #b66a00; } + +.highlight .nb { color: #ca7601; } + +.highlight .nc { color: #ca7601; } + +.highlight .no { color: #ca7601; } + +.highlight .nd { color: #ca7601; } + +.highlight .ni { color: #ca7601; } + +.highlight .ne { color: #ca7601; } + +.highlight .nf { color: #383942; } + +.highlight .nl { color: #ca7601; } + +.highlight .nn { color: #383942; } + +.highlight .nx { color: #383942; } + +.highlight .py { color: #ca7601; } + +.highlight .nt { color: #e35549; } + +.highlight .nv { color: #ca7601; } + +.highlight .ow { font-weight: 700; } + +.highlight .w { color: #f8f8f2; } + +.highlight .mf { color: #b66a00; } + +.highlight .mh { color: #b66a00; } + +.highlight .mi { color: #b66a00; } + +.highlight .mo { color: #b66a00; } + +.highlight .sb { color: #50a04f; } + +.highlight .sc { color: #50a04f; } + +.highlight .sd { color: #50a04f; } + +.highlight .s2 { color: #50a04f; } + +.highlight .se { color: #50a04f; } + +.highlight .sh { color: #50a04f; } + +.highlight .si { color: #50a04f; } + +.highlight .sx { color: #50a04f; } + +.highlight .sr { color: #0083bb; } + +.highlight .s1 { color: #50a04f; } + +.highlight .ss { color: #0083bb; } + +.highlight .bp { color: #ca7601; } + +.highlight .vc { color: #ca7601; } + +.highlight .vg { color: #ca7601; } + +.highlight .vi { color: #e35549; } + +.highlight .il { color: #b66a00; } + +.highlight .gu { color: #75715e; } + +.highlight .gd { color: #e05151; } + +.highlight .gi { color: #43d089; } + +.highlight .language-json .w + .s2 { color: #e35549; } + +.highlight .language-json .kc { color: #0083bb; } + +/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ +/* Document ========================================================================== */ +/** 1. Correct the line height in all browsers. 2. Prevent adjustments of font size after orientation changes in iOS. */ +html { line-height: 1.15; /* 1 */ -webkit-text-size-adjust: 100%; /* 2 */ } + +/* Sections ========================================================================== */ +/** Remove the margin in all browsers. */ +body { margin: 0; } + +/** Render the `main` element consistently in IE. */ +main { display: block; } + +/** Correct the font size and margin on `h1` elements within `section` and `article` contexts in Chrome, Firefox, and Safari. */ +h1 { font-size: 2em; margin: 0.67em 0; } + +/* Grouping content ========================================================================== */ +/** 1. Add the correct box sizing in Firefox. 2. Show the overflow in Edge and IE. */ +hr { box-sizing: content-box; /* 1 */ height: 0; /* 1 */ overflow: visible; /* 2 */ } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +pre { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/* Text-level semantics ========================================================================== */ +/** Remove the gray background on active links in IE 10. */ +a { background-color: transparent; } + +/** 1. Remove the bottom border in Chrome 57- 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. */ +abbr[title] { border-bottom: none; /* 1 */ text-decoration: underline; /* 2 */ text-decoration: underline dotted; /* 2 */ } + +/** Add the correct font weight in Chrome, Edge, and Safari. */ +b, strong { font-weight: bolder; } + +/** 1. Correct the inheritance and scaling of font size in all browsers. 2. Correct the odd `em` font sizing in all browsers. */ +code, kbd, samp { font-family: monospace, monospace; /* 1 */ font-size: 1em; /* 2 */ } + +/** Add the correct font size in all browsers. */ +small { font-size: 80%; } + +/** Prevent `sub` and `sup` elements from affecting the line height in all browsers. */ +sub, sup { font-size: 75%; line-height: 0; position: relative; vertical-align: baseline; } + +sub { bottom: -0.25em; } + +sup { top: -0.5em; } + +/* Embedded content ========================================================================== */ +/** Remove the border on images inside links in IE 10. */ +img { border-style: none; } + +/* Forms ========================================================================== */ +/** 1. Change the font styles in all browsers. 2. Remove the margin in Firefox and Safari. */ +button, input, optgroup, select, textarea { font-family: inherit; /* 1 */ font-size: 100%; /* 1 */ line-height: 1.15; /* 1 */ margin: 0; /* 2 */ } + +/** Show the overflow in IE. 1. Show the overflow in Edge. */ +button, input { /* 1 */ overflow: visible; } + +/** Remove the inheritance of text transform in Edge, Firefox, and IE. 1. Remove the inheritance of text transform in Firefox. */ +button, select { /* 1 */ text-transform: none; } + +/** Correct the inability to style clickable types in iOS and Safari. */ +button, [type="button"], [type="reset"], [type="submit"] { -webkit-appearance: button; } + +/** Remove the inner border and padding in Firefox. */ +button::-moz-focus-inner, [type="button"]::-moz-focus-inner, [type="reset"]::-moz-focus-inner, [type="submit"]::-moz-focus-inner { border-style: none; padding: 0; } + +/** Restore the focus styles unset by the previous rule. */ +button:-moz-focusring, [type="button"]:-moz-focusring, [type="reset"]:-moz-focusring, [type="submit"]:-moz-focusring { outline: 1px dotted ButtonText; } + +/** Correct the padding in Firefox. */ +fieldset { padding: 0.35em 0.75em 0.625em; } + +/** 1. Correct the text wrapping in Edge and IE. 2. Correct the color inheritance from `fieldset` elements in IE. 3. Remove the padding so developers are not caught out when they zero out `fieldset` elements in all browsers. */ +legend { box-sizing: border-box; /* 1 */ color: inherit; /* 2 */ display: table; /* 1 */ max-width: 100%; /* 1 */ padding: 0; /* 3 */ white-space: normal; /* 1 */ } + +/** Add the correct vertical alignment in Chrome, Firefox, and Opera. */ +progress { vertical-align: baseline; } + +/** Remove the default vertical scrollbar in IE 10+. */ +textarea { overflow: auto; } + +/** 1. Add the correct box sizing in IE 10. 2. Remove the padding in IE 10. */ +[type="checkbox"], [type="radio"] { box-sizing: border-box; /* 1 */ padding: 0; /* 2 */ } + +/** Correct the cursor style of increment and decrement buttons in Chrome. */ +[type="number"]::-webkit-inner-spin-button, [type="number"]::-webkit-outer-spin-button { height: auto; } + +/** 1. Correct the odd appearance in Chrome and Safari. 2. Correct the outline style in Safari. */ +[type="search"] { -webkit-appearance: textfield; /* 1 */ outline-offset: -2px; /* 2 */ } + +/** Remove the inner padding in Chrome and Safari on macOS. */ +[type="search"]::-webkit-search-decoration { -webkit-appearance: none; } + +/** 1. Correct the inability to style clickable types in iOS and Safari. 2. Change font properties to `inherit` in Safari. */ +::-webkit-file-upload-button { -webkit-appearance: button; /* 1 */ font: inherit; /* 2 */ } + +/* Interactive ========================================================================== */ +/* Add the correct display in Edge, IE 10+, and Firefox. */ +details { display: block; } + +/* Add the correct display in all browsers. */ +summary { display: list-item; } + +/* Misc ========================================================================== */ +/** Add the correct display in IE 10+. */ +template { display: none; } + +/** Add the correct display in IE 10. */ +[hidden] { display: none; } + +:root { color-scheme: light; } + +* { box-sizing: border-box; } + +html { font-size: 0.875rem !important; scroll-behavior: smooth; } +@media (min-width: 31.25rem) { html { font-size: 1rem !important; } } + +body { font-family: system-ui, -apple-system, blinkmacsystemfont, "Segoe UI", roboto, "Helvetica Neue", arial, sans-serif; font-size: inherit; line-height: 1.4; color: #5c5962; background-color: #fff; overflow-wrap: break-word; } + +ol, ul, dl, pre, address, blockquote, table, div, hr, form, fieldset, noscript .table-wrapper { margin-top: 0; } + +h1, h2, h3, h4, h5, h6, #toctitle { margin-top: 0; margin-bottom: 1em; font-weight: 500; line-height: 1.25; color: #27262b; } + +p { margin-top: 1em; margin-bottom: 1em; } + +a { color: #7253ed; text-decoration: none; } + +a:not([class]) { text-decoration: underline; text-decoration-color: #eeebee; text-underline-offset: 2px; } +a:not([class]):hover { text-decoration-color: rgba(114, 83, 237, 0.45); } + +code { font-family: "SFMono-Regular", menlo, consolas, monospace; font-size: 0.75em; line-height: 1.4; } + +figure, pre { margin: 0; } + +li { margin: 0.25em 0; } + +img { max-width: 100%; height: auto; } + +hr { height: 1px; padding: 0; margin: 2rem 0; background-color: #eeebee; border: 0; } + +blockquote { margin: 10px 0; margin-block-start: 0; margin-inline-start: 0; padding-left: 1rem; border-left: 3px solid #eeebee; } + +.side-bar { z-index: 0; display: flex; flex-wrap: wrap; background-color: #f5f6fa; } +@media (min-width: 50rem) { .side-bar { flex-flow: column nowrap; position: fixed; width: 15.5rem; height: 100%; border-right: 1px solid #eeebee; align-items: flex-end; } } +@media (min-width: 66.5rem) { .side-bar { width: calc((100% - 66.5rem) / 2 + 16.5rem); min-width: 16.5rem; } } + +@media (min-width: 50rem) { .main { position: relative; max-width: 50rem; margin-left: 15.5rem; } } +@media (min-width: 66.5rem) { .main { margin-left: Max(16.5rem, calc((100% - 66.5rem) / 2 + 16.5rem)); } } + +.main-content-wrap { padding-right: 1rem; padding-left: 1rem; padding-top: 1rem; padding-bottom: 1rem; } +@media (min-width: 50rem) { .main-content-wrap { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 50rem) { .main-content-wrap { padding-top: 2rem; padding-bottom: 2rem; } } + +.main-header { z-index: 0; display: none; background-color: #f5f6fa; } +@media (min-width: 50rem) { .main-header { display: flex; justify-content: space-between; height: 3.75rem; background-color: #fff; border-bottom: 1px solid #eeebee; } } +.main-header.nav-open { display: block; } +@media (min-width: 50rem) { .main-header.nav-open { display: flex; } } + +.site-nav, .site-header, .site-footer { width: 100%; } +@media (min-width: 66.5rem) { .site-nav, .site-header, .site-footer { width: 16.5rem; } } + +.site-nav { display: none; } +.site-nav.nav-open { display: block; } +@media (min-width: 50rem) { .site-nav { display: block; padding-top: 3rem; padding-bottom: 1rem; overflow-y: auto; flex: 1 1 auto; } } + +.site-header { display: flex; min-height: 3.75rem; align-items: center; } +@media (min-width: 50rem) { .site-header { height: 3.75rem; max-height: 3.75rem; border-bottom: 1px solid #eeebee; } } + +.site-title { padding-right: 1rem; padding-left: 1rem; flex-grow: 1; display: flex; height: 100%; align-items: center; padding-top: 0.75rem; padding-bottom: 0.75rem; color: #27262b; font-size: 1.125rem !important; } +@media (min-width: 50rem) { .site-title { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-title { font-size: 1.5rem !important; line-height: 1.25; } } +@media (min-width: 50rem) { .site-title { padding-top: 0.5rem; padding-bottom: 0.5rem; } } + +.site-logo { width: 100%; height: 100%; background-image: url("/v1.46/assets/logo.svg"); background-repeat: no-repeat; background-position: left center; background-size: contain; } + +.site-button { display: flex; height: 100%; padding: 1rem; align-items: center; } + +@media (min-width: 50rem) { .site-header .site-button { display: none; } } +.site-title:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 80%, rgba(235, 237, 245, 0) 100%); } + +.site-button:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 100%); } + +body { position: relative; padding-bottom: 4rem; overflow-y: scroll; } +@media (min-width: 50rem) { body { position: static; padding-bottom: 0; } } + +.site-footer { padding-right: 1rem; padding-left: 1rem; position: absolute; bottom: 0; left: 0; padding-top: 1rem; padding-bottom: 1rem; color: #959396; font-size: 0.6875rem !important; } +@media (min-width: 50rem) { .site-footer { padding-right: 2rem; padding-left: 2rem; } } +@media (min-width: 31.25rem) { .site-footer { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .site-footer { position: static; justify-self: end; } } + +.icon { width: 1.5rem; height: 1.5rem; color: #7253ed; } + +.main-content { line-height: 1.6; } +.main-content ol, .main-content ul, .main-content dl, .main-content pre, .main-content address, .main-content blockquote, .main-content .table-wrapper { margin-top: 0.5em; } +.main-content a { overflow: hidden; text-overflow: ellipsis; } +.main-content ul, .main-content ol { padding-left: 1.5em; } +.main-content li .highlight { margin-top: 0.25rem; } +.main-content ol { list-style-type: none; counter-reset: step-counter; } +.main-content ol > li { position: relative; } +.main-content ol > li::before { position: absolute; top: 0.2em; left: -1.6em; color: #959396; content: counter(step-counter); counter-increment: step-counter; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .main-content ol > li::before { font-size: 0.875rem !important; } } +@media (min-width: 31.25rem) { .main-content ol > li::before { top: 0.11em; } } +.main-content ol > li ol { counter-reset: sub-counter; } +.main-content ol > li ol > li::before { content: counter(sub-counter,lower-alpha); counter-increment: sub-counter; } +.main-content ul { list-style: none; } +.main-content ul > li::before { position: absolute; margin-left: -1.4em; color: #959396; content: "•"; } +.main-content .task-list-item::before { content: ""; } +.main-content .task-list-item-checkbox { margin-right: 0.6em; margin-left: -1.4em; } +.main-content hr + * { margin-top: 0; } +.main-content h1:first-of-type { margin-top: 0.5em; } +.main-content dl { display: grid; grid-template: auto / 10em 1fr; } +.main-content dt, .main-content dd { margin: 0.25em 0; } +.main-content dt { grid-column: 1; font-weight: 500; text-align: right; } +.main-content dt::after { content: ":"; } +.main-content dd { grid-column: 2; margin-bottom: 0; margin-left: 1em; } +.main-content dd blockquote:first-child, .main-content dd div:first-child, .main-content dd dl:first-child, .main-content dd dt:first-child, .main-content dd h1:first-child, .main-content dd h2:first-child, .main-content dd h3:first-child, .main-content dd h4:first-child, .main-content dd h5:first-child, .main-content dd h6:first-child, .main-content dd li:first-child, .main-content dd ol:first-child, .main-content dd p:first-child, .main-content dd pre:first-child, .main-content dd table:first-child, .main-content dd ul:first-child, .main-content dd .table-wrapper:first-child { margin-top: 0; } +.main-content dd dl:first-child dt:first-child, .main-content dd dl:first-child dd:nth-child(2), .main-content ol dl:first-child dt:first-child, .main-content ol dl:first-child dd:nth-child(2), .main-content ul dl:first-child dt:first-child, .main-content ul dl:first-child dd:nth-child(2) { margin-top: 0; } +.main-content .anchor-heading { position: absolute; right: -1rem; width: 1.5rem; height: 100%; padding-right: 0.25rem; padding-left: 0.25rem; overflow: visible; } +@media (min-width: 50rem) { .main-content .anchor-heading { right: auto; left: -1.5rem; } } +.main-content .anchor-heading svg { display: inline-block; width: 100%; height: 100%; color: #7253ed; visibility: hidden; } +.main-content .anchor-heading:hover svg, .main-content .anchor-heading:focus svg, .main-content h1:hover > .anchor-heading svg, .main-content h2:hover > .anchor-heading svg, .main-content h3:hover > .anchor-heading svg, .main-content h4:hover > .anchor-heading svg, .main-content h5:hover > .anchor-heading svg, .main-content h6:hover > .anchor-heading svg { visibility: visible; } +.main-content summary { cursor: pointer; } +.main-content h1, .main-content h2, .main-content h3, .main-content h4, .main-content h5, .main-content h6, .main-content #toctitle { position: relative; margin-top: 1.5em; margin-bottom: 0.25em; } +.main-content h1 + table, .main-content h1 + .table-wrapper, .main-content h1 + .code-example, .main-content h1 + .highlighter-rouge, .main-content h1 + .sectionbody .listingblock, .main-content h2 + table, .main-content h2 + .table-wrapper, .main-content h2 + .code-example, .main-content h2 + .highlighter-rouge, .main-content h2 + .sectionbody .listingblock, .main-content h3 + table, .main-content h3 + .table-wrapper, .main-content h3 + .code-example, .main-content h3 + .highlighter-rouge, .main-content h3 + .sectionbody .listingblock, .main-content h4 + table, .main-content h4 + .table-wrapper, .main-content h4 + .code-example, .main-content h4 + .highlighter-rouge, .main-content h4 + .sectionbody .listingblock, .main-content h5 + table, .main-content h5 + .table-wrapper, .main-content h5 + .code-example, .main-content h5 + .highlighter-rouge, .main-content h5 + .sectionbody .listingblock, .main-content h6 + table, .main-content h6 + .table-wrapper, .main-content h6 + .code-example, .main-content h6 + .highlighter-rouge, .main-content h6 + .sectionbody .listingblock, .main-content #toctitle + table, .main-content #toctitle + .table-wrapper, .main-content #toctitle + .code-example, .main-content #toctitle + .highlighter-rouge, .main-content #toctitle + .sectionbody .listingblock { margin-top: 1em; } +.main-content h1 + p:not(.label), .main-content h2 + p:not(.label), .main-content h3 + p:not(.label), .main-content h4 + p:not(.label), .main-content h5 + p:not(.label), .main-content h6 + p:not(.label), .main-content #toctitle + p:not(.label) { margin-top: 0; } +.main-content > h1:first-child, .main-content > h2:first-child, .main-content > h3:first-child, .main-content > h4:first-child, .main-content > h5:first-child, .main-content > h6:first-child, .main-content > .sect1:first-child > h2, .main-content > .sect2:first-child > h3, .main-content > .sect3:first-child > h4, .main-content > .sect4:first-child > h5, .main-content > .sect5:first-child > h6 { margin-top: 0.5rem; } + +.nav-list { padding: 0; margin-top: 0; margin-bottom: 0; list-style: none; } +.nav-list .nav-list-item { font-size: 0.875rem !important; position: relative; margin: 0; } +@media (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 1rem !important; } } +@media (min-width: 50rem) { .nav-list .nav-list-item { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .nav-list .nav-list-item { font-size: 0.875rem !important; } } + +.nav-list .nav-list-item .nav-list-link { display: block; min-height: 3rem; padding-top: 0.25rem; padding-bottom: 0.25rem; line-height: 2.5rem; padding-right: 3rem; padding-left: 1rem; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-link { min-height: 2rem; line-height: 1.5rem; padding-right: 2rem; padding-left: 2rem; } } +.nav-list .nav-list-item .nav-list-link.external > svg { width: 1rem; height: 1rem; vertical-align: text-bottom; } +.nav-list .nav-list-item .nav-list-link.active { font-weight: 600; text-decoration: none; } +.nav-list .nav-list-item .nav-list-link:hover, .nav-list .nav-list-item .nav-list-link.active { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 80%, rgba(235, 237, 245, 0) 100%); } +.nav-list .nav-list-item .nav-list-expander { position: absolute; right: 0; width: 3rem; height: 3rem; padding: 0.75rem; color: #7253ed; } +@media (min-width: 50rem) { .nav-list .nav-list-item .nav-list-expander { width: 2rem; height: 2rem; padding: 0.5rem; } } +.nav-list .nav-list-item .nav-list-expander:hover { background-image: linear-gradient(-90deg, #ebedf5 0%, rgba(235, 237, 245, 0.8) 100%); } +.nav-list .nav-list-item .nav-list-expander svg { transform: rotate(90deg); } +.nav-list .nav-list-item > .nav-list { display: none; padding-left: 0.75rem; list-style: none; } +.nav-list .nav-list-item > .nav-list .nav-list-item { position: relative; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-link { color: #5c5962; } +.nav-list .nav-list-item > .nav-list .nav-list-item .nav-list-expander { color: #5c5962; } +.nav-list .nav-list-item.active > .nav-list-expander svg { transform: rotate(-90deg); } +.nav-list .nav-list-item.active > .nav-list { display: block; } + +.nav-category { padding: 0.5rem 1rem; font-weight: 600; text-align: start; text-transform: uppercase; border-bottom: 1px solid #eeebee; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .nav-category { font-size: 0.75rem !important; } } +@media (min-width: 50rem) { .nav-category { padding: 0.5rem 2rem; margin-top: 1rem; text-align: start; } + .nav-category:first-child { margin-top: 0; } } + +.nav-list.nav-category-list > .nav-list-item { margin: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list { padding: 0; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-link { color: #7253ed; } +.nav-list.nav-category-list > .nav-list-item > .nav-list > .nav-list-item > .nav-list-expander { color: #7253ed; } + +.aux-nav { height: 100%; overflow-x: auto; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .aux-nav { font-size: 0.75rem !important; } } +.aux-nav .aux-nav-list { display: flex; height: 100%; padding: 0; margin: 0; list-style: none; } +.aux-nav .aux-nav-list-item { display: inline-block; height: 100%; padding: 0; margin: 0; } +@media (min-width: 50rem) { .aux-nav { padding-right: 1rem; } } + +@media (min-width: 50rem) { .breadcrumb-nav { margin-top: -1rem; } } + +.breadcrumb-nav-list { padding-left: 0; margin-bottom: 0.75rem; list-style: none; } + +.breadcrumb-nav-list-item { display: table-cell; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .breadcrumb-nav-list-item { font-size: 0.75rem !important; } } +.breadcrumb-nav-list-item::before { display: none; } +.breadcrumb-nav-list-item::after { display: inline-block; margin-right: 0.5rem; margin-left: 0.5rem; color: #959396; content: "/"; } +.breadcrumb-nav-list-item:last-child::after { content: ""; } + +h1, .text-alpha { font-size: 2rem !important; line-height: 1.25; font-weight: 300; } +@media (min-width: 31.25rem) { h1, .text-alpha { font-size: 2.25rem !important; } } + +h2, .text-beta, #toctitle { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { h2, .text-beta, #toctitle { font-size: 1.5rem !important; line-height: 1.25; } } + +h3, .text-gamma { font-size: 1rem !important; } +@media (min-width: 31.25rem) { h3, .text-gamma { font-size: 1.125rem !important; } } + +h4, .text-delta { font-size: 0.6875rem !important; font-weight: 400; text-transform: uppercase; letter-spacing: 0.1em; } +@media (min-width: 31.25rem) { h4, .text-delta { font-size: 0.75rem !important; } } + +h4 code { text-transform: none; } + +h5, .text-epsilon { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { h5, .text-epsilon { font-size: 0.875rem !important; } } + +h6, .text-zeta { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { h6, .text-zeta { font-size: 0.75rem !important; } } + +.text-small { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .text-small { font-size: 0.75rem !important; } } + +.text-mono { font-family: "SFMono-Regular", menlo, consolas, monospace !important; } + +.text-left { text-align: left !important; } + +.text-center { text-align: center !important; } + +.text-right { text-align: right !important; } + +.label:not(g), .label-blue:not(g) { display: inline-block; padding: 0.16em 0.56em; margin-right: 0.5rem; margin-left: 0.5rem; color: #fff; text-transform: uppercase; vertical-align: middle; background-color: #2869e6; font-size: 0.6875rem !important; border-radius: 12px; } +@media (min-width: 31.25rem) { .label:not(g), .label-blue:not(g) { font-size: 0.75rem !important; } } + +.label-green:not(g) { background-color: #009c7b; } + +.label-purple:not(g) { background-color: #5e41d0; } + +.label-red:not(g) { background-color: #e94c4c; } + +.label-yellow:not(g) { color: #44434d; background-color: #f7d12e; } + +.btn { display: inline-block; box-sizing: border-box; padding: 0.3em 1em; margin: 0; font-family: inherit; font-size: inherit; font-weight: 500; line-height: 1.5; color: #7253ed; text-decoration: none; vertical-align: baseline; cursor: pointer; background-color: #f7f7f7; border-width: 0; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); appearance: none; } +.btn:focus { text-decoration: none; outline: none; box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:focus:hover, .btn.selected:focus { box-shadow: 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn:hover, .btn.zeroclipboard-is-hover { color: #6a4aec; } +.btn:hover, .btn:active, .btn.zeroclipboard-is-hover, .btn.zeroclipboard-is-active { text-decoration: none; background-color: #f4f4f4; } +.btn:active, .btn.selected, .btn.zeroclipboard-is-active { background-color: #efefef; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn.selected:hover { background-color: #cfcfcf; } +.btn:disabled, .btn:disabled:hover, .btn.disabled, .btn.disabled:hover { color: rgba(102, 102, 102, 0.5); cursor: default; background-color: rgba(229, 229, 229, 0.5); background-image: none; box-shadow: none; } + +.btn-outline { color: #7253ed; background: transparent; box-shadow: inset 0 0 0 2px #e6e1e8; } +.btn-outline:hover, .btn-outline:active, .btn-outline.zeroclipboard-is-hover, .btn-outline.zeroclipboard-is-active { color: #6341eb; text-decoration: none; background-color: transparent; box-shadow: inset 0 0 0 3px #e6e1e8; } +.btn-outline:focus { text-decoration: none; outline: none; box-shadow: inset 0 0 0 2px #5c5962, 0 0 0 3px rgba(0, 0, 255, 0.25); } +.btn-outline:focus:hover, .btn-outline.selected:focus { box-shadow: inset 0 0 0 2px #5c5962; } + +.btn-primary { color: #fff; background-color: #5739ce; background-image: linear-gradient(#6f55d5, #5739ce); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-primary:hover, .btn-primary.zeroclipboard-is-hover { color: #fff; background-color: #5132cb; background-image: linear-gradient(#6549d2, #5132cb); } +.btn-primary:active, .btn-primary.selected, .btn-primary.zeroclipboard-is-active { background-color: #4f31c6; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-primary.selected:hover { background-color: #472cb2; } + +.btn-purple { color: #fff; background-color: #5739ce; background-image: linear-gradient(#6f55d5, #5739ce); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-purple:hover, .btn-purple.zeroclipboard-is-hover { color: #fff; background-color: #5132cb; background-image: linear-gradient(#6549d2, #5132cb); } +.btn-purple:active, .btn-purple.selected, .btn-purple.zeroclipboard-is-active { background-color: #4f31c6; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-purple.selected:hover { background-color: #472cb2; } + +.btn-blue { color: #fff; background-color: #227efa; background-image: linear-gradient(#4593fb, #227efa); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-blue:hover, .btn-blue.zeroclipboard-is-hover { color: #fff; background-color: #1878fa; background-image: linear-gradient(#368afa, #1878fa); } +.btn-blue:active, .btn-blue.selected, .btn-blue.zeroclipboard-is-active { background-color: #1375f9; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-blue.selected:hover { background-color: #0669ed; } + +.btn-green { color: #fff; background-color: #10ac7d; background-image: linear-gradient(#13cc95, #10ac7d); box-shadow: 0 1px 3px rgba(0, 0, 0, 0.25), 0 4px 10px rgba(0, 0, 0, 0.12); } +.btn-green:hover, .btn-green.zeroclipboard-is-hover { color: #fff; background-color: #0fa276; background-image: linear-gradient(#12be8b, #0fa276); } +.btn-green:active, .btn-green.selected, .btn-green.zeroclipboard-is-active { background-color: #0f9e73; background-image: none; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.15); } +.btn-green.selected:hover { background-color: #0d8662; } + +.btn-reset { background: none; border: none; margin: 0; text-align: inherit; font: inherit; border-radius: 0; appearance: none; } + +.search { position: relative; z-index: 2; flex-grow: 1; height: 4rem; padding: 0.5rem; transition: padding linear 200ms; } +@media (min-width: 50rem) { .search { position: relative !important; width: auto !important; height: 100% !important; padding: 0; transition: none; } } + +.search-input-wrap { position: relative; z-index: 1; height: 3rem; overflow: hidden; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); transition: height linear 200ms; } +@media (min-width: 50rem) { .search-input-wrap { position: absolute; width: 100%; max-width: 33.5rem; height: 100% !important; border-radius: 0; box-shadow: none; transition: width ease 400ms; } } + +.search-input { position: absolute; width: 100%; height: 100%; padding: 0.5rem 1rem 0.5rem 2.5rem; font-size: 1rem; color: #5c5962; background-color: #fff; border-top: 0; border-right: 0; border-bottom: 0; border-left: 0; border-radius: 0; } +@media (min-width: 50rem) { .search-input { padding: 0.5rem 1rem 0.5rem 3.5rem; font-size: 0.875rem; background-color: #fff; transition: padding-left linear 200ms; } } +.search-input:focus { outline: 0; } +.search-input:focus + .search-label .search-icon { color: #7253ed; } + +.search-label { position: absolute; display: flex; height: 100%; padding-left: 1rem; } +@media (min-width: 50rem) { .search-label { padding-left: 2rem; transition: padding-left linear 200ms; } } +.search-label .search-icon { width: 1.2rem; height: 1.2rem; align-self: center; color: #959396; } + +.search-results { position: absolute; left: 0; display: none; width: 100%; max-height: calc(100% - 4rem); overflow-y: auto; background-color: #fff; border-bottom-right-radius: 4px; border-bottom-left-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } +@media (min-width: 50rem) { .search-results { top: 100%; width: 33.5rem; max-height: calc(100vh - 200%) !important; } } + +.search-results-list { padding-left: 0; margin-bottom: 0.25rem; list-style: none; font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .search-results-list { font-size: 1rem !important; } } +@media (min-width: 50rem) { .search-results-list { font-size: 0.75rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-results-list { font-size: 0.875rem !important; } } + +.search-results-list-item { padding: 0; margin: 0; } + +.search-result { display: block; padding: 0.25rem 0.75rem; } +.search-result:hover, .search-result.active { background-color: #ebedf5; } + +.search-result-title { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; } +@media (min-width: 31.25rem) { .search-result-title { display: inline-block; width: 40%; padding-right: 0.5rem; vertical-align: top; } } + +.search-result-doc { display: flex; align-items: center; word-wrap: break-word; } +.search-result-doc.search-result-doc-parent { opacity: 0.5; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.875rem !important; } } +@media (min-width: 50rem) { .search-result-doc.search-result-doc-parent { font-size: 0.6875rem !important; } } +@media (min-width: 50rem) and (min-width: 31.25rem) { .search-result-doc.search-result-doc-parent { font-size: 0.75rem !important; } } + +.search-result-doc .search-result-icon { width: 1rem; height: 1rem; margin-right: 0.5rem; color: #7253ed; flex-shrink: 0; } +.search-result-doc .search-result-doc-title { overflow: auto; } + +.search-result-section { margin-left: 1.5rem; word-wrap: break-word; } + +.search-result-rel-url { display: block; margin-left: 1.5rem; overflow: hidden; color: #959396; text-overflow: ellipsis; white-space: nowrap; font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .search-result-rel-url { font-size: 0.625rem !important; } } + +.search-result-previews { display: block; padding-top: 0.5rem; padding-bottom: 0.5rem; padding-left: 1rem; margin-left: 0.5rem; color: #959396; word-wrap: break-word; border-left: 1px solid; border-left-color: #eeebee; font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .search-result-previews { font-size: 0.75rem !important; } } +@media (min-width: 31.25rem) { .search-result-previews { display: inline-block; width: 60%; padding-left: 0.5rem; margin-left: 0; vertical-align: top; } } + +.search-result-preview + .search-result-preview { margin-top: 0.25rem; } + +.search-result-highlight { font-weight: bold; } + +.search-no-result { padding: 0.5rem 0.75rem; font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .search-no-result { font-size: 0.875rem !important; } } + +.search-button { position: fixed; right: 1rem; bottom: 1rem; display: flex; width: 3.5rem; height: 3.5rem; background-color: #fff; border: 1px solid rgba(114, 83, 237, 0.3); border-radius: 1.75rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); align-items: center; justify-content: center; } + +.search-overlay { position: fixed; top: 0; left: 0; z-index: 1; width: 0; height: 0; background-color: rgba(0, 0, 0, 0.3); opacity: 0; transition: opacity ease 400ms, width 0s 400ms, height 0s 400ms; } + +.search-active .search { position: fixed; top: 0; left: 0; width: 100%; height: 100%; padding: 0; } +.search-active .search-input-wrap { height: 4rem; border-radius: 0; } +@media (min-width: 50rem) { .search-active .search-input-wrap { width: 33.5rem; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } } +.search-active .search-input { background-color: #fff; } +@media (min-width: 50rem) { .search-active .search-input { padding-left: 2.3rem; } } +@media (min-width: 50rem) { .search-active .search-label { padding-left: 0.6rem; } } +.search-active .search-results { display: block; } +.search-active .search-overlay { width: 100%; height: 100%; opacity: 1; transition: opacity ease 400ms, width 0s, height 0s; } +@media (min-width: 50rem) { .search-active .main { position: fixed; right: 0; left: 0; } } +.search-active .main-header { padding-top: 4rem; } +@media (min-width: 50rem) { .search-active .main-header { padding-top: 0; } } + +.table-wrapper { display: block; width: 100%; max-width: 100%; margin-bottom: 1.5rem; overflow-x: auto; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); } + +table { display: table; min-width: 100%; border-collapse: separate; } + +th, td { font-size: 0.75rem !important; min-width: 7.5rem; padding: 0.5rem 0.75rem; background-color: #fff; border-bottom: 1px solid rgba(238, 235, 238, 0.5); border-left: 1px solid #eeebee; } +@media (min-width: 31.25rem) { th, td { font-size: 0.875rem !important; } } +th:first-of-type, td:first-of-type { border-left: 0; } + +tbody tr:last-of-type th, tbody tr:last-of-type td { border-bottom: 0; } +tbody tr:last-of-type td { padding-bottom: 0.75rem; } + +thead th { border-bottom: 1px solid #eeebee; } + +:not(pre, figure) > code { padding: 0.2em 0.15em; font-weight: 400; background-color: #f5f6fa; border: 1px solid #eeebee; border-radius: 4px; } + +a:visited code { border-color: #eeebee; } + +div.highlighter-rouge, div.listingblock > div.content, figure.highlight { margin-top: 0; margin-bottom: 0.75rem; background-color: #f5f6fa; border-radius: 4px; box-shadow: none; -webkit-overflow-scrolling: touch; position: relative; padding: 0; } +div.highlighter-rouge > button, div.listingblock > div.content > button, figure.highlight > button { width: 0.75rem; opacity: 0; position: absolute; top: 0; right: 0; border: 0.75rem solid #f5f6fa; background-color: #f5f6fa; color: #5c5962; box-sizing: content-box; } +div.highlighter-rouge > button svg, div.listingblock > div.content > button svg, figure.highlight > button svg { fill: #5c5962; } +div.highlighter-rouge > button:active, div.listingblock > div.content > button:active, figure.highlight > button:active { text-decoration: none; outline: none; opacity: 1; } +div.highlighter-rouge > button:focus, div.listingblock > div.content > button:focus, figure.highlight > button:focus { opacity: 1; } +div.highlighter-rouge:hover > button, div.listingblock > div.content:hover > button, figure.highlight:hover > button { cursor: copy; opacity: 1; } + +div.highlighter-rouge div.highlight { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.highlighter-rouge pre.highlight, div.highlighter-rouge code { padding: 0; margin: 0; border: 0; } + +div.listingblock { margin-top: 0; margin-bottom: 0.75rem; } +div.listingblock div.content { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } +div.listingblock div.content > pre, div.listingblock code { padding: 0; margin: 0; border: 0; } + +figure.highlight pre, figure.highlight :not(pre) > code { overflow-x: auto; padding: 0.75rem; margin: 0; border: 0; } + +.highlight .table-wrapper { padding: 0.75rem 0; margin: 0; border: 0; box-shadow: none; } +.highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.6875rem !important; min-width: 0; padding: 0; background-color: #f5f6fa; border: 0; } +@media (min-width: 31.25rem) { .highlight .table-wrapper td, .highlight .table-wrapper pre { font-size: 0.75rem !important; } } +.highlight .table-wrapper td.gl { width: 1em; padding-right: 0.75rem; padding-left: 0.75rem; } +.highlight .table-wrapper pre { margin: 0; line-height: 2; } + +.code-example, .listingblock > .title { padding: 0.75rem; margin-bottom: 0.75rem; overflow: auto; border: 1px solid #eeebee; border-radius: 4px; } +.code-example + .highlighter-rouge, .code-example + .sectionbody .listingblock, .code-example + .content, .code-example + figure.highlight, .listingblock > .title + .highlighter-rouge, .listingblock > .title + .sectionbody .listingblock, .listingblock > .title + .content, .listingblock > .title + figure.highlight { position: relative; margin-top: -1rem; border-right: 1px solid #eeebee; border-bottom: 1px solid #eeebee; border-left: 1px solid #eeebee; border-top-left-radius: 0; border-top-right-radius: 0; } + +code.language-mermaid { padding: 0; background-color: inherit; border: 0; } + +.highlight, pre.highlight { background: #f5f6fa; color: #5c5962; } + +.highlight pre { background: #f5f6fa; } + +.text-grey-dk-000 { color: #959396 !important; } + +.text-grey-dk-100 { color: #5c5962 !important; } + +.text-grey-dk-200 { color: #44434d !important; } + +.text-grey-dk-250 { color: #302d36 !important; } + +.text-grey-dk-300 { color: #27262b !important; } + +.text-grey-lt-000 { color: #f5f6fa !important; } + +.text-grey-lt-100 { color: #eeebee !important; } + +.text-grey-lt-200 { color: #ecebed !important; } + +.text-grey-lt-300 { color: #e6e1e8 !important; } + +.text-blue-000 { color: #2c84fa !important; } + +.text-blue-100 { color: #2869e6 !important; } + +.text-blue-200 { color: #264caf !important; } + +.text-blue-300 { color: #183385 !important; } + +.text-green-000 { color: #41d693 !important; } + +.text-green-100 { color: #11b584 !important; } + +.text-green-200 { color: #009c7b !important; } + +.text-green-300 { color: #026e57 !important; } + +.text-purple-000 { color: #7253ed !important; } + +.text-purple-100 { color: #5e41d0 !important; } + +.text-purple-200 { color: #4e26af !important; } + +.text-purple-300 { color: #381885 !important; } + +.text-yellow-000 { color: #ffeb82 !important; } + +.text-yellow-100 { color: #fadf50 !important; } + +.text-yellow-200 { color: #f7d12e !important; } + +.text-yellow-300 { color: #e7af06 !important; } + +.text-red-000 { color: #f77e7e !important; } + +.text-red-100 { color: #f96e65 !important; } + +.text-red-200 { color: #e94c4c !important; } + +.text-red-300 { color: #dd2e2e !important; } + +.bg-grey-dk-000 { background-color: #959396 !important; } + +.bg-grey-dk-100 { background-color: #5c5962 !important; } + +.bg-grey-dk-200 { background-color: #44434d !important; } + +.bg-grey-dk-250 { background-color: #302d36 !important; } + +.bg-grey-dk-300 { background-color: #27262b !important; } + +.bg-grey-lt-000 { background-color: #f5f6fa !important; } + +.bg-grey-lt-100 { background-color: #eeebee !important; } + +.bg-grey-lt-200 { background-color: #ecebed !important; } + +.bg-grey-lt-300 { background-color: #e6e1e8 !important; } + +.bg-blue-000 { background-color: #2c84fa !important; } + +.bg-blue-100 { background-color: #2869e6 !important; } + +.bg-blue-200 { background-color: #264caf !important; } + +.bg-blue-300 { background-color: #183385 !important; } + +.bg-green-000 { background-color: #41d693 !important; } + +.bg-green-100 { background-color: #11b584 !important; } + +.bg-green-200 { background-color: #009c7b !important; } + +.bg-green-300 { background-color: #026e57 !important; } + +.bg-purple-000 { background-color: #7253ed !important; } + +.bg-purple-100 { background-color: #5e41d0 !important; } + +.bg-purple-200 { background-color: #4e26af !important; } + +.bg-purple-300 { background-color: #381885 !important; } + +.bg-yellow-000 { background-color: #ffeb82 !important; } + +.bg-yellow-100 { background-color: #fadf50 !important; } + +.bg-yellow-200 { background-color: #f7d12e !important; } + +.bg-yellow-300 { background-color: #e7af06 !important; } + +.bg-red-000 { background-color: #f77e7e !important; } + +.bg-red-100 { background-color: #f96e65 !important; } + +.bg-red-200 { background-color: #e94c4c !important; } + +.bg-red-300 { background-color: #dd2e2e !important; } + +.d-block { display: block !important; } + +.d-flex { display: flex !important; } + +.d-inline { display: inline !important; } + +.d-inline-block { display: inline-block !important; } + +.d-none { display: none !important; } + +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 20rem) { .d-xs-block { display: block !important; } + .d-xs-flex { display: flex !important; } + .d-xs-inline { display: inline !important; } + .d-xs-inline-block { display: inline-block !important; } + .d-xs-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 31.25rem) { .d-sm-block { display: block !important; } + .d-sm-flex { display: flex !important; } + .d-sm-inline { display: inline !important; } + .d-sm-inline-block { display: inline-block !important; } + .d-sm-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 50rem) { .d-md-block { display: block !important; } + .d-md-flex { display: flex !important; } + .d-md-inline { display: inline !important; } + .d-md-inline-block { display: inline-block !important; } + .d-md-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 66.5rem) { .d-lg-block { display: block !important; } + .d-lg-flex { display: flex !important; } + .d-lg-inline { display: inline !important; } + .d-lg-inline-block { display: inline-block !important; } + .d-lg-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +@media (min-width: 87.5rem) { .d-xl-block { display: block !important; } + .d-xl-flex { display: flex !important; } + .d-xl-inline { display: inline !important; } + .d-xl-inline-block { display: inline-block !important; } + .d-xl-none { display: none !important; } } +.float-left { float: left !important; } + +.float-right { float: right !important; } + +.flex-justify-start { justify-content: flex-start !important; } + +.flex-justify-end { justify-content: flex-end !important; } + +.flex-justify-between { justify-content: space-between !important; } + +.flex-justify-around { justify-content: space-around !important; } + +.v-align-baseline { vertical-align: baseline !important; } + +.v-align-bottom { vertical-align: bottom !important; } + +.v-align-middle { vertical-align: middle !important; } + +.v-align-text-bottom { vertical-align: text-bottom !important; } + +.v-align-text-top { vertical-align: text-top !important; } + +.v-align-top { vertical-align: top !important; } + +.fs-1 { font-size: 0.5625rem !important; } +@media (min-width: 31.25rem) { .fs-1 { font-size: 0.625rem !important; } } + +.fs-2 { font-size: 0.6875rem !important; } +@media (min-width: 31.25rem) { .fs-2 { font-size: 0.75rem !important; } } + +.fs-3 { font-size: 0.75rem !important; } +@media (min-width: 31.25rem) { .fs-3 { font-size: 0.875rem !important; } } + +.fs-4 { font-size: 0.875rem !important; } +@media (min-width: 31.25rem) { .fs-4 { font-size: 1rem !important; } } + +.fs-5 { font-size: 1rem !important; } +@media (min-width: 31.25rem) { .fs-5 { font-size: 1.125rem !important; } } + +.fs-6 { font-size: 1.125rem !important; } +@media (min-width: 31.25rem) { .fs-6 { font-size: 1.5rem !important; line-height: 1.25; } } + +.fs-7 { font-size: 1.5rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-7 { font-size: 2rem !important; } } + +.fs-8 { font-size: 2rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-8 { font-size: 2.25rem !important; } } + +.fs-9 { font-size: 2.25rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-9 { font-size: 2.625rem !important; } } + +.fs-10 { font-size: 2.625rem !important; line-height: 1.25; } +@media (min-width: 31.25rem) { .fs-10 { font-size: 3rem !important; } } + +.fw-300 { font-weight: 300 !important; } + +.fw-400 { font-weight: 400 !important; } + +.fw-500 { font-weight: 500 !important; } + +.fw-700 { font-weight: 700 !important; } + +.lh-0 { line-height: 0 !important; } + +.lh-default { line-height: 1.4; } + +.lh-tight { line-height: 1.25; } + +.ls-5 { letter-spacing: 0.05em !important; } + +.ls-10 { letter-spacing: 0.1em !important; } + +.ls-0 { letter-spacing: 0 !important; } + +.text-uppercase { text-transform: uppercase !important; } + +.list-style-none { padding: 0 !important; margin: 0 !important; list-style: none !important; } +.list-style-none li::before { display: none !important; } + +.mx-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-0 { margin: 0 !important; } + +.mt-0 { margin-top: 0 !important; } + +.mr-0 { margin-right: 0 !important; } + +.mb-0 { margin-bottom: 0 !important; } + +.ml-0 { margin-left: 0 !important; } + +.mx-0 { margin-right: 0 !important; margin-left: 0 !important; } + +.my-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + +.mxn-0 { margin-right: -0 !important; margin-left: -0 !important; } + +.mx-0-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-1 { margin: 0.25rem !important; } + +.mt-1 { margin-top: 0.25rem !important; } + +.mr-1 { margin-right: 0.25rem !important; } + +.mb-1 { margin-bottom: 0.25rem !important; } + +.ml-1 { margin-left: 0.25rem !important; } + +.mx-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + +.my-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + +.mxn-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } + +.mx-1-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-2 { margin: 0.5rem !important; } + +.mt-2 { margin-top: 0.5rem !important; } + +.mr-2 { margin-right: 0.5rem !important; } + +.mb-2 { margin-bottom: 0.5rem !important; } + +.ml-2 { margin-left: 0.5rem !important; } + +.mx-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + +.my-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + +.mxn-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } + +.mx-2-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-3 { margin: 0.75rem !important; } + +.mt-3 { margin-top: 0.75rem !important; } + +.mr-3 { margin-right: 0.75rem !important; } + +.mb-3 { margin-bottom: 0.75rem !important; } + +.ml-3 { margin-left: 0.75rem !important; } + +.mx-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + +.my-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + +.mxn-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } + +.mx-3-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-4 { margin: 1rem !important; } + +.mt-4 { margin-top: 1rem !important; } + +.mr-4 { margin-right: 1rem !important; } + +.mb-4 { margin-bottom: 1rem !important; } + +.ml-4 { margin-left: 1rem !important; } + +.mx-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + +.my-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + +.mxn-4 { margin-right: -1rem !important; margin-left: -1rem !important; } + +.mx-4-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-5 { margin: 1.5rem !important; } + +.mt-5 { margin-top: 1.5rem !important; } + +.mr-5 { margin-right: 1.5rem !important; } + +.mb-5 { margin-bottom: 1.5rem !important; } + +.ml-5 { margin-left: 1.5rem !important; } + +.mx-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + +.my-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + +.mxn-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } + +.mx-5-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-6 { margin: 2rem !important; } + +.mt-6 { margin-top: 2rem !important; } + +.mr-6 { margin-right: 2rem !important; } + +.mb-6 { margin-bottom: 2rem !important; } + +.ml-6 { margin-left: 2rem !important; } + +.mx-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + +.my-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + +.mxn-6 { margin-right: -2rem !important; margin-left: -2rem !important; } + +.mx-6-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-7 { margin: 2.5rem !important; } + +.mt-7 { margin-top: 2.5rem !important; } + +.mr-7 { margin-right: 2.5rem !important; } + +.mb-7 { margin-bottom: 2.5rem !important; } + +.ml-7 { margin-left: 2.5rem !important; } + +.mx-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + +.my-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + +.mxn-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } + +.mx-7-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-8 { margin: 3rem !important; } + +.mt-8 { margin-top: 3rem !important; } + +.mr-8 { margin-right: 3rem !important; } + +.mb-8 { margin-bottom: 3rem !important; } + +.ml-8 { margin-left: 3rem !important; } + +.mx-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + +.my-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + +.mxn-8 { margin-right: -3rem !important; margin-left: -3rem !important; } + +.mx-8-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-9 { margin: 3.5rem !important; } + +.mt-9 { margin-top: 3.5rem !important; } + +.mr-9 { margin-right: 3.5rem !important; } + +.mb-9 { margin-bottom: 3.5rem !important; } + +.ml-9 { margin-left: 3.5rem !important; } + +.mx-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + +.my-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + +.mxn-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } + +.mx-9-auto { margin-right: auto !important; margin-left: auto !important; } + +.m-10 { margin: 4rem !important; } + +.mt-10 { margin-top: 4rem !important; } + +.mr-10 { margin-right: 4rem !important; } + +.mb-10 { margin-bottom: 4rem !important; } + +.ml-10 { margin-left: 4rem !important; } + +.mx-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + +.my-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + +.mxn-10 { margin-right: -4rem !important; margin-left: -4rem !important; } + +.mx-10-auto { margin-right: auto !important; margin-left: auto !important; } + +@media (min-width: 20rem) { .m-xs-0 { margin: 0 !important; } + .mt-xs-0 { margin-top: 0 !important; } + .mr-xs-0 { margin-right: 0 !important; } + .mb-xs-0 { margin-bottom: 0 !important; } + .ml-xs-0 { margin-left: 0 !important; } + .mx-xs-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xs-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xs-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 20rem) { .m-xs-1 { margin: 0.25rem !important; } + .mt-xs-1 { margin-top: 0.25rem !important; } + .mr-xs-1 { margin-right: 0.25rem !important; } + .mb-xs-1 { margin-bottom: 0.25rem !important; } + .ml-xs-1 { margin-left: 0.25rem !important; } + .mx-xs-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xs-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xs-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 20rem) { .m-xs-2 { margin: 0.5rem !important; } + .mt-xs-2 { margin-top: 0.5rem !important; } + .mr-xs-2 { margin-right: 0.5rem !important; } + .mb-xs-2 { margin-bottom: 0.5rem !important; } + .ml-xs-2 { margin-left: 0.5rem !important; } + .mx-xs-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xs-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xs-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 20rem) { .m-xs-3 { margin: 0.75rem !important; } + .mt-xs-3 { margin-top: 0.75rem !important; } + .mr-xs-3 { margin-right: 0.75rem !important; } + .mb-xs-3 { margin-bottom: 0.75rem !important; } + .ml-xs-3 { margin-left: 0.75rem !important; } + .mx-xs-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xs-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xs-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 20rem) { .m-xs-4 { margin: 1rem !important; } + .mt-xs-4 { margin-top: 1rem !important; } + .mr-xs-4 { margin-right: 1rem !important; } + .mb-xs-4 { margin-bottom: 1rem !important; } + .ml-xs-4 { margin-left: 1rem !important; } + .mx-xs-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xs-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xs-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 20rem) { .m-xs-5 { margin: 1.5rem !important; } + .mt-xs-5 { margin-top: 1.5rem !important; } + .mr-xs-5 { margin-right: 1.5rem !important; } + .mb-xs-5 { margin-bottom: 1.5rem !important; } + .ml-xs-5 { margin-left: 1.5rem !important; } + .mx-xs-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xs-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xs-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 20rem) { .m-xs-6 { margin: 2rem !important; } + .mt-xs-6 { margin-top: 2rem !important; } + .mr-xs-6 { margin-right: 2rem !important; } + .mb-xs-6 { margin-bottom: 2rem !important; } + .ml-xs-6 { margin-left: 2rem !important; } + .mx-xs-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xs-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xs-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 20rem) { .m-xs-7 { margin: 2.5rem !important; } + .mt-xs-7 { margin-top: 2.5rem !important; } + .mr-xs-7 { margin-right: 2.5rem !important; } + .mb-xs-7 { margin-bottom: 2.5rem !important; } + .ml-xs-7 { margin-left: 2.5rem !important; } + .mx-xs-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xs-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xs-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 20rem) { .m-xs-8 { margin: 3rem !important; } + .mt-xs-8 { margin-top: 3rem !important; } + .mr-xs-8 { margin-right: 3rem !important; } + .mb-xs-8 { margin-bottom: 3rem !important; } + .ml-xs-8 { margin-left: 3rem !important; } + .mx-xs-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xs-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xs-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 20rem) { .m-xs-9 { margin: 3.5rem !important; } + .mt-xs-9 { margin-top: 3.5rem !important; } + .mr-xs-9 { margin-right: 3.5rem !important; } + .mb-xs-9 { margin-bottom: 3.5rem !important; } + .ml-xs-9 { margin-left: 3.5rem !important; } + .mx-xs-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xs-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xs-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 20rem) { .m-xs-10 { margin: 4rem !important; } + .mt-xs-10 { margin-top: 4rem !important; } + .mr-xs-10 { margin-right: 4rem !important; } + .mb-xs-10 { margin-bottom: 4rem !important; } + .ml-xs-10 { margin-left: 4rem !important; } + .mx-xs-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xs-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xs-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 31.25rem) { .m-sm-0 { margin: 0 !important; } + .mt-sm-0 { margin-top: 0 !important; } + .mr-sm-0 { margin-right: 0 !important; } + .mb-sm-0 { margin-bottom: 0 !important; } + .ml-sm-0 { margin-left: 0 !important; } + .mx-sm-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-sm-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-sm-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 31.25rem) { .m-sm-1 { margin: 0.25rem !important; } + .mt-sm-1 { margin-top: 0.25rem !important; } + .mr-sm-1 { margin-right: 0.25rem !important; } + .mb-sm-1 { margin-bottom: 0.25rem !important; } + .ml-sm-1 { margin-left: 0.25rem !important; } + .mx-sm-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-sm-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-sm-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 31.25rem) { .m-sm-2 { margin: 0.5rem !important; } + .mt-sm-2 { margin-top: 0.5rem !important; } + .mr-sm-2 { margin-right: 0.5rem !important; } + .mb-sm-2 { margin-bottom: 0.5rem !important; } + .ml-sm-2 { margin-left: 0.5rem !important; } + .mx-sm-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-sm-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-sm-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-3 { margin: 0.75rem !important; } + .mt-sm-3 { margin-top: 0.75rem !important; } + .mr-sm-3 { margin-right: 0.75rem !important; } + .mb-sm-3 { margin-bottom: 0.75rem !important; } + .ml-sm-3 { margin-left: 0.75rem !important; } + .mx-sm-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-sm-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-sm-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 31.25rem) { .m-sm-4 { margin: 1rem !important; } + .mt-sm-4 { margin-top: 1rem !important; } + .mr-sm-4 { margin-right: 1rem !important; } + .mb-sm-4 { margin-bottom: 1rem !important; } + .ml-sm-4 { margin-left: 1rem !important; } + .mx-sm-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-sm-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-sm-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 31.25rem) { .m-sm-5 { margin: 1.5rem !important; } + .mt-sm-5 { margin-top: 1.5rem !important; } + .mr-sm-5 { margin-right: 1.5rem !important; } + .mb-sm-5 { margin-bottom: 1.5rem !important; } + .ml-sm-5 { margin-left: 1.5rem !important; } + .mx-sm-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-sm-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-sm-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-6 { margin: 2rem !important; } + .mt-sm-6 { margin-top: 2rem !important; } + .mr-sm-6 { margin-right: 2rem !important; } + .mb-sm-6 { margin-bottom: 2rem !important; } + .ml-sm-6 { margin-left: 2rem !important; } + .mx-sm-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-sm-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-sm-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 31.25rem) { .m-sm-7 { margin: 2.5rem !important; } + .mt-sm-7 { margin-top: 2.5rem !important; } + .mr-sm-7 { margin-right: 2.5rem !important; } + .mb-sm-7 { margin-bottom: 2.5rem !important; } + .ml-sm-7 { margin-left: 2.5rem !important; } + .mx-sm-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-sm-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-sm-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-8 { margin: 3rem !important; } + .mt-sm-8 { margin-top: 3rem !important; } + .mr-sm-8 { margin-right: 3rem !important; } + .mb-sm-8 { margin-bottom: 3rem !important; } + .ml-sm-8 { margin-left: 3rem !important; } + .mx-sm-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-sm-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-sm-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 31.25rem) { .m-sm-9 { margin: 3.5rem !important; } + .mt-sm-9 { margin-top: 3.5rem !important; } + .mr-sm-9 { margin-right: 3.5rem !important; } + .mb-sm-9 { margin-bottom: 3.5rem !important; } + .ml-sm-9 { margin-left: 3.5rem !important; } + .mx-sm-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-sm-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-sm-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 31.25rem) { .m-sm-10 { margin: 4rem !important; } + .mt-sm-10 { margin-top: 4rem !important; } + .mr-sm-10 { margin-right: 4rem !important; } + .mb-sm-10 { margin-bottom: 4rem !important; } + .ml-sm-10 { margin-left: 4rem !important; } + .mx-sm-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-sm-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-sm-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 50rem) { .m-md-0 { margin: 0 !important; } + .mt-md-0 { margin-top: 0 !important; } + .mr-md-0 { margin-right: 0 !important; } + .mb-md-0 { margin-bottom: 0 !important; } + .ml-md-0 { margin-left: 0 !important; } + .mx-md-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-md-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-md-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 50rem) { .m-md-1 { margin: 0.25rem !important; } + .mt-md-1 { margin-top: 0.25rem !important; } + .mr-md-1 { margin-right: 0.25rem !important; } + .mb-md-1 { margin-bottom: 0.25rem !important; } + .ml-md-1 { margin-left: 0.25rem !important; } + .mx-md-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-md-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-md-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 50rem) { .m-md-2 { margin: 0.5rem !important; } + .mt-md-2 { margin-top: 0.5rem !important; } + .mr-md-2 { margin-right: 0.5rem !important; } + .mb-md-2 { margin-bottom: 0.5rem !important; } + .ml-md-2 { margin-left: 0.5rem !important; } + .mx-md-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-md-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-md-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 50rem) { .m-md-3 { margin: 0.75rem !important; } + .mt-md-3 { margin-top: 0.75rem !important; } + .mr-md-3 { margin-right: 0.75rem !important; } + .mb-md-3 { margin-bottom: 0.75rem !important; } + .ml-md-3 { margin-left: 0.75rem !important; } + .mx-md-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-md-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-md-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 50rem) { .m-md-4 { margin: 1rem !important; } + .mt-md-4 { margin-top: 1rem !important; } + .mr-md-4 { margin-right: 1rem !important; } + .mb-md-4 { margin-bottom: 1rem !important; } + .ml-md-4 { margin-left: 1rem !important; } + .mx-md-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-md-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-md-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 50rem) { .m-md-5 { margin: 1.5rem !important; } + .mt-md-5 { margin-top: 1.5rem !important; } + .mr-md-5 { margin-right: 1.5rem !important; } + .mb-md-5 { margin-bottom: 1.5rem !important; } + .ml-md-5 { margin-left: 1.5rem !important; } + .mx-md-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-md-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-md-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 50rem) { .m-md-6 { margin: 2rem !important; } + .mt-md-6 { margin-top: 2rem !important; } + .mr-md-6 { margin-right: 2rem !important; } + .mb-md-6 { margin-bottom: 2rem !important; } + .ml-md-6 { margin-left: 2rem !important; } + .mx-md-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-md-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-md-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 50rem) { .m-md-7 { margin: 2.5rem !important; } + .mt-md-7 { margin-top: 2.5rem !important; } + .mr-md-7 { margin-right: 2.5rem !important; } + .mb-md-7 { margin-bottom: 2.5rem !important; } + .ml-md-7 { margin-left: 2.5rem !important; } + .mx-md-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-md-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-md-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 50rem) { .m-md-8 { margin: 3rem !important; } + .mt-md-8 { margin-top: 3rem !important; } + .mr-md-8 { margin-right: 3rem !important; } + .mb-md-8 { margin-bottom: 3rem !important; } + .ml-md-8 { margin-left: 3rem !important; } + .mx-md-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-md-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-md-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 50rem) { .m-md-9 { margin: 3.5rem !important; } + .mt-md-9 { margin-top: 3.5rem !important; } + .mr-md-9 { margin-right: 3.5rem !important; } + .mb-md-9 { margin-bottom: 3.5rem !important; } + .ml-md-9 { margin-left: 3.5rem !important; } + .mx-md-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-md-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-md-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 50rem) { .m-md-10 { margin: 4rem !important; } + .mt-md-10 { margin-top: 4rem !important; } + .mr-md-10 { margin-right: 4rem !important; } + .mb-md-10 { margin-bottom: 4rem !important; } + .ml-md-10 { margin-left: 4rem !important; } + .mx-md-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-md-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-md-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 66.5rem) { .m-lg-0 { margin: 0 !important; } + .mt-lg-0 { margin-top: 0 !important; } + .mr-lg-0 { margin-right: 0 !important; } + .mb-lg-0 { margin-bottom: 0 !important; } + .ml-lg-0 { margin-left: 0 !important; } + .mx-lg-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-lg-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-lg-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 66.5rem) { .m-lg-1 { margin: 0.25rem !important; } + .mt-lg-1 { margin-top: 0.25rem !important; } + .mr-lg-1 { margin-right: 0.25rem !important; } + .mb-lg-1 { margin-bottom: 0.25rem !important; } + .ml-lg-1 { margin-left: 0.25rem !important; } + .mx-lg-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-lg-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-lg-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 66.5rem) { .m-lg-2 { margin: 0.5rem !important; } + .mt-lg-2 { margin-top: 0.5rem !important; } + .mr-lg-2 { margin-right: 0.5rem !important; } + .mb-lg-2 { margin-bottom: 0.5rem !important; } + .ml-lg-2 { margin-left: 0.5rem !important; } + .mx-lg-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-lg-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-lg-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-3 { margin: 0.75rem !important; } + .mt-lg-3 { margin-top: 0.75rem !important; } + .mr-lg-3 { margin-right: 0.75rem !important; } + .mb-lg-3 { margin-bottom: 0.75rem !important; } + .ml-lg-3 { margin-left: 0.75rem !important; } + .mx-lg-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-lg-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-lg-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 66.5rem) { .m-lg-4 { margin: 1rem !important; } + .mt-lg-4 { margin-top: 1rem !important; } + .mr-lg-4 { margin-right: 1rem !important; } + .mb-lg-4 { margin-bottom: 1rem !important; } + .ml-lg-4 { margin-left: 1rem !important; } + .mx-lg-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-lg-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-lg-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 66.5rem) { .m-lg-5 { margin: 1.5rem !important; } + .mt-lg-5 { margin-top: 1.5rem !important; } + .mr-lg-5 { margin-right: 1.5rem !important; } + .mb-lg-5 { margin-bottom: 1.5rem !important; } + .ml-lg-5 { margin-left: 1.5rem !important; } + .mx-lg-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-lg-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-lg-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-6 { margin: 2rem !important; } + .mt-lg-6 { margin-top: 2rem !important; } + .mr-lg-6 { margin-right: 2rem !important; } + .mb-lg-6 { margin-bottom: 2rem !important; } + .ml-lg-6 { margin-left: 2rem !important; } + .mx-lg-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-lg-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-lg-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 66.5rem) { .m-lg-7 { margin: 2.5rem !important; } + .mt-lg-7 { margin-top: 2.5rem !important; } + .mr-lg-7 { margin-right: 2.5rem !important; } + .mb-lg-7 { margin-bottom: 2.5rem !important; } + .ml-lg-7 { margin-left: 2.5rem !important; } + .mx-lg-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-lg-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-lg-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-8 { margin: 3rem !important; } + .mt-lg-8 { margin-top: 3rem !important; } + .mr-lg-8 { margin-right: 3rem !important; } + .mb-lg-8 { margin-bottom: 3rem !important; } + .ml-lg-8 { margin-left: 3rem !important; } + .mx-lg-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-lg-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-lg-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 66.5rem) { .m-lg-9 { margin: 3.5rem !important; } + .mt-lg-9 { margin-top: 3.5rem !important; } + .mr-lg-9 { margin-right: 3.5rem !important; } + .mb-lg-9 { margin-bottom: 3.5rem !important; } + .ml-lg-9 { margin-left: 3.5rem !important; } + .mx-lg-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-lg-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-lg-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 66.5rem) { .m-lg-10 { margin: 4rem !important; } + .mt-lg-10 { margin-top: 4rem !important; } + .mr-lg-10 { margin-right: 4rem !important; } + .mb-lg-10 { margin-bottom: 4rem !important; } + .ml-lg-10 { margin-left: 4rem !important; } + .mx-lg-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-lg-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-lg-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +@media (min-width: 87.5rem) { .m-xl-0 { margin: 0 !important; } + .mt-xl-0 { margin-top: 0 !important; } + .mr-xl-0 { margin-right: 0 !important; } + .mb-xl-0 { margin-bottom: 0 !important; } + .ml-xl-0 { margin-left: 0 !important; } + .mx-xl-0 { margin-right: 0 !important; margin-left: 0 !important; } + .my-xl-0 { margin-top: 0 !important; margin-bottom: 0 !important; } + .mxn-xl-0 { margin-right: -0 !important; margin-left: -0 !important; } } +@media (min-width: 87.5rem) { .m-xl-1 { margin: 0.25rem !important; } + .mt-xl-1 { margin-top: 0.25rem !important; } + .mr-xl-1 { margin-right: 0.25rem !important; } + .mb-xl-1 { margin-bottom: 0.25rem !important; } + .ml-xl-1 { margin-left: 0.25rem !important; } + .mx-xl-1 { margin-right: 0.25rem !important; margin-left: 0.25rem !important; } + .my-xl-1 { margin-top: 0.25rem !important; margin-bottom: 0.25rem !important; } + .mxn-xl-1 { margin-right: -0.25rem !important; margin-left: -0.25rem !important; } } +@media (min-width: 87.5rem) { .m-xl-2 { margin: 0.5rem !important; } + .mt-xl-2 { margin-top: 0.5rem !important; } + .mr-xl-2 { margin-right: 0.5rem !important; } + .mb-xl-2 { margin-bottom: 0.5rem !important; } + .ml-xl-2 { margin-left: 0.5rem !important; } + .mx-xl-2 { margin-right: 0.5rem !important; margin-left: 0.5rem !important; } + .my-xl-2 { margin-top: 0.5rem !important; margin-bottom: 0.5rem !important; } + .mxn-xl-2 { margin-right: -0.5rem !important; margin-left: -0.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-3 { margin: 0.75rem !important; } + .mt-xl-3 { margin-top: 0.75rem !important; } + .mr-xl-3 { margin-right: 0.75rem !important; } + .mb-xl-3 { margin-bottom: 0.75rem !important; } + .ml-xl-3 { margin-left: 0.75rem !important; } + .mx-xl-3 { margin-right: 0.75rem !important; margin-left: 0.75rem !important; } + .my-xl-3 { margin-top: 0.75rem !important; margin-bottom: 0.75rem !important; } + .mxn-xl-3 { margin-right: -0.75rem !important; margin-left: -0.75rem !important; } } +@media (min-width: 87.5rem) { .m-xl-4 { margin: 1rem !important; } + .mt-xl-4 { margin-top: 1rem !important; } + .mr-xl-4 { margin-right: 1rem !important; } + .mb-xl-4 { margin-bottom: 1rem !important; } + .ml-xl-4 { margin-left: 1rem !important; } + .mx-xl-4 { margin-right: 1rem !important; margin-left: 1rem !important; } + .my-xl-4 { margin-top: 1rem !important; margin-bottom: 1rem !important; } + .mxn-xl-4 { margin-right: -1rem !important; margin-left: -1rem !important; } } +@media (min-width: 87.5rem) { .m-xl-5 { margin: 1.5rem !important; } + .mt-xl-5 { margin-top: 1.5rem !important; } + .mr-xl-5 { margin-right: 1.5rem !important; } + .mb-xl-5 { margin-bottom: 1.5rem !important; } + .ml-xl-5 { margin-left: 1.5rem !important; } + .mx-xl-5 { margin-right: 1.5rem !important; margin-left: 1.5rem !important; } + .my-xl-5 { margin-top: 1.5rem !important; margin-bottom: 1.5rem !important; } + .mxn-xl-5 { margin-right: -1.5rem !important; margin-left: -1.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-6 { margin: 2rem !important; } + .mt-xl-6 { margin-top: 2rem !important; } + .mr-xl-6 { margin-right: 2rem !important; } + .mb-xl-6 { margin-bottom: 2rem !important; } + .ml-xl-6 { margin-left: 2rem !important; } + .mx-xl-6 { margin-right: 2rem !important; margin-left: 2rem !important; } + .my-xl-6 { margin-top: 2rem !important; margin-bottom: 2rem !important; } + .mxn-xl-6 { margin-right: -2rem !important; margin-left: -2rem !important; } } +@media (min-width: 87.5rem) { .m-xl-7 { margin: 2.5rem !important; } + .mt-xl-7 { margin-top: 2.5rem !important; } + .mr-xl-7 { margin-right: 2.5rem !important; } + .mb-xl-7 { margin-bottom: 2.5rem !important; } + .ml-xl-7 { margin-left: 2.5rem !important; } + .mx-xl-7 { margin-right: 2.5rem !important; margin-left: 2.5rem !important; } + .my-xl-7 { margin-top: 2.5rem !important; margin-bottom: 2.5rem !important; } + .mxn-xl-7 { margin-right: -2.5rem !important; margin-left: -2.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-8 { margin: 3rem !important; } + .mt-xl-8 { margin-top: 3rem !important; } + .mr-xl-8 { margin-right: 3rem !important; } + .mb-xl-8 { margin-bottom: 3rem !important; } + .ml-xl-8 { margin-left: 3rem !important; } + .mx-xl-8 { margin-right: 3rem !important; margin-left: 3rem !important; } + .my-xl-8 { margin-top: 3rem !important; margin-bottom: 3rem !important; } + .mxn-xl-8 { margin-right: -3rem !important; margin-left: -3rem !important; } } +@media (min-width: 87.5rem) { .m-xl-9 { margin: 3.5rem !important; } + .mt-xl-9 { margin-top: 3.5rem !important; } + .mr-xl-9 { margin-right: 3.5rem !important; } + .mb-xl-9 { margin-bottom: 3.5rem !important; } + .ml-xl-9 { margin-left: 3.5rem !important; } + .mx-xl-9 { margin-right: 3.5rem !important; margin-left: 3.5rem !important; } + .my-xl-9 { margin-top: 3.5rem !important; margin-bottom: 3.5rem !important; } + .mxn-xl-9 { margin-right: -3.5rem !important; margin-left: -3.5rem !important; } } +@media (min-width: 87.5rem) { .m-xl-10 { margin: 4rem !important; } + .mt-xl-10 { margin-top: 4rem !important; } + .mr-xl-10 { margin-right: 4rem !important; } + .mb-xl-10 { margin-bottom: 4rem !important; } + .ml-xl-10 { margin-left: 4rem !important; } + .mx-xl-10 { margin-right: 4rem !important; margin-left: 4rem !important; } + .my-xl-10 { margin-top: 4rem !important; margin-bottom: 4rem !important; } + .mxn-xl-10 { margin-right: -4rem !important; margin-left: -4rem !important; } } +.p-0 { padding: 0 !important; } + +.pt-0 { padding-top: 0 !important; } + +.pr-0 { padding-right: 0 !important; } + +.pb-0 { padding-bottom: 0 !important; } + +.pl-0 { padding-left: 0 !important; } + +.px-0 { padding-right: 0 !important; padding-left: 0 !important; } + +.py-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + +.p-1 { padding: 0.25rem !important; } + +.pt-1 { padding-top: 0.25rem !important; } + +.pr-1 { padding-right: 0.25rem !important; } + +.pb-1 { padding-bottom: 0.25rem !important; } + +.pl-1 { padding-left: 0.25rem !important; } + +.px-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + +.py-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + +.p-2 { padding: 0.5rem !important; } + +.pt-2 { padding-top: 0.5rem !important; } + +.pr-2 { padding-right: 0.5rem !important; } + +.pb-2 { padding-bottom: 0.5rem !important; } + +.pl-2 { padding-left: 0.5rem !important; } + +.px-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + +.py-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + +.p-3 { padding: 0.75rem !important; } + +.pt-3 { padding-top: 0.75rem !important; } + +.pr-3 { padding-right: 0.75rem !important; } + +.pb-3 { padding-bottom: 0.75rem !important; } + +.pl-3 { padding-left: 0.75rem !important; } + +.px-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + +.py-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + +.p-4 { padding: 1rem !important; } + +.pt-4 { padding-top: 1rem !important; } + +.pr-4 { padding-right: 1rem !important; } + +.pb-4 { padding-bottom: 1rem !important; } + +.pl-4 { padding-left: 1rem !important; } + +.px-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + +.py-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + +.p-5 { padding: 1.5rem !important; } + +.pt-5 { padding-top: 1.5rem !important; } + +.pr-5 { padding-right: 1.5rem !important; } + +.pb-5 { padding-bottom: 1.5rem !important; } + +.pl-5 { padding-left: 1.5rem !important; } + +.px-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + +.py-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + +.p-6 { padding: 2rem !important; } + +.pt-6 { padding-top: 2rem !important; } + +.pr-6 { padding-right: 2rem !important; } + +.pb-6 { padding-bottom: 2rem !important; } + +.pl-6 { padding-left: 2rem !important; } + +.px-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + +.py-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + +.p-7 { padding: 2.5rem !important; } + +.pt-7 { padding-top: 2.5rem !important; } + +.pr-7 { padding-right: 2.5rem !important; } + +.pb-7 { padding-bottom: 2.5rem !important; } + +.pl-7 { padding-left: 2.5rem !important; } + +.px-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + +.py-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + +.p-8 { padding: 3rem !important; } + +.pt-8 { padding-top: 3rem !important; } + +.pr-8 { padding-right: 3rem !important; } + +.pb-8 { padding-bottom: 3rem !important; } + +.pl-8 { padding-left: 3rem !important; } + +.px-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + +.py-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + +.p-9 { padding: 3.5rem !important; } + +.pt-9 { padding-top: 3.5rem !important; } + +.pr-9 { padding-right: 3.5rem !important; } + +.pb-9 { padding-bottom: 3.5rem !important; } + +.pl-9 { padding-left: 3.5rem !important; } + +.px-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + +.py-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + +.p-10 { padding: 4rem !important; } + +.pt-10 { padding-top: 4rem !important; } + +.pr-10 { padding-right: 4rem !important; } + +.pb-10 { padding-bottom: 4rem !important; } + +.pl-10 { padding-left: 4rem !important; } + +.px-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + +.py-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } + +@media (min-width: 20rem) { .p-xs-0 { padding: 0 !important; } + .pt-xs-0 { padding-top: 0 !important; } + .pr-xs-0 { padding-right: 0 !important; } + .pb-xs-0 { padding-bottom: 0 !important; } + .pl-xs-0 { padding-left: 0 !important; } + .px-xs-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xs-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xs-1 { padding: 0.25rem !important; } + .pt-xs-1 { padding-top: 0.25rem !important; } + .pr-xs-1 { padding-right: 0.25rem !important; } + .pb-xs-1 { padding-bottom: 0.25rem !important; } + .pl-xs-1 { padding-left: 0.25rem !important; } + .px-xs-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xs-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xs-2 { padding: 0.5rem !important; } + .pt-xs-2 { padding-top: 0.5rem !important; } + .pr-xs-2 { padding-right: 0.5rem !important; } + .pb-xs-2 { padding-bottom: 0.5rem !important; } + .pl-xs-2 { padding-left: 0.5rem !important; } + .px-xs-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xs-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xs-3 { padding: 0.75rem !important; } + .pt-xs-3 { padding-top: 0.75rem !important; } + .pr-xs-3 { padding-right: 0.75rem !important; } + .pb-xs-3 { padding-bottom: 0.75rem !important; } + .pl-xs-3 { padding-left: 0.75rem !important; } + .px-xs-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xs-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xs-4 { padding: 1rem !important; } + .pt-xs-4 { padding-top: 1rem !important; } + .pr-xs-4 { padding-right: 1rem !important; } + .pb-xs-4 { padding-bottom: 1rem !important; } + .pl-xs-4 { padding-left: 1rem !important; } + .px-xs-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xs-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xs-5 { padding: 1.5rem !important; } + .pt-xs-5 { padding-top: 1.5rem !important; } + .pr-xs-5 { padding-right: 1.5rem !important; } + .pb-xs-5 { padding-bottom: 1.5rem !important; } + .pl-xs-5 { padding-left: 1.5rem !important; } + .px-xs-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xs-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xs-6 { padding: 2rem !important; } + .pt-xs-6 { padding-top: 2rem !important; } + .pr-xs-6 { padding-right: 2rem !important; } + .pb-xs-6 { padding-bottom: 2rem !important; } + .pl-xs-6 { padding-left: 2rem !important; } + .px-xs-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xs-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xs-7 { padding: 2.5rem !important; } + .pt-xs-7 { padding-top: 2.5rem !important; } + .pr-xs-7 { padding-right: 2.5rem !important; } + .pb-xs-7 { padding-bottom: 2.5rem !important; } + .pl-xs-7 { padding-left: 2.5rem !important; } + .px-xs-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xs-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xs-8 { padding: 3rem !important; } + .pt-xs-8 { padding-top: 3rem !important; } + .pr-xs-8 { padding-right: 3rem !important; } + .pb-xs-8 { padding-bottom: 3rem !important; } + .pl-xs-8 { padding-left: 3rem !important; } + .px-xs-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xs-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xs-9 { padding: 3.5rem !important; } + .pt-xs-9 { padding-top: 3.5rem !important; } + .pr-xs-9 { padding-right: 3.5rem !important; } + .pb-xs-9 { padding-bottom: 3.5rem !important; } + .pl-xs-9 { padding-left: 3.5rem !important; } + .px-xs-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xs-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xs-10 { padding: 4rem !important; } + .pt-xs-10 { padding-top: 4rem !important; } + .pr-xs-10 { padding-right: 4rem !important; } + .pb-xs-10 { padding-bottom: 4rem !important; } + .pl-xs-10 { padding-left: 4rem !important; } + .px-xs-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xs-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 31.25rem) { .p-sm-0 { padding: 0 !important; } + .pt-sm-0 { padding-top: 0 !important; } + .pr-sm-0 { padding-right: 0 !important; } + .pb-sm-0 { padding-bottom: 0 !important; } + .pl-sm-0 { padding-left: 0 !important; } + .px-sm-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-sm-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-sm-1 { padding: 0.25rem !important; } + .pt-sm-1 { padding-top: 0.25rem !important; } + .pr-sm-1 { padding-right: 0.25rem !important; } + .pb-sm-1 { padding-bottom: 0.25rem !important; } + .pl-sm-1 { padding-left: 0.25rem !important; } + .px-sm-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-sm-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-sm-2 { padding: 0.5rem !important; } + .pt-sm-2 { padding-top: 0.5rem !important; } + .pr-sm-2 { padding-right: 0.5rem !important; } + .pb-sm-2 { padding-bottom: 0.5rem !important; } + .pl-sm-2 { padding-left: 0.5rem !important; } + .px-sm-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-sm-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-sm-3 { padding: 0.75rem !important; } + .pt-sm-3 { padding-top: 0.75rem !important; } + .pr-sm-3 { padding-right: 0.75rem !important; } + .pb-sm-3 { padding-bottom: 0.75rem !important; } + .pl-sm-3 { padding-left: 0.75rem !important; } + .px-sm-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-sm-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-sm-4 { padding: 1rem !important; } + .pt-sm-4 { padding-top: 1rem !important; } + .pr-sm-4 { padding-right: 1rem !important; } + .pb-sm-4 { padding-bottom: 1rem !important; } + .pl-sm-4 { padding-left: 1rem !important; } + .px-sm-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-sm-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-sm-5 { padding: 1.5rem !important; } + .pt-sm-5 { padding-top: 1.5rem !important; } + .pr-sm-5 { padding-right: 1.5rem !important; } + .pb-sm-5 { padding-bottom: 1.5rem !important; } + .pl-sm-5 { padding-left: 1.5rem !important; } + .px-sm-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-sm-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-sm-6 { padding: 2rem !important; } + .pt-sm-6 { padding-top: 2rem !important; } + .pr-sm-6 { padding-right: 2rem !important; } + .pb-sm-6 { padding-bottom: 2rem !important; } + .pl-sm-6 { padding-left: 2rem !important; } + .px-sm-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-sm-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-sm-7 { padding: 2.5rem !important; } + .pt-sm-7 { padding-top: 2.5rem !important; } + .pr-sm-7 { padding-right: 2.5rem !important; } + .pb-sm-7 { padding-bottom: 2.5rem !important; } + .pl-sm-7 { padding-left: 2.5rem !important; } + .px-sm-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-sm-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-sm-8 { padding: 3rem !important; } + .pt-sm-8 { padding-top: 3rem !important; } + .pr-sm-8 { padding-right: 3rem !important; } + .pb-sm-8 { padding-bottom: 3rem !important; } + .pl-sm-8 { padding-left: 3rem !important; } + .px-sm-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-sm-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-sm-9 { padding: 3.5rem !important; } + .pt-sm-9 { padding-top: 3.5rem !important; } + .pr-sm-9 { padding-right: 3.5rem !important; } + .pb-sm-9 { padding-bottom: 3.5rem !important; } + .pl-sm-9 { padding-left: 3.5rem !important; } + .px-sm-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-sm-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-sm-10 { padding: 4rem !important; } + .pt-sm-10 { padding-top: 4rem !important; } + .pr-sm-10 { padding-right: 4rem !important; } + .pb-sm-10 { padding-bottom: 4rem !important; } + .pl-sm-10 { padding-left: 4rem !important; } + .px-sm-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-sm-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 50rem) { .p-md-0 { padding: 0 !important; } + .pt-md-0 { padding-top: 0 !important; } + .pr-md-0 { padding-right: 0 !important; } + .pb-md-0 { padding-bottom: 0 !important; } + .pl-md-0 { padding-left: 0 !important; } + .px-md-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-md-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-md-1 { padding: 0.25rem !important; } + .pt-md-1 { padding-top: 0.25rem !important; } + .pr-md-1 { padding-right: 0.25rem !important; } + .pb-md-1 { padding-bottom: 0.25rem !important; } + .pl-md-1 { padding-left: 0.25rem !important; } + .px-md-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-md-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-md-2 { padding: 0.5rem !important; } + .pt-md-2 { padding-top: 0.5rem !important; } + .pr-md-2 { padding-right: 0.5rem !important; } + .pb-md-2 { padding-bottom: 0.5rem !important; } + .pl-md-2 { padding-left: 0.5rem !important; } + .px-md-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-md-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-md-3 { padding: 0.75rem !important; } + .pt-md-3 { padding-top: 0.75rem !important; } + .pr-md-3 { padding-right: 0.75rem !important; } + .pb-md-3 { padding-bottom: 0.75rem !important; } + .pl-md-3 { padding-left: 0.75rem !important; } + .px-md-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-md-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-md-4 { padding: 1rem !important; } + .pt-md-4 { padding-top: 1rem !important; } + .pr-md-4 { padding-right: 1rem !important; } + .pb-md-4 { padding-bottom: 1rem !important; } + .pl-md-4 { padding-left: 1rem !important; } + .px-md-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-md-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-md-5 { padding: 1.5rem !important; } + .pt-md-5 { padding-top: 1.5rem !important; } + .pr-md-5 { padding-right: 1.5rem !important; } + .pb-md-5 { padding-bottom: 1.5rem !important; } + .pl-md-5 { padding-left: 1.5rem !important; } + .px-md-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-md-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-md-6 { padding: 2rem !important; } + .pt-md-6 { padding-top: 2rem !important; } + .pr-md-6 { padding-right: 2rem !important; } + .pb-md-6 { padding-bottom: 2rem !important; } + .pl-md-6 { padding-left: 2rem !important; } + .px-md-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-md-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-md-7 { padding: 2.5rem !important; } + .pt-md-7 { padding-top: 2.5rem !important; } + .pr-md-7 { padding-right: 2.5rem !important; } + .pb-md-7 { padding-bottom: 2.5rem !important; } + .pl-md-7 { padding-left: 2.5rem !important; } + .px-md-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-md-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-md-8 { padding: 3rem !important; } + .pt-md-8 { padding-top: 3rem !important; } + .pr-md-8 { padding-right: 3rem !important; } + .pb-md-8 { padding-bottom: 3rem !important; } + .pl-md-8 { padding-left: 3rem !important; } + .px-md-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-md-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-md-9 { padding: 3.5rem !important; } + .pt-md-9 { padding-top: 3.5rem !important; } + .pr-md-9 { padding-right: 3.5rem !important; } + .pb-md-9 { padding-bottom: 3.5rem !important; } + .pl-md-9 { padding-left: 3.5rem !important; } + .px-md-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-md-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-md-10 { padding: 4rem !important; } + .pt-md-10 { padding-top: 4rem !important; } + .pr-md-10 { padding-right: 4rem !important; } + .pb-md-10 { padding-bottom: 4rem !important; } + .pl-md-10 { padding-left: 4rem !important; } + .px-md-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-md-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 66.5rem) { .p-lg-0 { padding: 0 !important; } + .pt-lg-0 { padding-top: 0 !important; } + .pr-lg-0 { padding-right: 0 !important; } + .pb-lg-0 { padding-bottom: 0 !important; } + .pl-lg-0 { padding-left: 0 !important; } + .px-lg-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-lg-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-lg-1 { padding: 0.25rem !important; } + .pt-lg-1 { padding-top: 0.25rem !important; } + .pr-lg-1 { padding-right: 0.25rem !important; } + .pb-lg-1 { padding-bottom: 0.25rem !important; } + .pl-lg-1 { padding-left: 0.25rem !important; } + .px-lg-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-lg-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-lg-2 { padding: 0.5rem !important; } + .pt-lg-2 { padding-top: 0.5rem !important; } + .pr-lg-2 { padding-right: 0.5rem !important; } + .pb-lg-2 { padding-bottom: 0.5rem !important; } + .pl-lg-2 { padding-left: 0.5rem !important; } + .px-lg-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-lg-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-lg-3 { padding: 0.75rem !important; } + .pt-lg-3 { padding-top: 0.75rem !important; } + .pr-lg-3 { padding-right: 0.75rem !important; } + .pb-lg-3 { padding-bottom: 0.75rem !important; } + .pl-lg-3 { padding-left: 0.75rem !important; } + .px-lg-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-lg-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-lg-4 { padding: 1rem !important; } + .pt-lg-4 { padding-top: 1rem !important; } + .pr-lg-4 { padding-right: 1rem !important; } + .pb-lg-4 { padding-bottom: 1rem !important; } + .pl-lg-4 { padding-left: 1rem !important; } + .px-lg-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-lg-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-lg-5 { padding: 1.5rem !important; } + .pt-lg-5 { padding-top: 1.5rem !important; } + .pr-lg-5 { padding-right: 1.5rem !important; } + .pb-lg-5 { padding-bottom: 1.5rem !important; } + .pl-lg-5 { padding-left: 1.5rem !important; } + .px-lg-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-lg-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-lg-6 { padding: 2rem !important; } + .pt-lg-6 { padding-top: 2rem !important; } + .pr-lg-6 { padding-right: 2rem !important; } + .pb-lg-6 { padding-bottom: 2rem !important; } + .pl-lg-6 { padding-left: 2rem !important; } + .px-lg-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-lg-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-lg-7 { padding: 2.5rem !important; } + .pt-lg-7 { padding-top: 2.5rem !important; } + .pr-lg-7 { padding-right: 2.5rem !important; } + .pb-lg-7 { padding-bottom: 2.5rem !important; } + .pl-lg-7 { padding-left: 2.5rem !important; } + .px-lg-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-lg-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-lg-8 { padding: 3rem !important; } + .pt-lg-8 { padding-top: 3rem !important; } + .pr-lg-8 { padding-right: 3rem !important; } + .pb-lg-8 { padding-bottom: 3rem !important; } + .pl-lg-8 { padding-left: 3rem !important; } + .px-lg-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-lg-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-lg-9 { padding: 3.5rem !important; } + .pt-lg-9 { padding-top: 3.5rem !important; } + .pr-lg-9 { padding-right: 3.5rem !important; } + .pb-lg-9 { padding-bottom: 3.5rem !important; } + .pl-lg-9 { padding-left: 3.5rem !important; } + .px-lg-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-lg-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-lg-10 { padding: 4rem !important; } + .pt-lg-10 { padding-top: 4rem !important; } + .pr-lg-10 { padding-right: 4rem !important; } + .pb-lg-10 { padding-bottom: 4rem !important; } + .pl-lg-10 { padding-left: 4rem !important; } + .px-lg-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-lg-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media (min-width: 87.5rem) { .p-xl-0 { padding: 0 !important; } + .pt-xl-0 { padding-top: 0 !important; } + .pr-xl-0 { padding-right: 0 !important; } + .pb-xl-0 { padding-bottom: 0 !important; } + .pl-xl-0 { padding-left: 0 !important; } + .px-xl-0 { padding-right: 0 !important; padding-left: 0 !important; } + .py-xl-0 { padding-top: 0 !important; padding-bottom: 0 !important; } + .p-xl-1 { padding: 0.25rem !important; } + .pt-xl-1 { padding-top: 0.25rem !important; } + .pr-xl-1 { padding-right: 0.25rem !important; } + .pb-xl-1 { padding-bottom: 0.25rem !important; } + .pl-xl-1 { padding-left: 0.25rem !important; } + .px-xl-1 { padding-right: 0.25rem !important; padding-left: 0.25rem !important; } + .py-xl-1 { padding-top: 0.25rem !important; padding-bottom: 0.25rem !important; } + .p-xl-2 { padding: 0.5rem !important; } + .pt-xl-2 { padding-top: 0.5rem !important; } + .pr-xl-2 { padding-right: 0.5rem !important; } + .pb-xl-2 { padding-bottom: 0.5rem !important; } + .pl-xl-2 { padding-left: 0.5rem !important; } + .px-xl-2 { padding-right: 0.5rem !important; padding-left: 0.5rem !important; } + .py-xl-2 { padding-top: 0.5rem !important; padding-bottom: 0.5rem !important; } + .p-xl-3 { padding: 0.75rem !important; } + .pt-xl-3 { padding-top: 0.75rem !important; } + .pr-xl-3 { padding-right: 0.75rem !important; } + .pb-xl-3 { padding-bottom: 0.75rem !important; } + .pl-xl-3 { padding-left: 0.75rem !important; } + .px-xl-3 { padding-right: 0.75rem !important; padding-left: 0.75rem !important; } + .py-xl-3 { padding-top: 0.75rem !important; padding-bottom: 0.75rem !important; } + .p-xl-4 { padding: 1rem !important; } + .pt-xl-4 { padding-top: 1rem !important; } + .pr-xl-4 { padding-right: 1rem !important; } + .pb-xl-4 { padding-bottom: 1rem !important; } + .pl-xl-4 { padding-left: 1rem !important; } + .px-xl-4 { padding-right: 1rem !important; padding-left: 1rem !important; } + .py-xl-4 { padding-top: 1rem !important; padding-bottom: 1rem !important; } + .p-xl-5 { padding: 1.5rem !important; } + .pt-xl-5 { padding-top: 1.5rem !important; } + .pr-xl-5 { padding-right: 1.5rem !important; } + .pb-xl-5 { padding-bottom: 1.5rem !important; } + .pl-xl-5 { padding-left: 1.5rem !important; } + .px-xl-5 { padding-right: 1.5rem !important; padding-left: 1.5rem !important; } + .py-xl-5 { padding-top: 1.5rem !important; padding-bottom: 1.5rem !important; } + .p-xl-6 { padding: 2rem !important; } + .pt-xl-6 { padding-top: 2rem !important; } + .pr-xl-6 { padding-right: 2rem !important; } + .pb-xl-6 { padding-bottom: 2rem !important; } + .pl-xl-6 { padding-left: 2rem !important; } + .px-xl-6 { padding-right: 2rem !important; padding-left: 2rem !important; } + .py-xl-6 { padding-top: 2rem !important; padding-bottom: 2rem !important; } + .p-xl-7 { padding: 2.5rem !important; } + .pt-xl-7 { padding-top: 2.5rem !important; } + .pr-xl-7 { padding-right: 2.5rem !important; } + .pb-xl-7 { padding-bottom: 2.5rem !important; } + .pl-xl-7 { padding-left: 2.5rem !important; } + .px-xl-7 { padding-right: 2.5rem !important; padding-left: 2.5rem !important; } + .py-xl-7 { padding-top: 2.5rem !important; padding-bottom: 2.5rem !important; } + .p-xl-8 { padding: 3rem !important; } + .pt-xl-8 { padding-top: 3rem !important; } + .pr-xl-8 { padding-right: 3rem !important; } + .pb-xl-8 { padding-bottom: 3rem !important; } + .pl-xl-8 { padding-left: 3rem !important; } + .px-xl-8 { padding-right: 3rem !important; padding-left: 3rem !important; } + .py-xl-8 { padding-top: 3rem !important; padding-bottom: 3rem !important; } + .p-xl-9 { padding: 3.5rem !important; } + .pt-xl-9 { padding-top: 3.5rem !important; } + .pr-xl-9 { padding-right: 3.5rem !important; } + .pb-xl-9 { padding-bottom: 3.5rem !important; } + .pl-xl-9 { padding-left: 3.5rem !important; } + .px-xl-9 { padding-right: 3.5rem !important; padding-left: 3.5rem !important; } + .py-xl-9 { padding-top: 3.5rem !important; padding-bottom: 3.5rem !important; } + .p-xl-10 { padding: 4rem !important; } + .pt-xl-10 { padding-top: 4rem !important; } + .pr-xl-10 { padding-right: 4rem !important; } + .pb-xl-10 { padding-bottom: 4rem !important; } + .pl-xl-10 { padding-left: 4rem !important; } + .px-xl-10 { padding-right: 4rem !important; padding-left: 4rem !important; } + .py-xl-10 { padding-top: 4rem !important; padding-bottom: 4rem !important; } } +@media print { .site-footer, .site-button, #edit-this-page, #back-to-top, .site-nav, .main-header { display: none !important; } + .side-bar { width: 100%; height: auto; border-right: 0 !important; } + .site-header { border-bottom: 1px solid #eeebee; } + .site-title { font-size: 1rem !important; font-weight: 700 !important; } + .text-small { font-size: 8pt !important; } + pre.highlight { border: 1px solid #eeebee; } + .main { max-width: none; margin-left: 0; } } +a.skip-to-main { left: -999px; position: absolute; top: auto; width: 1px; height: 1px; overflow: hidden; z-index: -999; } + +a.skip-to-main:focus, a.skip-to-main:active { color: #7253ed; background-color: #fff; left: auto; top: auto; width: 30%; height: auto; overflow: auto; margin: 10px 35%; padding: 5px; border-radius: 15px; border: 4px solid #5e41d0; text-align: center; font-size: 1.2em; z-index: 999; } + +div.opaque { background-color: #fff; } + +p.note, blockquote.note { background: #f5f7f9; border-left: 4px solid #2ce5b5; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note > .note-title, blockquote.note > .note-title { color: #2ce5b5; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.note-title, blockquote.note-title { background: #f5f7f9; border-left: 4px solid #2ce5b5; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.note-title > p:first-child, blockquote.note-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #2ce5b5; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.note { margin-left: 0; margin-right: 0; } +blockquote.note > p:first-child { margin-top: 0; } +blockquote.note > p:last-child { margin-bottom: 0; } + +blockquote.note-title { margin-left: 0; margin-right: 0; } +blockquote.note-title > p:nth-child(2) { margin-top: 0; } +blockquote.note-title > p:last-child { margin-bottom: 0; } + +p.warning, blockquote.warning { background: rgba(247, 126, 126, 0.2); border-left: 4px solid #dd2e2e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning::before, blockquote.warning::before { color: #dd2e2e; content: "⚠️ Warning ⚠️"; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } +p.warning > .warning-title, blockquote.warning > .warning-title { color: #dd2e2e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.warning-title, blockquote.warning-title { background: rgba(247, 126, 126, 0.2); border-left: 4px solid #dd2e2e; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.warning-title > p:first-child, blockquote.warning-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #dd2e2e; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.warning { margin-left: 0; margin-right: 0; } +blockquote.warning > p:first-child { margin-top: 0; } +blockquote.warning > p:last-child { margin-bottom: 0; } + +blockquote.warning-title { margin-left: 0; margin-right: 0; } +blockquote.warning-title > p:nth-child(2) { margin-top: 0; } +blockquote.warning-title > p:last-child { margin-bottom: 0; } + +p.tip, blockquote.tip { background: rgba(65, 214, 147, 0.2); border-left: 4px solid #026e57; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip > .tip-title, blockquote.tip > .tip-title { color: #026e57; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.tip-title, blockquote.tip-title { background: rgba(65, 214, 147, 0.2); border-left: 4px solid #026e57; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.tip-title > p:first-child, blockquote.tip-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #026e57; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.tip { margin-left: 0; margin-right: 0; } +blockquote.tip > p:first-child { margin-top: 0; } +blockquote.tip > p:last-child { margin-bottom: 0; } + +blockquote.tip-title { margin-left: 0; margin-right: 0; } +blockquote.tip-title > p:nth-child(2) { margin-top: 0; } +blockquote.tip-title > p:last-child { margin-bottom: 0; } + +p.fubar, blockquote.fubar { background: rgba(114, 83, 237, 0.2); border-left: 4px solid #381885; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar > .fubar-title, blockquote.fubar > .fubar-title { color: #381885; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +p.fubar-title, blockquote.fubar-title { background: rgba(114, 83, 237, 0.2); border-left: 4px solid #381885; border-radius: 4px; box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08); padding: .8rem; } +p.fubar-title > p:first-child, blockquote.fubar-title > p:first-child { margin-top: 0; margin-bottom: 0; color: #381885; display: block; font-weight: bold; text-transform: uppercase; font-size: .75em; padding-bottom: .125rem; } + +blockquote.fubar { margin-left: 0; margin-right: 0; } +blockquote.fubar > p:first-child { margin-top: 0; } +blockquote.fubar > p:last-child { margin-bottom: 0; } + +blockquote.fubar-title { margin-left: 0; margin-right: 0; } +blockquote.fubar-title > p:nth-child(2) { margin-top: 0; } +blockquote.fubar-title > p:last-child { margin-bottom: 0; } + +@font-face { font-family: GalanoGrotesque-ExtraLight; src: local(GalanoGrotesque-ExtraLight), url("../fonts/Rene Bieder - Galano Grotesque ExtraLight.otf"); } +@font-face { font-family: GalanoGrotesque; src: local(GalanoGrotesque-Regular), url("../fonts/Rene Bieder - Galano Grotesque.otf"); } +@font-face { font-family: GalanoGrotesque-Bold; src: local(GalanoGrotesque-Bold), url("../fonts/Rene Bieder - Galano Grotesque Bold.otf"); } +@font-face { font-family: GalanoGrotesque-Medium; src: local(Rene-Bieder-Galano-Grotesque-Medium), url("../fonts/Rene Bieder - Galano Grotesque Medium.otf"); } +html { scroll-padding-top: 90px; } + +body { font-family: GalanoGrotesque; color: #000000; } +body .body-wrapper, body footer { display: flex; } +body .body-wrapper .main, body .body-wrapper .footer-main, body footer .main, body footer .footer-main { width: 75%; max-width: 75%; margin-left: 0; float: right; position: relative; } +body .body-wrapper .main .main-content-wrap, body .body-wrapper .footer-main .main-content-wrap, body footer .main .main-content-wrap, body footer .footer-main .main-content-wrap { max-width: 1300px; padding: 80px; } +body .body-wrapper .side-bar, body .body-wrapper .footer-sidebar, body footer .side-bar, body footer .footer-sidebar { width: 25%; max-width: 25%; position: relative; height: auto; background-color: #f5f7f9; } +body a { color: #279890; } +body p, body li { font-family: GalanoGrotesque; font-size: 1em; } +body h1 { margin-top: 40px !important; font-family: GalanoGrotesque-Medium; font-size: 42px !important; color: #000000; } +body h2 { margin-top: 40px !important; font-size: 28px !important; font-family: GalanoGrotesque-Medium; } +body h2#table-of-contents { font-family: GalanoGrotesque-Medium; color: #ffffff; font-size: 14px !important; } +body h3 { margin-top: 25px !important; font-size: 18px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body h4 { margin-top: 25px !important; font-size: 16px; font-weight: 600; font-family: GalanoGrotesque-Medium; } +body .text-delta { text-transform: none; width: 100%; display: block; background-color: #279890; color: #ffffff; border-radius: 2px 2px 0 0; font-size: 14px !important; padding: 10px 15px 8px 15px; } +body .text-delta + ul { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body .text-delta + ul li { text-decoration: underline; } +body .text-delta + ul li::before { content: none; } +body .text-delta + ul li a:hover { background-image: none; } +body .text-delta + ul li a { white-space: normal; } +body #markdown-toc { background-color: #f5f7f9; width: 100%; border: none; margin-top: 0; border-radius: 0; position: relative; top: -3px; padding: 15px 25px 15px 25px; } +body #markdown-toc li::before { display: none; } +body #markdown-toc li a { color: #279890; font-size: 1em; background-image: none; white-space: normal; } +body #markdown-toc li a:hover { background-image: none; color: #279890; } +body #markdown-toc li ol, body #markdown-toc li ul { border: none; } +body #markdown-toc li ol li a, body #markdown-toc li ul li a { padding-bottom: 0; } +body div.highlighter-rouge pre.highlight { padding: 10px; } +body .highlight code { background-color: transparent; } +body .btn { padding: 6px 14px; color: #2f3433; font-family: GalanoGrotesque; border-radius: 4px; background: #ffffff; border: solid 2px #2f3433; box-shadow: none; } +body .btn:hover { border-color: #279890; color: #279890; background: #ffffff; } +body .btn-green { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; background-color: #279890; border: 2px solid #279890; } +body .btn-green:hover { background-color: #ffffff; color: #279890; } +body .main-content h1:hover a.anchor-heading, body .main-content h2:hover a.anchor-heading, body .main-content h3:hover a.anchor-heading, body .main-content h4:hover a.anchor-heading, body .main-content h5:hover a.anchor-heading, body .main-content h6:hover a.anchor-heading { left: -1.7rem; background-image: url("../../assets/icons/anchor.svg"); width: 24px; height: 24px; background-repeat: no-repeat; background-size: contain; background-position: center center; } +body .main-content h1:hover a.anchor-heading svg, body .main-content h2:hover a.anchor-heading svg, body .main-content h3:hover a.anchor-heading svg, body .main-content h4:hover a.anchor-heading svg, body .main-content h5:hover a.anchor-heading svg, body .main-content h6:hover a.anchor-heading svg { display: none; } +body .main-content h1:hover a.anchor-heading:hover, body .main-content h2:hover a.anchor-heading:hover, body .main-content h3:hover a.anchor-heading:hover, body .main-content h4:hover a.anchor-heading:hover, body .main-content h5:hover a.anchor-heading:hover, body .main-content h6:hover a.anchor-heading:hover { background-image: url("../../assets/icons/anchor green.svg"); } +body .main-content .toc-block { display: table-cell; } + +.main .main-header, .side-bar .site-header { background-color: white; height: 80px; max-height: 80px; min-height: 80px; border: 0; position: fixed; z-index: 2; } + +.main .main-header { width: 75%; } +.main .main-header nav.aux-nav { width: 100%; padding: 0; max-width: 1080px; } +.main .main-header nav.aux-nav .aux-nav-list { justify-content: flex-end; padding: 20px 80px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #000000; margin: 0 4px; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item a.site-button:hover { background-image: none; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item:hover a, .main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.active a { color: #279890; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button { font-size: 16px; line-height: 1.63; letter-spacing: 0.05px; color: #ffffff; margin: 0 0 0 20px; padding: 18px; border-radius: 6px; background-color: #f24886; border: 2px solid #f24886; } +.main .main-header nav.aux-nav .aux-nav-list .aux-nav-list-item.button a.site-button:hover { background-color: #b62d55; border: 2px solid #b62d55; color: #ffffff; } + +.side-bar:after { content: ' '; position: fixed; width: 100vw; top: 80px; box-shadow: 0 0 6px 2px rgba(0, 0, 0, 0.23); left: 0; right: 0; z-index: 0; } + +.side-bar .site-header { width: 25%; justify-content: flex-end; top: 0; } +.side-bar .site-header a.site-title { max-width: 290px; padding: 0; } +.side-bar .site-header a.site-title .site-logo { width: 150px; height: inherit; } +.side-bar .site-header a.site-title:hover { background-image: none; background-color: transparent; } + +.site-logo { width: 135px; height: 30px; } + +.side-bar { padding: 80px 0; border: 0; display: block; z-index: 1; } +.side-bar nav.site-nav, .side-bar nav.aux-nav { padding: 0; width: 100%; display: initial; } +.side-bar nav.site-nav .nav-list, .side-bar nav.site-nav .aux-nav-list, .side-bar nav.aux-nav .nav-list, .side-bar nav.aux-nav .aux-nav-list { width: 100%; padding: 0; margin-top: 30px; } +.side-bar nav.site-nav .nav-list .nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item { padding: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .nav-list .nav-list-item:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active:hover { background-color: transparent; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li { padding-right: 0; padding-left: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.active .nav_list li:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.active .nav_list li:hover { background-color: #ffffff; } +.side-bar nav.site-nav .nav-list .nav-list-item.header, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .nav-list .nav-list-item.header, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header { padding: 12px 40px 12px calc(100% - 290px); color: #279890; font-family: GalanoGrotesque-Bold; font-size: 18px !important; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:hover { background-color: transparent; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .nav-list .aux-nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .nav-list-item.header:not(:first-of-type), .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item.header:not(:first-of-type) { padding-top: 36px; border-top: 1px solid #dadada; margin-top: 20px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link { padding: 12px 40px 12px calc(100% - 290px); font-size: 14px !important; color: #576489; font-family: GalanoGrotesque; letter-spacing: 1.1px; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link:hover { background-color: #ffffff; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .nav-list .aux-nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .nav-list-item a.nav-list-link.active, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item a.nav-list-link.active { font-family: GalanoGrotesque-Medium !important; color: #279890 !important; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander { right: 40px; width: 22px; height: 22px; padding: 0; margin: 14px 0 0 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander svg, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander svg { color: #000000; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list-expander:hover, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list-expander:hover { background-color: none; background-image: none; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list { margin-top: 0; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list a.aux-nav-list-item { margin-left: 20px; font-size: 13px !important; letter-spacing: 0.05px; } +.side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.site-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .nav-list .aux-nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .nav-list a.aux-nav-list-item, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.nav-list-link, .side-bar nav.aux-nav .aux-nav-list .aux-nav-list-item .aux-nav-list .aux-nav-list a.aux-nav-list-item { margin-left: 30px; color: #949494; font-size: 12px !important; } +.side-bar nav.site-nav .nav-version, .side-bar nav.aux-nav .nav-version { display: flex; align-items: center; justify-content: center; padding: 12px 40px 12px calc(100% - 290px); text-transform: none; } +.side-bar nav.site-nav .nav-version select, .side-bar nav.aux-nav .nav-version select { height: 2em; margin: 0 0 0 10px; flex-grow: 1; } +.side-bar nav.site-nav .search, .side-bar nav.aux-nav .search { margin-top: 35px; margin-left: calc(100% - 290px); margin-right: 40px; max-width: 250px !important; max-height: 40px !important; width: 250px !important; z-index: 1; } +.side-bar nav.site-nav .search .search-input, .side-bar nav.aux-nav .search .search-input { background-color: #f5f7f9; border: 1px solid #dadada; border-radius: 6px; color: #6e7171; font-family: GalanoGrotesque-Medium; } +.side-bar nav.site-nav .search .search-input:focus, .side-bar nav.site-nav .search .search-input:active, .side-bar nav.aux-nav .search .search-input:focus, .side-bar nav.aux-nav .search .search-input:active { background-color: #f5f7f9; color: #6e7171; } +.side-bar nav.site-nav .search .search-input .search-icon, .side-bar nav.aux-nav .search .search-input .search-icon { color: #949494; } +.side-bar nav.site-nav .search .search-input:focus + .search-label .search-icon, .side-bar nav.aux-nav .search .search-input:focus + .search-label .search-icon { color: #949494; } +.side-bar nav.site-nav .mobile-menu, .side-bar nav.aux-nav .mobile-menu { display: none; } + +.label { text-transform: none !important; margin-left: 0 !important; margin-right: 0 !important; } + +.search-active .main { position: inherit; right: inherit; left: inherit; } + +.search-overlay { display: none; } + +.search-result-doc .search-result-icon { color: #279890; } + +footer { height: 360px; } +footer .footer-main { padding: 80px; } +footer .footer-main ul { list-style: none; padding: 0; margin: 0; } +footer .footer-main ul li { display: inline-block; padding: 0; margin: 0; } +footer .footer-main .row { display: flex; } +footer .footer-main .row .left { width: 70%; } +footer .footer-main .row .right { width: 30%; display: flex; justify-content: flex-end; } +footer .footer-main .row .site-title { padding: 0; margin-bottom: 30px; } +footer .footer-main .row .site-title .site-logo { height: 45px; } +footer .footer-main .row .footer-links li { margin-right: 40px; } +footer .footer-main .row .footer-links li a { padding: 0; color: #000000; font-size: 18px; } +footer .footer-main .row .footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a, footer .footer-main .row .bottom-footer-links li a { padding: 0 15px; font-size: 16px; color: #949494; } +footer .footer-main .row .footer-social li a:last-of-type, footer .footer-main .row .bottom-footer-links li a:last-of-type { padding-right: 0; } +footer .footer-main .row .footer-social li a:hover, footer .footer-main .row .bottom-footer-links li a:hover { color: #279890; } +footer .footer-main .row .footer-social li a img.hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.no-hover { display: none; } +footer .footer-main .row .footer-social li a:hover img.hover { display: block; } +footer .footer-main .top-border { padding-top: 30px; margin-top: 30px; border-top: 1px solid #dadada; } +footer .footer-main a:hover { background-color: transparent !important; background-image: none !important; } +footer .footer-sidebar { z-index: -1; } + +@media (max-width: 799px) { body { padding-bottom: 0; } + body .body-wrapper { display: block; } + body .body-wrapper .side-bar { width: 100%; max-width: 100%; padding: 0; } + body .body-wrapper .side-bar .site-header { width: 100%; flex-direction: row-reverse; justify-content: flex-end; } + body .body-wrapper .side-bar .site-header .site-button { padding: 11px 30px 0; width: 24px; height: 24px; background-repeat: no-repeat; background-size: auto; background-position: center center; background-image: url("../../assets/icons/menu.svg"); } + body .body-wrapper .side-bar .site-header .site-button:hover, body .body-wrapper .side-bar .site-header .site-button:active, body .body-wrapper .side-bar .site-header .site-button:focus { background-color: transparent; } + body .body-wrapper .side-bar .site-header .site-button.nav-open { background-image: url("../../assets/icons/close.svg"); } + body .body-wrapper .side-bar .site-header .site-button svg { display: none; } + body .body-wrapper .side-bar nav { display: none; overflow: scroll; background: white; } + body .body-wrapper .side-bar nav.nav-open { display: block; position: fixed; top: 0; height: 100vh; } + body .body-wrapper .side-bar nav.nav-open .search { position: relative; margin-top: 120px; margin-left: 30px; margin-right: 30px; padding: 0; } + body .body-wrapper .side-bar nav.nav-open .search .search-results { display: block !important; max-height: 60vh !important; } + body .body-wrapper .side-bar nav.nav-open .search .search-input { background-color: #ffffff; } + body .body-wrapper .side-bar nav.nav-open .search .search-input-wrap { box-shadow: none; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item { padding-left: 30px !important; padding-right: 30px !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a { padding-left: 0 !important; padding-right: 0 !important; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .nav-list .aux-nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list-item a.active, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .aux-nav-list-item a.active { background: unset; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list { margin-top: 0; } + body .body-wrapper .side-bar nav.nav-open .nav-list .nav-list a.nav-list-link, body .body-wrapper .side-bar nav.nav-open .aux-nav-list .nav-list a.nav-list-link { text-indent: 0; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu { display: block; border-top: 1px solid #dadada; margin-top: 18px; padding-bottom: 80px; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav { display: block; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list { flex-direction: column; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list a.site-button { font-size: 18px !important; color: #000000; } + body .body-wrapper .side-bar nav.nav-open .mobile-menu nav .aux-nav-list .button .site-button { border-radius: 6px; border: solid 2px #279890; background-color: #ffffff; color: #279890; padding: 10px 20px !important; display: inline-block; margin-top: 10px; } + body .body-wrapper .main { width: 100%; max-width: 100%; float: none; } + body .body-wrapper .main .main-header { display: none; } + body .body-wrapper .main .main-content a { white-space: normal; } + body .body-wrapper .main .main-content-wrap { padding: 80px 30px; } + body footer { border-top: 1px solid #dadada; height: auto; } + body footer .footer-sidebar { display: none; } + body footer .footer-main { width: 100%; max-width: 100%; padding: 60px 30px; float: none; } + body footer .footer-main .row { flex-direction: column; } + body footer .footer-main .row .right, body footer .footer-main .row .left { width: 100%; display: block; } + body footer .footer-main .row .site-title { margin-bottom: 40px; } + body footer .footer-main .row .footer-links { display: flex; flex-wrap: wrap; margin-bottom: 20px; } + body footer .footer-main .row .footer-links li { margin: 0 0 30px; width: 50%; } + body footer .footer-main .row .footer-social { width: 100%; display: flex; justify-content: space-between; } + body footer .footer-main .row .footer-social li a { padding: 0; } + body footer .footer-main .row .footer-social li a img { width: 34px; height: 34px; } + body footer .footer-main .row .bottom-footer-links li { display: block; margin-top: 30px; } + body footer .footer-main .row .bottom-footer-links li a { padding: 0; } + body footer .footer-main .row .top-border { padding-top: 40px; margin-top: 40px; } } +.copy-code-container { position: absolute; right: 0.3em; } + +.copy-code-button { align-items: center; justify-content: center; border: none; cursor: pointer; font-size: 1rem; color: #000000; padding: 0.4em 0.5em; background-color: transparent; outline: none; } +.copy-code-button:hover { color: #279890 !important; } + +.highlighter-rouge { position: relative; } + +.language-shell.highlighter-rouge pre.highlight::before, .language-sh.highlighter-rouge pre.highlight::before, .language-bash.highlighter-rouge pre.highlight::before { content: "$ "; color: #000000; font-weight: 400; font-family: monospace; font-size: 0.75em; } + +th, td { background-color: transparent; } + +.main-content ul.ui-tabs-nav > li::before { content: ""; } + +.tabs { background: transparent; } +.tabs.ui-widget .copy-code-button { font-family: "Font Awesome 6 Free" !important; } + +.tabs .ui-widget-header { background: transparent; border: none; border-bottom: 1px solid #d3d3d3; -moz-border-radius: 0px; -webkit-border-radius: 0px; border-radius: 0px; } + +.tabs .ui-tabs-nav .ui-state-default { background: transparent; border: none; } + +.tabs .ui-tabs-nav .ui-state-active { border: none; } + +.tabs .ui-tabs-nav .ui-state-default a { color: #c0c0c0; } + +.tabs .ui-tabs-nav .ui-state-active a { color: #279890; } + +/*overridden*/ +div.highlighter-rouge { border-radius: 4px; } +div.highlighter-rouge:hover > button { cursor: pointer; opacity: 0.5; } + +.highlighter-rouge button.copy-code-button { color: #000000; } + +/* Tooltip container */ +.tooltip { position: relative; display: inline-block; border-bottom: 0; } +.tooltip > a, .tooltip span { text-decoration: underline; text-decoration-style: dotted; background-image: none; } +.tooltip .tooltiptext { visibility: hidden; text-decoration: none; background-color: #2ce5b5; color: #000000; padding: 10px 20px; border-radius: 6px; position: absolute; z-index: 1; width: 400px; top: 100%; left: 50%; margin-left: -200px; } +.tooltip:hover .tooltiptext { visibility: visible; } +.tooltip .tooltiptext::after { content: " "; position: absolute; color: #2ce5b5; bottom: 100%; /* At the top of the tooltip */ left: 50%; margin-left: -5px; border-width: 5px; border-style: solid; border-color: transparent transparent #2ce5b5 transparent; } + +.ui-dialog-titlebar { display: none; } + +.feedback-container { position: sticky; top: 180px; z-index: 9999; } +@media (max-width: 1024px) { .feedback-container { display: none; } } +.feedback-container .feedback-buttons { position: absolute; right: 70px; margin-bottom: 20px; top: -55px; } +.feedback-container .feedback-buttons .tooltiptext { font-size: small; width: 150px; margin-left: -75px; text-align: center; } +.feedback-container .feedback-buttons .page-helpful-btn { background: none !important; border-radius: 50%; border: solid #279890 2px; color: #279890; height: 40px; width: 40px; margin: 0 3px; cursor: pointer; } +.feedback-container .feedback-buttons .page-helpful-btn:focus-visible { outline: none; } + +#is-helpful-ty { display: none; } +#is-helpful-ty a { color: #279890 !important; } + +.badge { padding: 2px 4px; background: #ffddec; text-transform: uppercase; text-align: center; border-radius: 6px; font-size: x-small; } + +.quickstart-steps { padding: 10px; } +.quickstart-steps img { width: 50px; height: 50px; } + +.row { display: flex; flex-direction: row; } + +.col { display: flex; flex-direction: column; justify-content: center; margin-bottom: 0.3em; } +.col.step-num { margin-right: 20px; } +.col.step-num > img { width: 40px; } + +img.quickstart { box-shadow: 3px 3px 1px #ccc; border-width: thin; border-color: black; border-style: ridge; } + +.breadcrumb-nav { margin-top: 1rem; } diff --git a/v1.46/assets/css/pygments/github.css b/v1.46/assets/css/pygments/github.css new file mode 100644 index 000000000..6e24c4d06 --- /dev/null +++ b/v1.46/assets/css/pygments/github.css @@ -0,0 +1,212 @@ +.highlighter-rouge .highlight table td { padding: 5px; } +.highlighter-rouge .highlight table pre { margin: 0; } +.highlighter-rouge .highlight .cm { + color: #999988; + font-style: italic; +} +.highlighter-rouge .highlight .cp { + color: #999999; + font-weight: bold; +} +.highlighter-rouge .highlight .c1 { + color: #999988; + font-style: italic; +} +.highlighter-rouge .highlight .cs { + color: #999999; + font-weight: bold; + font-style: italic; +} +.highlighter-rouge .highlight .c, .highlighter-rouge .highlight .ch, .highlighter-rouge .highlight .cd, .highlighter-rouge .highlight .cpf { + color: #999988; + font-style: italic; +} +.highlighter-rouge .highlight .err { + color: #a61717; + background-color: #e3d2d2; +} +.highlighter-rouge .highlight .gd { + color: #000000; + background-color: #ffdddd; +} +.highlighter-rouge .highlight .ge { + color: #000000; + font-style: italic; +} +.highlighter-rouge .highlight .gr { + color: #aa0000; +} +.highlighter-rouge .highlight .gh { + color: #999999; +} +.highlighter-rouge .highlight .gi { + color: #000000; + background-color: #ddffdd; +} +.highlighter-rouge .highlight .go { + color: #888888; +} +.highlighter-rouge .highlight .gp { + color: #555555; +} +.highlighter-rouge .highlight .gs { + font-weight: bold; +} +.highlighter-rouge .highlight .gu { + color: #aaaaaa; +} +.highlighter-rouge .highlight .gt { + color: #aa0000; +} +.highlighter-rouge .highlight .kc { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .kd { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .kn { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .kp { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .kr { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .kt { + color: #445588; + font-weight: bold; +} +.highlighter-rouge .highlight .k, .highlighter-rouge .highlight .kv { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .mf { + color: #009999; +} +.highlighter-rouge .highlight .mh { + color: #009999; +} +.highlighter-rouge .highlight .il { + color: #009999; +} +.highlighter-rouge .highlight .mi { + color: #009999; +} +.highlighter-rouge .highlight .mo { + color: #009999; +} +.highlighter-rouge .highlight .m, .highlighter-rouge .highlight .mb, .highlighter-rouge .highlight .mx { + color: #009999; +} +.highlighter-rouge .highlight .sb { + color: #d14; +} +.highlighter-rouge .highlight .sc { + color: #d14; +} +.highlighter-rouge .highlight .sd { + color: #d14; +} +.highlighter-rouge .highlight .s2 { + color: #d14; +} +.highlighter-rouge .highlight .se { + color: #d14; +} +.highlighter-rouge .highlight .sh { + color: #d14; +} +.highlighter-rouge .highlight .si { + color: #d14; +} +.highlighter-rouge .highlight .sx { + color: #d14; +} +.highlighter-rouge .highlight .sr { + color: #009926; +} +.highlighter-rouge .highlight .s1 { + color: #d14; +} +.highlighter-rouge .highlight .ss { + color: #990073; +} +.highlighter-rouge .highlight .s, .highlighter-rouge .highlight .sa, .highlighter-rouge .highlight .dl { + color: #d14; +} +.highlighter-rouge .highlight .na { + color: #008080; +} +.highlighter-rouge .highlight .n { + color: #032f62; +} +.highlighter-rouge .highlight .bp { + color: #999999; +} +.highlighter-rouge .highlight .nb { + color: #0086B3; +} +.highlighter-rouge .highlight .nc { + color: #445588; + font-weight: bold; +} +.highlighter-rouge .highlight .no { + color: #008080; +} +.highlighter-rouge .highlight .nd { + color: #3c5d5d; + font-weight: bold; +} +.highlighter-rouge .highlight .ni { + color: #800080; +} +.highlighter-rouge .highlight .ne { + color: #990000; + font-weight: bold; +} +.highlighter-rouge .highlight .nf, .highlighter-rouge .highlight .fm { + color: #990000; + font-weight: bold; +} +.highlighter-rouge .highlight .nl { + color: #990000; + font-weight: bold; +} +.highlighter-rouge .highlight .nn { + color: #555555; +} +.highlighter-rouge .highlight .nt { + color: #000080; +} +.highlighter-rouge .highlight .vc { + color: #008080; +} +.highlighter-rouge .highlight .vg { + color: #008080; +} +.highlighter-rouge .highlight .vi { + color: #008080; +} +.highlighter-rouge .highlight .nv, .highlighter-rouge .highlight .vm { + color: #008080; +} +.highlighter-rouge .highlight .ow { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .o { + color: #000000; + font-weight: bold; +} +.highlighter-rouge .highlight .w { + color: #bbbbbb; +} +.highlighter-rouge .highlight { + background-color: #f8f8f8; +} diff --git a/v1.46/assets/css/pygments/igorpro.css b/v1.46/assets/css/pygments/igorpro.css new file mode 100644 index 000000000..1a61746e9 --- /dev/null +++ b/v1.46/assets/css/pygments/igorpro.css @@ -0,0 +1,35 @@ +.highlight table td { padding: 5px; } +.highlight table pre { margin: 0; } +.highlight, .highlight .w { + color: #444444; +} +.highlight .cp { + color: #CC00A3; +} +.highlight .cs { + color: #CC00A3; +} +.highlight .c, .highlight .ch, .highlight .cd, .highlight .cm, .highlight .cpf, .highlight .c1 { + color: #FF0000; +} +.highlight .kc { + color: #C34E00; +} +.highlight .kd { + color: #0000FF; +} +.highlight .kr { + color: #007575; +} +.highlight .k, .highlight .kn, .highlight .kp, .highlight .kt, .highlight .kv { + color: #0000FF; +} +.highlight .s, .highlight .sb, .highlight .sc, .highlight .dl, .highlight .sd, .highlight .s2, .highlight .se, .highlight .sh, .highlight .si, .highlight .sx, .highlight .sr, .highlight .s1, .highlight .ss { + color: #009C00; +} +.highlight .sa { + color: #0000FF; +} +.highlight .nb, .highlight .bp { + color: #C34E00; +} diff --git a/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Bold.otf b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Bold.otf new file mode 100644 index 000000000..9e8e153c2 Binary files /dev/null and b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Bold.otf differ diff --git a/v1.46/assets/fonts/Rene Bieder - Galano Grotesque ExtraLight.otf b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque ExtraLight.otf new file mode 100644 index 000000000..b55c212b1 Binary files /dev/null and b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque ExtraLight.otf differ diff --git a/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Medium.otf b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Medium.otf new file mode 100644 index 000000000..be0f7785e Binary files /dev/null and b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque Medium.otf differ diff --git a/v1.46/assets/fonts/Rene Bieder - Galano Grotesque.otf b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque.otf new file mode 100644 index 000000000..eaa23bbe0 Binary files /dev/null and b/v1.46/assets/fonts/Rene Bieder - Galano Grotesque.otf differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.eot b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.eot new file mode 100644 index 000000000..0b1daeba8 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.eot differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.otf b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.otf new file mode 100644 index 000000000..9e8e153c2 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.otf differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.svg b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.svg new file mode 100644 index 000000000..96bdbf7ba --- /dev/null +++ b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.svg @@ -0,0 +1,3431 @@ + + + + +Created by FontForge 20190801 at Sun Dec 28 12:36:43 2014 + By root +Copyright (c) 2014 by Rene' Bieder. All rights reserved. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.ttf b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.ttf new file mode 100644 index 000000000..36de3c958 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.ttf differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff new file mode 100644 index 000000000..ae6bc589b Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff2 b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff2 new file mode 100644 index 000000000..531556103 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Bold.woff2 differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.eot b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.eot new file mode 100644 index 000000000..527b99c4e Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.eot differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.otf b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.otf new file mode 100644 index 000000000..be0f7785e Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.otf differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.svg b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.svg new file mode 100644 index 000000000..ccf28c15d --- /dev/null +++ b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.svg @@ -0,0 +1,3431 @@ + + + + +Created by FontForge 20190801 at Sun Dec 28 12:36:42 2014 + By root +Copyright (c) 2014 by Rene' Bieder. All rights reserved. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.ttf b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.ttf new file mode 100644 index 000000000..af048c8c9 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.ttf differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff new file mode 100644 index 000000000..84a3426a9 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff differ diff --git a/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff2 b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff2 new file mode 100644 index 000000000..c383567a8 Binary files /dev/null and b/v1.46/assets/fonts/Rene-Bieder-Galano-Grotesque-Medium.woff2 differ diff --git a/v1.46/assets/icons/anchor green.svg b/v1.46/assets/icons/anchor green.svg new file mode 100644 index 000000000..3e5bd0424 --- /dev/null +++ b/v1.46/assets/icons/anchor green.svg @@ -0,0 +1,3 @@ + + + diff --git a/v1.46/assets/icons/anchor.svg b/v1.46/assets/icons/anchor.svg new file mode 100644 index 000000000..b542e0954 --- /dev/null +++ b/v1.46/assets/icons/anchor.svg @@ -0,0 +1,3 @@ + + + diff --git a/v1.46/assets/icons/close.svg b/v1.46/assets/icons/close.svg new file mode 100644 index 000000000..5e6fee306 --- /dev/null +++ b/v1.46/assets/icons/close.svg @@ -0,0 +1,9 @@ + + + close menu@2x + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/github-hover.svg b/v1.46/assets/icons/github-hover.svg new file mode 100644 index 000000000..e5e7d7333 --- /dev/null +++ b/v1.46/assets/icons/github-hover.svg @@ -0,0 +1,11 @@ + + + icon / slack copy 2@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/github.png b/v1.46/assets/icons/github.png new file mode 100644 index 000000000..aab26142b Binary files /dev/null and b/v1.46/assets/icons/github.png differ diff --git a/v1.46/assets/icons/github.svg b/v1.46/assets/icons/github.svg new file mode 100644 index 000000000..11e642774 --- /dev/null +++ b/v1.46/assets/icons/github.svg @@ -0,0 +1,11 @@ + + + icon / slack copy 2@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/linkedin-hover.svg b/v1.46/assets/icons/linkedin-hover.svg new file mode 100644 index 000000000..4c63319b7 --- /dev/null +++ b/v1.46/assets/icons/linkedin-hover.svg @@ -0,0 +1,11 @@ + + + icon / slack copy 3@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/linkedin.png b/v1.46/assets/icons/linkedin.png new file mode 100644 index 000000000..f0e320908 Binary files /dev/null and b/v1.46/assets/icons/linkedin.png differ diff --git a/v1.46/assets/icons/linkedin.svg b/v1.46/assets/icons/linkedin.svg new file mode 100644 index 000000000..61c45f9ac --- /dev/null +++ b/v1.46/assets/icons/linkedin.svg @@ -0,0 +1,13 @@ + + + icon / slack copy 3@2x + + + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/mastodon-hover.svg b/v1.46/assets/icons/mastodon-hover.svg new file mode 100644 index 000000000..9cd766431 --- /dev/null +++ b/v1.46/assets/icons/mastodon-hover.svg @@ -0,0 +1,60 @@ + + + icon / slack copy 2@2x + + + + + + + + icon / slack copy 2@2x + + + + diff --git a/v1.46/assets/icons/mastodon.svg b/v1.46/assets/icons/mastodon.svg new file mode 100644 index 000000000..34945a399 --- /dev/null +++ b/v1.46/assets/icons/mastodon.svg @@ -0,0 +1,60 @@ + + + icon / slack copy 2@2x + + + + + + + + icon / slack copy 2@2x + + + + diff --git a/v1.46/assets/icons/menu.svg b/v1.46/assets/icons/menu.svg new file mode 100644 index 000000000..6640d05ed --- /dev/null +++ b/v1.46/assets/icons/menu.svg @@ -0,0 +1,11 @@ + + + mobile menu@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/search.png b/v1.46/assets/icons/search.png new file mode 100644 index 000000000..72d8bdc6e Binary files /dev/null and b/v1.46/assets/icons/search.png differ diff --git a/v1.46/assets/icons/search.svg b/v1.46/assets/icons/search.svg new file mode 100644 index 000000000..d148b0312 --- /dev/null +++ b/v1.46/assets/icons/search.svg @@ -0,0 +1,8 @@ + + + search on@2x + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/slack-hover.svg b/v1.46/assets/icons/slack-hover.svg new file mode 100644 index 000000000..6240311fd --- /dev/null +++ b/v1.46/assets/icons/slack-hover.svg @@ -0,0 +1,11 @@ + + + icon / slack@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/slack.png b/v1.46/assets/icons/slack.png new file mode 100644 index 000000000..6dfe2d61f Binary files /dev/null and b/v1.46/assets/icons/slack.png differ diff --git a/v1.46/assets/icons/slack.svg b/v1.46/assets/icons/slack.svg new file mode 100644 index 000000000..247941f65 --- /dev/null +++ b/v1.46/assets/icons/slack.svg @@ -0,0 +1,11 @@ + + + icon / slack@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/twitter-hover.svg b/v1.46/assets/icons/twitter-hover.svg new file mode 100644 index 000000000..844b11169 --- /dev/null +++ b/v1.46/assets/icons/twitter-hover.svg @@ -0,0 +1,11 @@ + + + icon / slack copy@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/twitter.png b/v1.46/assets/icons/twitter.png new file mode 100644 index 000000000..61c923970 Binary files /dev/null and b/v1.46/assets/icons/twitter.png differ diff --git a/v1.46/assets/icons/twitter.svg b/v1.46/assets/icons/twitter.svg new file mode 100644 index 000000000..935abeb0c --- /dev/null +++ b/v1.46/assets/icons/twitter.svg @@ -0,0 +1,11 @@ + + + icon / slack copy@2x + + + + + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/youtube-hover.svg b/v1.46/assets/icons/youtube-hover.svg new file mode 100644 index 000000000..79028a4b6 --- /dev/null +++ b/v1.46/assets/icons/youtube-hover.svg @@ -0,0 +1,7 @@ + + + Youtube-hover + + + + \ No newline at end of file diff --git a/v1.46/assets/icons/youtube.png b/v1.46/assets/icons/youtube.png new file mode 100644 index 000000000..edf4be698 Binary files /dev/null and b/v1.46/assets/icons/youtube.png differ diff --git a/v1.46/assets/icons/youtube.svg b/v1.46/assets/icons/youtube.svg new file mode 100644 index 000000000..6614ca8a0 --- /dev/null +++ b/v1.46/assets/icons/youtube.svg @@ -0,0 +1,7 @@ + + + Youtube + + + + \ No newline at end of file diff --git a/v1.46/assets/images/large-image.jpg b/v1.46/assets/images/large-image.jpg new file mode 100644 index 000000000..c007781c2 Binary files /dev/null and b/v1.46/assets/images/large-image.jpg differ diff --git a/v1.46/assets/images/small-image.jpg b/v1.46/assets/images/small-image.jpg new file mode 100644 index 000000000..5bf58a949 Binary files /dev/null and b/v1.46/assets/images/small-image.jpg differ diff --git a/v1.46/assets/img/404-illustration.png b/v1.46/assets/img/404-illustration.png new file mode 100644 index 000000000..3b99633cb Binary files /dev/null and b/v1.46/assets/img/404-illustration.png differ diff --git a/v1.46/assets/img/Databricks-arch.excalidraw b/v1.46/assets/img/Databricks-arch.excalidraw new file mode 100644 index 000000000..7938c6a85 --- /dev/null +++ b/v1.46/assets/img/Databricks-arch.excalidraw @@ -0,0 +1,2288 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 599, + "versionNonce": 421591746, + "index": "aV", + "isDeleted": false, + "id": "fKdWdkNM9XYc-7n1H_JyG", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 312.9164573811969, + "y": 3438.744149522377, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 364.0026135070274, + "height": 353.9706516152905, + "seed": 1911862587, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "94vLrbjNaZ-Is-8OTvFrb", + "type": "arrow" + }, + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow" + }, + { + "id": "yt9EYZ_KA4Mchy3BUUQct", + "type": "arrow" + }, + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow" + } + ], + "updated": 1725544570539, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 448, + "versionNonce": 1172078110, + "index": "aVG", + "isDeleted": false, + "id": "rc8yXewY6HEPU-FwSwn3-", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 883.7566840331633, + "y": 3421.342955144116, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 181.98388158898558, + "height": 93.00907374489543, + "seed": 1701992539, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "HCNSYE_W_xRA4UTqgTuuu", + "type": "arrow" + }, + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow" + } + ], + "updated": 1725544671737, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 188, + "versionNonce": 1377980475, + "index": "aWG", + "isDeleted": false, + "id": "C1-kKCZJAGayLq8AFHcP_", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1267.0388624342286, + "y": 3282.233183635602, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 179.22793579101562, + "height": 35, + "seed": 413344987, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722537743081, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "Object Store", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Object Store", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 524, + "versionNonce": 1295073589, + "index": "aWV", + "isDeleted": false, + "id": "9OomwBL5b71Vw2A8hnvQG", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1224.8463054463277, + "y": 3587.374912933678, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 165.03779590129852, + "height": 52.303226761469986, + "seed": 386188827, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722540974228, + "link": null, + "locked": false, + "fontSize": 20.921290704587996, + "fontFamily": 1, + "text": "lakeFS Managed\nBucket/s*", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakeFS Managed\nBucket/s*", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 564, + "versionNonce": 615590805, + "index": "aWd", + "isDeleted": false, + "id": "9amrjmm6vzKNvV1txfde1", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1236.8717429018543, + "y": 3529.01430153436, + "strokeColor": "#258c81", + "backgroundColor": "transparent", + "width": 61.157129574892735, + "height": 30.444620954311894, + "seed": 235752443, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow" + } + ], + "updated": 1722537938459, + "link": null, + "locked": false, + "fontSize": 12.177848381724752, + "fontFamily": 1, + "text": "Metadata\nFiles", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Metadata\nFiles", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 675, + "versionNonce": 904819125, + "index": "aWh", + "isDeleted": false, + "id": "ToMIIJgI6tA6N828WY6rq", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1318.8706249429092, + "y": 3530.2828956536414, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 32.62775346636772, + "height": 30.44462095431188, + "seed": 785567637, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538759880, + "link": null, + "locked": false, + "fontSize": 12.177848381724752, + "fontFamily": 1, + "text": "Data\nFiles", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Data\nFiles", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 503, + "versionNonce": 518870299, + "index": "aWl", + "isDeleted": false, + "id": "SSwnkbraAHrWf4k26Yw0H", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1250.5732655789477, + "y": 3827.328135003164, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 105.2199496626854, + "height": 50, + "seed": 1034420533, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Unmanaged\nBucket/s", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Unmanaged\nBucket/s", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 457, + "versionNonce": 1211580482, + "index": "ac", + "isDeleted": false, + "id": "XC4ZfmwenvusZshR_P-vg", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 923.775497770621, + "y": 3429.7902151724725, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 97.38623046777609, + "height": 76.28588053309127, + "seed": 111869403, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "P6npyxRsQMDzyRRtFJCAZ", + "type": "arrow" + } + ], + "updated": 1725544674087, + "link": null, + "locked": false, + "status": "saved", + "fileId": "57c766831721e9fb1cd3ac887423f3539a447a55", + "scale": [ + 1, + 1 + ] + }, + { + "type": "image", + "version": 242, + "versionNonce": 1515296795, + "index": "ad", + "isDeleted": false, + "id": "PLXjmnHvBPwOF5s_2M80N", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1195.0597551231497, + "y": 3273.658229707978, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 57.78143547920078, + "height": 57.78143547920078, + "seed": 1661221877, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722537740595, + "link": null, + "locked": false, + "status": "saved", + "fileId": "14425b5cfc24549457fe531a77b482c2336c0d1d", + "scale": [ + 1, + 1 + ] + }, + { + "type": "ellipse", + "version": 413, + "versionNonce": 973815451, + "index": "am", + "isDeleted": false, + "id": "UgXMbNZ2cISgBrh6ismcD", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1214.1092737197202, + "y": 3426.8532767163338, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 185.48977829962644, + "height": 39.201122455943676, + "seed": 1418848981, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 530, + "versionNonce": 106559931, + "index": "amG", + "isDeleted": false, + "id": "_hfeDSwLW7xKar-modPKg", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1237.1477084934322, + "y": 3716.930821703292, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.96422399257345, + "height": 27.043760905302562, + "seed": 1436675061, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 469, + "versionNonce": 776064149, + "index": "an", + "isDeleted": false, + "id": "cwJIg6NqVo2NMUoXhDnjW", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1400.4484126719815, + "y": 3445.719602817568, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 27.435039617214706, + "height": 131.1158512677214, + "seed": 219662389, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -27.435039617214706, + 131.1158512677214 + ] + ] + }, + { + "type": "line", + "version": 586, + "versionNonce": 1846086235, + "index": "anG", + "isDeleted": false, + "id": "2z_yZfH_SMtR9BrRtmG8E", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1365.6978827081718, + "y": 3729.9461738227797, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 18.926668558261248, + "height": 90.45316844089105, + "seed": 1052424667, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -18.926668558261248, + 90.45316844089105 + ] + ] + }, + { + "type": "line", + "version": 673, + "versionNonce": 687780667, + "index": "ao", + "isDeleted": false, + "id": "H_XtdgYlVr7iUFJpqCbWp", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1212.980854385111, + "y": 3446.065940141501, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 27.435039617214724, + "height": 131.11585126772144, + "seed": 1740306837, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 27.435039617214724, + 131.11585126772144 + ] + ] + }, + { + "type": "line", + "version": 790, + "versionNonce": 2060257019, + "index": "aoG", + "isDeleted": false, + "id": "CFRTG3kjWCUrlOVzJ7Vpo", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1236.3692434706413, + "y": 3730.185102281819, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 18.92666855826126, + "height": 90.45316844089108, + "seed": 717923669, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.92666855826126, + 90.45316844089108 + ] + ] + }, + { + "type": "line", + "version": 648, + "versionNonce": 1910862325, + "index": "ap", + "isDeleted": false, + "id": "akl7vry0vw3MofnYpEndn", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1241.0102979672567, + "y": 3576.4837884948356, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 132.9629216786985, + "height": 9.835979999683579, + "seed": 1571880693, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 64.33734309428377, + 9.835979999683579 + ], + [ + 132.9629216786985, + 1.3938745221653432 + ] + ] + }, + { + "type": "line", + "version": 767, + "versionNonce": 321417115, + "index": "apG", + "isDeleted": false, + "id": "dZYE85DV9lUebPHuRCjgl", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1255.70597474399, + "y": 3820.156737982185, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 91.72741079519767, + "height": 6.785568236718933, + "seed": 949798523, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 44.384538373350125, + 6.785568236718933 + ], + [ + 91.72741079519767, + 0.961595152072412 + ] + ] + }, + { + "type": "line", + "version": 1251, + "versionNonce": 517641179, + "index": "aq", + "isDeleted": false, + "id": "Fr7P_CVOlfsTYH98ZGC7W", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1401.550970267136, + "y": 3447.863697266036, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 204.20793772300001, + "height": 73.92223842015714, + "seed": 629102677, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.41304386281733, + -6.992816983523688 + ], + [ + -8.914834167083862, + -46.345262208693256 + ], + [ + -56.84501538608395, + -71.12383284278236 + ], + [ + -123.82021026334682, + -73.92223842015714 + ], + [ + -171.6030768976081, + -53.21233210229683 + ], + [ + -195.7948938601827, + -6.2553848968735215 + ], + [ + -188.10231038835357, + -0.4688874539394121 + ] + ] + }, + { + "type": "line", + "version": 1368, + "versionNonce": 692962363, + "index": "aqV", + "isDeleted": false, + "id": "nya0fBWtyDxdIUTsvDnGM", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1366.458506438301, + "y": 3731.4253247753554, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 140.87736005396957, + "height": 50.996890297369326, + "seed": 1096102581, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.803924287309746, + -4.824149379127804 + ], + [ + -6.150095421272407, + -31.972303641918348 + ], + [ + -39.21579047863243, + -49.066348348330976 + ], + [ + -85.42010921675873, + -50.996890297369326 + ], + [ + -118.38417605130402, + -36.709703611302615 + ], + [ + -135.07343576665983, + -4.315415552496223 + ], + [ + -129.76653700655336, + -0.32347237531487866 + ] + ] + }, + { + "type": "line", + "version": 688, + "versionNonce": 122856315, + "index": "au", + "isDeleted": false, + "id": "Vz387B1b2vmibhxvnqQSH", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1272.6330163883117, + "y": 3476.5593270274376, + "strokeColor": "#1e1e1e", + "backgroundColor": "#2ce5b4", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 1643133019, + "groupIds": [ + "KDDtU5RMPJKd3ckJPcy7u" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536755118, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 572, + "versionNonce": 633764891, + "index": "av", + "isDeleted": false, + "id": "LmVPBMi0EdUrxG5bIS9xR", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1281.2718300799863, + "y": 3490.260528414957, + "strokeColor": "#1e1e1e", + "backgroundColor": "#2ce5b4", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 94270715, + "groupIds": [ + "KDDtU5RMPJKd3ckJPcy7u" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536755118, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 797, + "versionNonce": 882216149, + "index": "aw", + "isDeleted": false, + "id": "ev_I2v8T0dl-snNInpQe2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1340.3672103751042, + "y": 3477.9419639990383, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 2016632891, + "groupIds": [ + "dRFmKGMiTbRgGwfpJBDTA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538773231, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 681, + "versionNonce": 854988539, + "index": "ax", + "isDeleted": false, + "id": "nURCUYMNMyP0_7-yyrBkJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1349.0060240667788, + "y": 3491.643165386558, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 1600001269, + "groupIds": [ + "dRFmKGMiTbRgGwfpJBDTA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538773231, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 885, + "versionNonce": 376624187, + "index": "axG", + "isDeleted": false, + "id": "XYysRAd24zX_Q2p2eGnra", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1331.0303939180938, + "y": 3759.3131073009517, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 998622907, + "groupIds": [ + "irGlI1gUH6Ec0F0oKCLIA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 769, + "versionNonce": 1577050357, + "index": "axV", + "isDeleted": false, + "id": "d48yd9wY4bhWwtyxNljv4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1339.6692076097684, + "y": 3773.0143086884714, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 2108380789, + "groupIds": [ + "irGlI1gUH6Ec0F0oKCLIA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 914, + "versionNonce": 1797428443, + "index": "axd", + "isDeleted": false, + "id": "O0Ik_-buSPvejH_LvCcMc", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1287.1445715076575, + "y": 3759.4733474492523, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 392454165, + "groupIds": [ + "hhRkWUwlJ4OviPrDNswcV" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 798, + "versionNonce": 1034025557, + "index": "axl", + "isDeleted": false, + "id": "1ITqAjtgxj2u9TJWayiE8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1295.783385199332, + "y": 3773.174548836772, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 1201739707, + "groupIds": [ + "hhRkWUwlJ4OviPrDNswcV" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "arrow", + "version": 293, + "versionNonce": 1478691422, + "index": "b00", + "isDeleted": false, + "id": "HCNSYE_W_xRA4UTqgTuuu", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1066.740565622149, + "y": 3460.648650738039, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 182.92968465593754, + "height": 40.87623321251385, + "seed": 1298328277, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "9JxM9dyoRp9meqG8llmiS" + } + ], + "updated": 1725544671737, + "link": null, + "locked": false, + "startBinding": { + "elementId": "rc8yXewY6HEPU-FwSwn3-", + "focus": -0.5176202515159919, + "gap": 1, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 64.87394947035204, + 24.382785695793245 + ], + [ + 182.92968465593754, + 40.87623321251385 + ] + ] + }, + { + "type": "text", + "version": 13, + "versionNonce": 145362939, + "index": "b00V", + "isDeleted": false, + "id": "9JxM9dyoRp9meqG8llmiS", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1091.4385315145344, + "y": 3475.0314364338324, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 80.35196715593338, + "height": 20, + "seed": 364298389, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538051132, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Metadata", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "HCNSYE_W_xRA4UTqgTuuu", + "originalText": "Metadata", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 776, + "versionNonce": 427765918, + "index": "b0G", + "isDeleted": false, + "id": "GyPIp862EuIWsednp_LJE", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 701.419009853068, + "y": 3463.5176299298537, + "strokeColor": "#1e1e1e", + "backgroundColor": "#ffffff", + "width": 179.96696509205026, + "height": 0.33654184094984885, + "seed": 52134741, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1725544688079, + "link": null, + "locked": false, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": -0.8561968624583398, + "gap": 24.49993896484375, + "fixedPoint": null + }, + "endBinding": { + "elementId": "rc8yXewY6HEPU-FwSwn3-", + "focus": 0.10371774472304966, + "gap": 2.3707090880450323, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 179.96696509205026, + -0.33654184094984885 + ] + ] + }, + { + "type": "arrow", + "version": 648, + "versionNonce": 2127617154, + "index": "b0H", + "isDeleted": false, + "id": "94vLrbjNaZ-Is-8OTvFrb", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 702.7601963765055, + "y": 3741.599849175025, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "width": 519.5107918836129, + "height": 0.589509264049866, + "seed": 382577883, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "K2M3nmYuNkeXvtPaSRB-X" + } + ], + "updated": 1725544587073, + "link": null, + "locked": false, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": 0.7116931220340239, + "gap": 25.84112548828125, + "fixedPoint": null + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 519.5107918836129, + -0.589509264049866 + ] + ] + }, + { + "type": "text", + "version": 7, + "versionNonce": 2025606677, + "index": "b0HG", + "isDeleted": false, + "id": "K2M3nmYuNkeXvtPaSRB-X", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 940.0115674424673, + "y": 3697.1162259565835, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "width": 42.895988285541534, + "height": 20, + "seed": 1105348949, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538767196, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Data", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "94vLrbjNaZ-Is-8OTvFrb", + "originalText": "Data", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 1107, + "versionNonce": 1844428866, + "index": "b0HV", + "isDeleted": false, + "id": "FoG-M0XjPajxjfIcaEGnO", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 703.8018833882243, + "y": 3558.5500363712927, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "width": 521.800621702034, + "height": 3.848623689989836, + "seed": 915786645, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "e1j3hXwZh4R5fHWNy3zFH" + } + ], + "updated": 1725544585386, + "link": null, + "locked": false, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": -0.31200300975951706, + "gap": 26.8828125, + "fixedPoint": null + }, + "endBinding": { + "elementId": "9amrjmm6vzKNvV1txfde1", + "focus": -0.6574472910135761, + "gap": 11.269237811595985, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 521.800621702034, + -3.848623689989836 + ] + ] + }, + { + "type": "text", + "version": 47, + "versionNonce": 942598901, + "index": "b0Hl", + "isDeleted": false, + "id": "e1j3hXwZh4R5fHWNy3zFH", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 944.103324640474, + "y": 3548.844679102245, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "width": 42.895988285541534, + "height": 20, + "seed": 696045493, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538756147, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Data", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "FoG-M0XjPajxjfIcaEGnO", + "originalText": "Data", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "image", + "version": 405, + "versionNonce": 287162754, + "index": "b0O", + "isDeleted": false, + "id": "WdkrYra56assCyd-YXdgZ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 374.5715355296081, + "y": 3367.835634753072, + "strokeColor": "transparent", + "backgroundColor": "#ffffff", + "width": 235.0192366810598, + "height": 62.67179644828261, + "seed": 66171035, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "yt9EYZ_KA4Mchy3BUUQct", + "type": "arrow" + } + ], + "updated": 1725544661135, + "link": null, + "locked": false, + "status": "saved", + "fileId": "052b11aa8b22d6eabf95831f5c83abbbdebc45ef", + "scale": [ + 1, + 1 + ] + }, + { + "type": "text", + "version": 57, + "versionNonce": 1717000277, + "index": "b0R", + "isDeleted": false, + "id": "9OvAOafyzEx2eTcf6a6sZ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1185.3806807731926, + "y": 3331.483415371421, + "strokeColor": "#1e1e1e", + "backgroundColor": "#ffffff", + "width": 275.1998291015625, + "height": 20, + "seed": 1762599413, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538651923, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "S3 / Azure Blob / GCS / MinIO ...", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "S3 / Azure Blob / GCS / MinIO ...", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 528, + "versionNonce": 1326136149, + "index": "b0S", + "isDeleted": false, + "id": "Fhakw2nF56ggm_i2A1Hlz", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 317.8442255491933, + "y": 3839.670578499913, + "strokeColor": "#1e1e1e", + "backgroundColor": "#a5d8ff", + "width": 595.9036865234375, + "height": 20, + "seed": 1798373467, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538964014, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "* All Data & Metadata is always stored within your Account and Network", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "* All Data & Metadata is always stored within your Account and Network", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "AW4wY3hWkhVIqmOykjdMl", + "type": "image", + "x": 706.5731906592259, + "y": 3282.5365783516645, + "width": 46, + "height": 47, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0X", + "roundness": null, + "seed": 404631682, + "version": 105, + "versionNonce": 582825474, + "isDeleted": false, + "boundElements": null, + "updated": 1725544513961, + "link": null, + "locked": false, + "status": "saved", + "fileId": "f8e4899a9159ac15f50bae4ac5c2dccd836011de", + "scale": [ + 1, + 1 + ] + }, + { + "type": "rectangle", + "version": 488, + "versionNonce": 1176997214, + "index": "b0Y", + "isDeleted": false, + "id": "z1W92KzkwojMzlqN6Plu8", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 701.8841063100456, + "y": 3261.78898422824, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 181.98388158898558, + "height": 93.00907374489543, + "seed": 38669854, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "P6npyxRsQMDzyRRtFJCAZ", + "type": "arrow" + }, + { + "id": "yt9EYZ_KA4Mchy3BUUQct", + "type": "arrow" + } + ], + "updated": 1725544559802, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 314, + "versionNonce": 1116347714, + "index": "b0Z", + "isDeleted": false, + "id": "13cE0-qFZjKqduZujcHyv", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 773.2452792822728, + "y": 3282.6468231026415, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 77.21994018554688, + "height": 50, + "seed": 542653954, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725544513961, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Unity\nCatalog", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Unity\nCatalog", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 711, + "versionNonce": 2117289310, + "index": "b0a", + "isDeleted": false, + "id": "P6npyxRsQMDzyRRtFJCAZ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 885.2679146968808, + "y": 3305.3532800529015, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 81.15973111234541, + "height": 114.3790722231729, + "seed": 1458601886, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "j8Zt4Ulc-RcPxroAJla6n" + } + ], + "updated": 1725544677019, + "link": null, + "locked": false, + "startBinding": { + "elementId": "z1W92KzkwojMzlqN6Plu8", + "focus": -0.6640351786153109, + "gap": 1.3999267978495595, + "fixedPoint": null + }, + "endBinding": { + "elementId": "XC4ZfmwenvusZshR_P-vg", + "focus": 0.1903316542076412, + "gap": 10.05786289639832, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 57.02657441915176, + 49.83874970645229 + ], + [ + 81.15973111234541, + 114.3790722231729 + ] + ] + }, + { + "type": "text", + "version": 17, + "versionNonce": 601030786, + "index": "b0b", + "isDeleted": false, + "id": "j8Zt4Ulc-RcPxroAJla6n", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 994.3189449911908, + "y": 3515.248472020096, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 80.35196715593338, + "height": 20, + "seed": 2050677726, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725544490151, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Metadata", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "P6npyxRsQMDzyRRtFJCAZ", + "originalText": "Metadata", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 1223, + "versionNonce": 2005420482, + "index": "b0c", + "isDeleted": false, + "id": "yt9EYZ_KA4Mchy3BUUQct", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 724.0218806174502, + "y": 3362.2912147005477, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 99.7599142214134, + "height": 67.68391677137743, + "seed": 1260352862, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "b3_R9mIkILcj8av3_nA5y" + } + ], + "updated": 1725544661135, + "link": null, + "locked": false, + "startBinding": { + "elementId": "z1W92KzkwojMzlqN6Plu8", + "focus": -0.05957350657654702, + "gap": 7.4931567274120425, + "fixedPoint": null + }, + "endBinding": { + "elementId": "WdkrYra56assCyd-YXdgZ", + "focus": 1.084245357795154, + "gap": 14.671194185368904, + "fixedPoint": null + }, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + -54.73563261209824, + 37.751072704499165 + ], + [ + -99.7599142214134, + 67.68391677137743 + ] + ] + }, + { + "type": "text", + "version": 21, + "versionNonce": 1149890242, + "index": "b0d", + "isDeleted": false, + "id": "b3_R9mIkILcj8av3_nA5y", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 909.800388939104, + "y": 3430.3460898952812, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "width": 80.35196715593338, + "height": 20, + "seed": 8722846, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725544527659, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Metadata", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "yt9EYZ_KA4Mchy3BUUQct", + "originalText": "Metadata", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 645, + "versionNonce": 529689694, + "index": "b0g", + "isDeleted": false, + "id": "xTeR0_bCh8KU0wcP-0srj", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 514.0731808416692, + "y": 3477.921130353618, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 125.70667818470588, + "height": 56.12847900390625, + "seed": 5315998, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725545244443, + "link": null, + "locked": false, + "fontSize": 22.4513916015625, + "fontFamily": 1, + "text": "All-Purpose\nCompute", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "All-Purpose Compute", + "autoResize": false, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 595, + "versionNonce": 494503362, + "index": "b0h", + "isDeleted": false, + "id": "3S9sBolfAkDLZjOd2Fe9H", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 339.6697297312135, + "y": 3565.9532501045946, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 125.70667818470588, + "height": 28.064239501953125, + "seed": 320580994, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725545208914, + "link": null, + "locked": false, + "fontSize": 22.4513916015625, + "fontFamily": 1, + "text": "Jobs", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Jobs", + "autoResize": false, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 586, + "versionNonce": 406510914, + "index": "b0i", + "isDeleted": false, + "id": "iqXJML6s2q5suJoS4jKvq", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 339.6697297312135, + "y": 3690.836123639751, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 125.70667818470588, + "height": 56.12847900390625, + "seed": 992415582, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725545145468, + "link": null, + "locked": false, + "fontSize": 22.4513916015625, + "fontFamily": 1, + "text": "SQL \nWarehouse", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "SQL Warehouse", + "autoResize": false, + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 794, + "versionNonce": 1501638814, + "index": "b0j", + "isDeleted": false, + "id": "y9hEhmdS3EwfPhiLFa3li", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 514.0731808416692, + "y": 3630.8682433907275, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 125.70667818470588, + "height": 28.064239501953125, + "seed": 147045314, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725545244444, + "link": null, + "locked": false, + "fontSize": 22.4513916015625, + "fontFamily": 1, + "text": "Notebooks", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Notebooks", + "autoResize": false, + "lineHeight": 1.25 + }, + { + "id": "rUqXwJHrDEWBy717ee_5r", + "type": "image", + "x": 341.6351816862896, + "y": 3616.324382001567, + "width": 121.77577427455374, + "height": 63.93228149414071, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0k", + "roundness": null, + "seed": 161296322, + "version": 238, + "versionNonce": 1041493726, + "isDeleted": false, + "boundElements": null, + "updated": 1725545232109, + "link": null, + "locked": false, + "status": "saved", + "fileId": "39f6aaecb1c0b0b2b31e3bed9bf23bd8bfe551f5", + "scale": [ + 1, + 1 + ] + }, + { + "type": "image", + "version": 411, + "versionNonce": 697134302, + "index": "b0l", + "isDeleted": false, + "id": "_oBaI02DYBrY_e6-i5-bH", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 516.0386327967452, + "y": 3542.324382001567, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 121.77577427455374, + "height": 63.93228149414071, + "seed": 160702018, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1725545244444, + "link": null, + "locked": false, + "status": "saved", + "fileId": "39f6aaecb1c0b0b2b31e3bed9bf23bd8bfe551f5", + "scale": [ + 1, + 1 + ] + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "57c766831721e9fb1cd3ac887423f3539a447a55": { + "mimeType": "image/png", + "id": "57c766831721e9fb1cd3ac887423f3539a447a55", + "dataURL": "", + "created": 1722535141187, + "lastRetrieved": 1722535141187 + }, + "14425b5cfc24549457fe531a77b482c2336c0d1d": { + "mimeType": "image/png", + "id": "14425b5cfc24549457fe531a77b482c2336c0d1d", + "dataURL": "", + "created": 1722535283692, + "lastRetrieved": 1722535283692 + }, + "052b11aa8b22d6eabf95831f5c83abbbdebc45ef": { + "mimeType": "image/png", + "id": "052b11aa8b22d6eabf95831f5c83abbbdebc45ef", + "dataURL": "", + "created": 1722538330971, + "lastRetrieved": 1722538330971 + }, + "f8e4899a9159ac15f50bae4ac5c2dccd836011de": { + "mimeType": "image/png", + "id": "f8e4899a9159ac15f50bae4ac5c2dccd836011de", + "dataURL": "", + "created": 1725544409286, + "lastRetrieved": 1725544409286 + }, + "39f6aaecb1c0b0b2b31e3bed9bf23bd8bfe551f5": { + "mimeType": "image/png", + "id": "39f6aaecb1c0b0b2b31e3bed9bf23bd8bfe551f5", + "dataURL": "", + "created": 1725545071196, + "lastRetrieved": 1725545071196 + } + } +} \ No newline at end of file diff --git a/v1.46/assets/img/UI-Import-Dialog.png b/v1.46/assets/img/UI-Import-Dialog.png new file mode 100644 index 000000000..d7ea7e8df Binary files /dev/null and b/v1.46/assets/img/UI-Import-Dialog.png differ diff --git a/v1.46/assets/img/add_branch_protection_rule.png b/v1.46/assets/img/add_branch_protection_rule.png new file mode 100644 index 000000000..69c07f757 Binary files /dev/null and b/v1.46/assets/img/add_branch_protection_rule.png differ diff --git a/v1.46/assets/img/airbyte.png b/v1.46/assets/img/airbyte.png new file mode 100644 index 000000000..8207a2228 Binary files /dev/null and b/v1.46/assets/img/airbyte.png differ diff --git a/v1.46/assets/img/architecture.excalidraw b/v1.46/assets/img/architecture.excalidraw new file mode 100644 index 000000000..a51dbc395 --- /dev/null +++ b/v1.46/assets/img/architecture.excalidraw @@ -0,0 +1,3546 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 1242, + "versionNonce": 25930805, + "isDeleted": false, + "id": "_aEm-L6P94yZvYLPiUQdu", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 577.625, + "y": 224.6171875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 566.3671875, + "height": 441.734375, + "seed": 883304279, + "groupIds": [], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354334763, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 502, + "versionNonce": 1695398075, + "isDeleted": false, + "id": "TgGwOOzgVI1in0sDAxa-r", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 585.609375, + "y": 194.30078125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 135, + "height": 25, + "seed": 1816928153, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354175302, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "lakeFS server", + "baseline": 18, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakeFS server" + }, + { + "type": "rectangle", + "version": 719, + "versionNonce": 1032096629, + "isDeleted": false, + "id": "9B7vsP3VbZUcmy_h0fyUT", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 631.6484375, + "y": 370.693359375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 1114015833, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "nJIgCOrDzHKFPTaxD6jNH", + "type": "text" + }, + { + "id": "nJIgCOrDzHKFPTaxD6jNH", + "type": "text" + }, + { + "type": "text", + "id": "nJIgCOrDzHKFPTaxD6jNH" + }, + { + "id": "f4fx7Okaga8CbENsmpkmP", + "type": "arrow" + }, + { + "id": "Bv7_JtCztVbjCF9lOPXMj", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 957, + "versionNonce": 1212742971, + "isDeleted": false, + "id": "Bxgkr0odgFyw9uf9SBh9m", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 630.6875, + "y": 247.59375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 102, + "height": 66, + "seed": 379535735, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "gYy1ZNZFlRjyv6QM22Mwh", + "type": "text" + }, + { + "id": "gYy1ZNZFlRjyv6QM22Mwh", + "type": "text" + }, + { + "id": "gYy1ZNZFlRjyv6QM22Mwh", + "type": "text" + }, + { + "type": "text", + "id": "gYy1ZNZFlRjyv6QM22Mwh" + }, + { + "id": "qlZi2lbDur0dqp82LYO_L", + "type": "arrow" + } + ], + "updated": 1665354284481, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1437, + "versionNonce": 2106152757, + "isDeleted": false, + "id": "2ZfD2SrM-46E8aH7dhW-L", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 294.03125, + "y": 378.376953125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 90, + "seed": 1564026839, + "groupIds": [], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "gFQ7_SRIz8GTTeavtNxc1", + "type": "text" + }, + { + "id": "gFQ7_SRIz8GTTeavtNxc1", + "type": "text" + }, + { + "id": "gFQ7_SRIz8GTTeavtNxc1", + "type": "text" + }, + { + "id": "gFQ7_SRIz8GTTeavtNxc1", + "type": "text" + }, + { + "id": "gFQ7_SRIz8GTTeavtNxc1", + "type": "text" + }, + { + "id": "qlZi2lbDur0dqp82LYO_L", + "type": "arrow" + }, + { + "type": "text", + "id": "gFQ7_SRIz8GTTeavtNxc1" + }, + { + "id": "f4fx7Okaga8CbENsmpkmP", + "type": "arrow" + } + ], + "updated": 1665354301139, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1734, + "versionNonce": 1185922709, + "isDeleted": false, + "id": "LFEQGiwWugw-bHxiMcZeO", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 292.69140625, + "y": 538.07421875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 110, + "seed": 513927961, + "groupIds": [], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "qlZi2lbDur0dqp82LYO_L", + "type": "arrow" + }, + { + "type": "text", + "id": "6tgg9ZzYCBwT1BVPy4fTj" + }, + { + "id": "AB97GotmGG213cLmZEUpn", + "type": "arrow" + } + ], + "updated": 1665354315695, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1406, + "versionNonce": 42022811, + "isDeleted": false, + "id": "FcI_PQ6EPBwrqiPhB5Ph8", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 925.701171875, + "y": 241.80859375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 329647831, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "ePEr9LdcAMy2hfkyeKgHd", + "type": "text" + }, + { + "id": "ePEr9LdcAMy2hfkyeKgHd", + "type": "text" + }, + { + "id": "ePEr9LdcAMy2hfkyeKgHd", + "type": "text" + }, + { + "id": "ePEr9LdcAMy2hfkyeKgHd", + "type": "text" + }, + { + "type": "text", + "id": "ePEr9LdcAMy2hfkyeKgHd" + }, + { + "id": "KcyC9x5YsNNHh13B4n1nx", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1052, + "versionNonce": 1432963765, + "isDeleted": false, + "id": "5ER2G9M0ztifcYbrsZIow", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 925.7011718750001, + "y": 346.91145833333337, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 726083447, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "xZMYT3ni3ok2L7lxP6yGX", + "type": "text" + }, + { + "id": "xZMYT3ni3ok2L7lxP6yGX", + "type": "text" + }, + { + "id": "xZMYT3ni3ok2L7lxP6yGX", + "type": "text" + }, + { + "id": "xZMYT3ni3ok2L7lxP6yGX", + "type": "text" + }, + { + "id": "xZMYT3ni3ok2L7lxP6yGX", + "type": "text" + }, + { + "type": "text", + "id": "xZMYT3ni3ok2L7lxP6yGX" + }, + { + "id": "_WCe-x56UysqDesPMcovo", + "type": "arrow" + } + ], + "updated": 1665354187260, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1189, + "versionNonce": 1112378427, + "isDeleted": false, + "id": "svs0ZIIoEf9geQINc8L5R", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 925.7011718750001, + "y": 452.0143229166667, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 1969047831, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "type": "text" + }, + { + "type": "text", + "id": "lEnBUWHkXGiRrMqpAQ7DR" + }, + { + "id": "wECT1xqiaxw7G2ECloOnC", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1273, + "versionNonce": 2085398773, + "isDeleted": false, + "id": "APPf-R5uuQPPULrXAaGQl", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 925.7011718750001, + "y": 557.1171875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 1676121463, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "id": "iKQzPTb5mtu0ByF_0GqRF", + "type": "text" + }, + { + "type": "text", + "id": "iKQzPTb5mtu0ByF_0GqRF" + }, + { + "id": "S4SJSvhxNOOGmTdSWiyvI", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 472, + "versionNonce": 988171483, + "isDeleted": false, + "id": "_philqtA2msBgS2RN0gJq", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 631.6484375, + "y": 520.80078125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 93, + "seed": 1462723511, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "WlOxNA-p-slsjxP8fsP_x", + "type": "text" + }, + { + "id": "WlOxNA-p-slsjxP8fsP_x", + "type": "text" + }, + { + "id": "WlOxNA-p-slsjxP8fsP_x", + "type": "text" + }, + { + "type": "text", + "id": "WlOxNA-p-slsjxP8fsP_x" + }, + { + "id": "AB97GotmGG213cLmZEUpn", + "type": "arrow" + }, + { + "id": "tqXh-I1GblNxzaOjAhp0X", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 642, + "versionNonce": 118866517, + "isDeleted": false, + "id": "nJIgCOrDzHKFPTaxD6jNH", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 696.6484375, + "y": 404.693359375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 39, + "height": 25, + "seed": 1667728823, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "API", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "9B7vsP3VbZUcmy_h0fyUT", + "originalText": "API" + }, + { + "type": "text", + "version": 891, + "versionNonce": 40550549, + "isDeleted": false, + "id": "gYy1ZNZFlRjyv6QM22Mwh", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 645.1875, + "y": 268.09375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 73, + "height": 25, + "seed": 386463097, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278569, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Web UI", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "Bxgkr0odgFyw9uf9SBh9m", + "originalText": "Web UI" + }, + { + "type": "text", + "version": 1480, + "versionNonce": 2076353179, + "isDeleted": false, + "id": "gFQ7_SRIz8GTTeavtNxc1", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 318.03125, + "y": 383.376953125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 121, + "height": 80, + "seed": 204416281, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354301139, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Web UI\nlakectl\nSpark/lakeFSFS\nClient SDKs", + "baseline": 74, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "2ZfD2SrM-46E8aH7dhW-L", + "originalText": "Web UI\nlakectl\nSpark/lakeFSFS\nClient SDKs" + }, + { + "type": "text", + "version": 1364, + "versionNonce": 1171190037, + "isDeleted": false, + "id": "ePEr9LdcAMy2hfkyeKgHd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 939.2011718749999, + "y": 263.30859375, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 142, + "height": 50, + "seed": 1724102169, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Authentication\nAuthorization", + "baseline": 43, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "FcI_PQ6EPBwrqiPhB5Ph8", + "originalText": "Authentication\nAuthorization" + }, + { + "type": "text", + "version": 986, + "versionNonce": 2007242427, + "isDeleted": false, + "id": "xZMYT3ni3ok2L7lxP6yGX", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 967.201171875, + "y": 380.91145833333337, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 86, + "height": 25, + "seed": 1840934777, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Graveler", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "5ER2G9M0ztifcYbrsZIow", + "originalText": "Graveler" + }, + { + "type": "text", + "version": 1130, + "versionNonce": 1431963253, + "isDeleted": false, + "id": "lEnBUWHkXGiRrMqpAQ7DR", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 966.201171875, + "y": 473.5143229166667, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 88, + "height": 50, + "seed": 281589721, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Storage \nAdapter", + "baseline": 43, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "svs0ZIIoEf9geQINc8L5R", + "originalText": "Storage Adapter" + }, + { + "type": "text", + "version": 1204, + "versionNonce": 1242007387, + "isDeleted": false, + "id": "iKQzPTb5mtu0ByF_0GqRF", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 982.201171875, + "y": 591.1171875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 56, + "height": 25, + "seed": 1612702073, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Hooks", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "APPf-R5uuQPPULrXAaGQl", + "originalText": "Hooks" + }, + { + "type": "text", + "version": 394, + "versionNonce": 1637576661, + "isDeleted": false, + "id": "WlOxNA-p-slsjxP8fsP_x", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 657.1484375, + "y": 554.80078125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 118, + "height": 25, + "seed": 11736889, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "S3 gateway", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "_philqtA2msBgS2RN0gJq", + "originalText": "S3 gateway" + }, + { + "type": "rectangle", + "version": 1587, + "versionNonce": 1349194491, + "isDeleted": false, + "id": "S6p1P0dhiH25w1GCOnTPp", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1274.0637531815069, + "y": 504.4251538793582, + "strokeColor": "#000000", + "backgroundColor": "#40c05788", + "width": 65.08084106445301, + "height": 65.08084106445301, + "seed": 1208316473, + "groupIds": [ + "p8XMXpqXhpKUaBRdzQEwH", + "0GDE3jfCf0MWKFg5Aq0E4", + "0wcYe7v3VXAclSCBbcLcS", + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "S6p1P0dhiH25w1GCOnTPp", + "type": "arrow" + }, + { + "id": "wECT1xqiaxw7G2ECloOnC", + "type": "arrow" + } + ], + "updated": 1665354780006, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 1371, + "versionNonce": 745430933, + "isDeleted": false, + "id": "tdjFsnYUHbm7qID9T53R7", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1287.6766083591658, + "y": 515.4922499048741, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 39.10563151041674, + "height": 10.753995455228363, + "seed": 988588503, + "groupIds": [ + "0wcYe7v3VXAclSCBbcLcS", + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 1989, + "versionNonce": 1608044603, + "isDeleted": false, + "id": "i5RrCViQzPUN9hFfmBwDj", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1288.0345247754321, + "y": 521.3707179537521, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 38.295773237179446, + "height": 40.48662140675077, + "seed": 921486105, + "groupIds": [ + "0wcYe7v3VXAclSCBbcLcS", + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.8452774439103905, + 32.14685684595355 + ], + [ + 6.768312737880933, + 37.98883103590748 + ], + [ + 20.34867663261207, + 40.48662140675077 + ], + [ + 32.537268066406114, + 38.400380608974274 + ], + [ + 35.36686823918285, + 31.944173177083314 + ], + [ + 38.295773237179446, + 0.09767190004004078 + ] + ] + }, + { + "type": "ellipse", + "version": 1381, + "versionNonce": 1124574453, + "isDeleted": false, + "id": "n4YLocvG2ZKKd-4yTwDUV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1304.8048128613698, + "y": 531.6332171321251, + "strokeColor": "#000000", + "backgroundColor": "#000", + "width": 4.69818115234375, + "height": 4.69818115234375, + "seed": 226973431, + "groupIds": [ + "0wcYe7v3VXAclSCBbcLcS", + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 2067, + "versionNonce": 241766619, + "isDeleted": false, + "id": "CDm-90FSnkFU14GpRbSiW", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1307.2949312695728, + "y": 534.2062075496056, + "strokeColor": "#000000", + "backgroundColor": "#000", + "width": 23.24691772460949, + "height": 11.53113708496096, + "seed": 1920399353, + "groupIds": [ + "0wcYe7v3VXAclSCBbcLcS", + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 7.0229309082030795, + 7.4353179931640625 + ], + [ + 17.00493774414076, + 11.53113708496096 + ], + [ + 22.923687744140693, + 10.143841552734386 + ], + [ + 23.24691772460949, + 5.890612792968767 + ], + [ + 17.962628173828193, + 1.7561248779296932 + ] + ] + }, + { + "type": "text", + "version": 1900, + "versionNonce": 890322517, + "isDeleted": false, + "id": "gvnBdXocomaSYbJ15dp6t", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1222.1041737137334, + "y": 573.5953984820812, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 169, + "height": 75, + "seed": 1318019095, + "groupIds": [ + "9p5ZZoHV7wS9XjrQwglVX" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Object Store\nBucket\n(S3, GCS, Azure)", + "baseline": 68, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Object Store\nBucket\n(S3, GCS, Azure)" + }, + { + "type": "text", + "version": 2360, + "versionNonce": 1315454331, + "isDeleted": false, + "id": "Dstrr1oxpMuw-m4JGSQNd", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1182.6666737137334, + "y": 379.8079656063978, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 248, + "height": 75, + "seed": 4475417, + "groupIds": [ + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Key Value\nStore\n(PostgreSQL, DynamoDB)", + "baseline": 68, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Key Value\nStore\n(PostgreSQL, DynamoDB)" + }, + { + "type": "rectangle", + "version": 2053, + "versionNonce": 1499507637, + "isDeleted": false, + "id": "QrK4-de5GsjEjuLVX52yz", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1274.1262531815069, + "y": 308.92678350367504, + "strokeColor": "#000000", + "backgroundColor": "#4c6ef588", + "width": 65.08084106445301, + "height": 65.08084106445301, + "seed": 1284489207, + "groupIds": [ + "ViKZFsTx2amo6qdF5pqIB", + "KLrkluwnW3w9OX_UyMG76", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "QrK4-de5GsjEjuLVX52yz", + "type": "arrow" + }, + { + "id": "KcyC9x5YsNNHh13B4n1nx", + "type": "arrow" + } + ], + "updated": 1665354780006, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 2902, + "versionNonce": 594369211, + "isDeleted": false, + "id": "-EDC7qSOsy8GhwdGINNIA", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 6.070517456686609, + "x": 1324.9076059293707, + "y": 358.25696050529285, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 4.515147036494501, + "height": 6.809419856658472, + "seed": 1430626041, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 4.515147036494501, + 6.809419856658472 + ] + ] + }, + { + "type": "line", + "version": 2795, + "versionNonce": 414646901, + "isDeleted": false, + "id": "Vgf2G2dX5nG0xUedD_BMI", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0.029928014204294584, + "x": 1328.4106760236623, + "y": 316.08795953786733, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 4.878221980723982, + "height": 7.187766181342914, + "seed": 1117956375, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + -4.878221980723982, + 7.187766181342914 + ] + ] + }, + { + "type": "line", + "version": 3402, + "versionNonce": 872958811, + "isDeleted": false, + "id": "v9QeKKJVHsGUaok6amhuc", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 5.765434113477622, + "x": 1332.629635512782, + "y": 361.39783418032465, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 7.027484543816502, + "height": 5.025822775075784, + "seed": 755587033, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -2.6558630674341366, + 5.025822775075784 + ], + [ + -7.027484543816502, + 2.0346398876757292 + ] + ] + }, + { + "type": "line", + "version": 3619, + "versionNonce": 670624725, + "isDeleted": false, + "id": "Y6DGEH9GO3DBRnS6V3Bmu", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 2.48666424209895, + "x": 1285.537585021598, + "y": 315.4442679408971, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 6.480608834450265, + "height": 4.428740972665685, + "seed": 79392311, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -3.0517914343207093, + 4.428740972665685 + ], + [ + -6.4806088344502655, + 0.4183784164032155 + ] + ] + }, + { + "type": "line", + "version": 2996, + "versionNonce": 557590523, + "isDeleted": false, + "id": "aQCrDAYHZkiHVAz3R-cLD", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 6.070517456686609, + "x": 1282.3188097297736, + "y": 316.7779651160744, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 4.515147036494501, + "height": 6.809419856658472, + "seed": 1235416249, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 4.515147036494501, + 6.809419856658472 + ] + ] + }, + { + "type": "line", + "version": 2859, + "versionNonce": 1039575349, + "isDeleted": false, + "id": "jI92EQqdfBMt7di-mknre", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0.06885648930104615, + "x": 1287.3425001592234, + "y": 358.0083078048653, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 4.878221980723982, + "height": 7.187766181342914, + "seed": 1304334167, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + -4.878221980723982, + 7.187766181342914 + ] + ] + }, + { + "type": "line", + "version": 3628, + "versionNonce": 1091268763, + "isDeleted": false, + "id": "1Rmd-DXZ6u1CzIxIxpIX-", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 3.854199750177644, + "x": 1331.6476757175305, + "y": 314.65376079179435, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 6.480608834450265, + "height": 4.428740972665685, + "seed": 1588983193, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -3.0517914343207093, + 4.428740972665685 + ], + [ + -6.4806088344502655, + 0.4183784164032155 + ] + ] + }, + { + "type": "line", + "version": 3622, + "versionNonce": 1008056981, + "isDeleted": false, + "id": "grDakEOTelrLa7WPFOrwt", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0.663921886409196, + "x": 1285.56544494387, + "y": 362.2619734878205, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 6.480608834450265, + "height": 4.428740972665685, + "seed": 134253687, + "groupIds": [ + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -3.0517914343207093, + 4.428740972665685 + ], + [ + -6.4806088344502655, + 0.4183784164032155 + ] + ] + }, + { + "type": "line", + "version": 4551, + "versionNonce": 86549819, + "isDeleted": false, + "id": "Yu5RqzMvmG_q5RNZGObUb", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1293.9760669426707, + "y": 326.56302247891716, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 23.135415315317864, + "height": 36.920965026381495, + "seed": 1601559161, + "groupIds": [ + "pWdcZI6tf4oqswgXag4nZ", + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 25.503446923456124 + ], + [ + 0.40532440068439735, + 30.11539925866262 + ], + [ + 3.1876357067575456, + 31.80990219489879 + ], + [ + 7.354871715271549, + 33.14980858335201 + ], + [ + 11.79869252961072, + 33.49583129123406 + ], + [ + 15.592667073872917, + 33.15082359908378 + ], + [ + 19.15112117952793, + 32.29793104408936 + ], + [ + 22.700411705907744, + 29.933635302086913 + ], + [ + 22.943034924407762, + 25.503446923456124 + ], + [ + 23.135415315317864, + 3.4738605093386736 + ], + [ + 22.843298221964968, + -0.11105662756083756 + ], + [ + 19.96759931086617, + -2.0234583686940315 + ], + [ + 16.74018432777816, + -3.10679842415807 + ], + [ + 12.544405636380267, + -3.4251337351474422 + ], + [ + 9.881394068972043, + -3.4158696025941406 + ], + [ + 4.21068157261921, + -2.726823717701885 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "ellipse", + "version": 2708, + "versionNonce": 990617589, + "isDeleted": false, + "id": "5N1mH9nR7sZUzHHqQQx9k", + "fillStyle": "solid", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1293.8908851715294, + "y": 323.01507548738044, + "strokeColor": "#000000", + "backgroundColor": "#ffffff", + "width": 23.09445309598474, + "height": 6.279263528942145, + "seed": 1598416279, + "groupIds": [ + "pWdcZI6tf4oqswgXag4nZ", + "eGSGWFEsoUXpvRXdhwmB4", + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "qDAjBIpEBo14rHQOTksb5", + "type": "arrow" + }, + { + "id": "k87rpiXpFp_U0HYO0AxcM", + "type": "arrow" + } + ], + "updated": 1665354780006, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 2549, + "versionNonce": 1200018043, + "isDeleted": false, + "id": "i0C0dBGFwbpjXPU7tEH70", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1294.290593207425, + "y": 335.6774826586258, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 22.7078851177541, + "height": 2.9175197700512814, + "seed": 212214617, + "groupIds": [ + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.6822069967714703, + 0.7376254855918456 + ], + [ + 2.3335499504986355, + 1.5019708109206908 + ], + [ + 4.313367541800834, + 1.9645855665289391 + ], + [ + 8.191658726607782, + 2.5738270010049926 + ], + [ + 11.917316337340536, + 2.6118955005120905 + ], + [ + 15.218956419281389, + 2.402545763947514 + ], + [ + 18.07580483939641, + 2.022006211239024 + ], + [ + 20.76827744650372, + 1.311150431337028 + ], + [ + 22.157553094084978, + 0.515659662955493 + ], + [ + 22.707885117754103, + -0.3056242695391904 + ] + ] + }, + { + "type": "line", + "version": 2577, + "versionNonce": 1994516149, + "isDeleted": false, + "id": "yaYPo2SPPV5_7S6L6vQCg", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1293.6484666284866, + "y": 346.41196748549527, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 22.7078851177541, + "height": 2.9175197700512814, + "seed": 314776247, + "groupIds": [ + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.6822069967714703, + 0.7376254855918456 + ], + [ + 2.3335499504986355, + 1.5019708109206908 + ], + [ + 4.313367541800834, + 1.9645855665289391 + ], + [ + 8.191658726607782, + 2.5738270010049926 + ], + [ + 11.917316337340536, + 2.6118955005120905 + ], + [ + 15.218956419281389, + 2.402545763947514 + ], + [ + 18.07580483939641, + 2.022006211239024 + ], + [ + 20.76827744650372, + 1.311150431337028 + ], + [ + 22.157553094084978, + 0.515659662955493 + ], + [ + 22.707885117754103, + -0.3056242695391904 + ] + ] + }, + { + "type": "arrow", + "version": 6012, + "versionNonce": 1537912763, + "isDeleted": false, + "id": "qDAjBIpEBo14rHQOTksb5", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 6.034244644020102, + "x": 1293.570274904469, + "y": 333.0746597642921, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 9.312361834934453, + "height": 17.671409793681782, + "seed": 1117409337, + "groupIds": [ + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780815, + "link": null, + "locked": false, + "startBinding": { + "elementId": "5N1mH9nR7sZUzHHqQQx9k", + "focus": 0.17059876156304296, + "gap": 5.982831142860286 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -3.465638014918799, + 0.7718575351198613 + ], + [ + -5.876234540924429, + 2.2686732691115044 + ], + [ + -7.611221385428145, + 4.187007865361685 + ], + [ + -8.997800821851968, + 7.29372360223152 + ], + [ + -9.312361834934453, + 9.676626083368124 + ], + [ + -8.554263497776105, + 13.118230666911925 + ], + [ + -7.085140858725698, + 15.667973469486617 + ], + [ + -4.895857300277928, + 17.671409793681782 + ] + ] + }, + { + "type": "arrow", + "version": 6142, + "versionNonce": 224755061, + "isDeleted": false, + "id": "k87rpiXpFp_U0HYO0AxcM", + "fillStyle": "cross-hatch", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0.31086799431261625, + "x": 1316.7807046060636, + "y": 333.0966330332307, + "strokeColor": "#000000", + "backgroundColor": "#fd7e14", + "width": 10.586513682546029, + "height": 18.03828294930969, + "seed": 119319511, + "groupIds": [ + "AcdY7zGYermAUxB_oNgFC", + "gJO33xk2S3V-m-z0IgTNC", + "3vUy3YGgxOz_BM44Ug3Wm" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780815, + "link": null, + "locked": false, + "startBinding": { + "elementId": "5N1mH9nR7sZUzHHqQQx9k", + "focus": -0.31376123802392564, + "gap": 5.844781649134341 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 3.9398194479572393, + 0.7878819391097368 + ], + [ + 6.68024272166653, + 2.31577281187833 + ], + [ + 8.652616894185485, + 4.273933628848496 + ], + [ + 10.228913266236956, + 7.445147366688905 + ], + [ + 10.586513682546029, + 9.877520883980232 + ], + [ + 9.724689522220247, + 13.390576039308046 + ], + [ + 8.05455608074463, + 15.993253621786636 + ], + [ + 5.565726634756559, + 18.03828294930969 + ] + ] + }, + { + "type": "arrow", + "version": 2899, + "versionNonce": 270232853, + "isDeleted": false, + "id": "KcyC9x5YsNNHh13B4n1nx", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1125.0397205887334, + "y": 343.70943497603946, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 134.44921875, + "height": 0, + "seed": 1818528151, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": { + "elementId": "ovxFkdb6JYmDVntxtXYhP", + "focus": -0.4924045350392734, + "gap": 16.183593749999943 + }, + "endBinding": { + "elementId": "QrK4-de5GsjEjuLVX52yz", + "focus": -0.06890602221680925, + "gap": 14.637313842773438 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 134.44921875, + 0 + ] + ] + }, + { + "type": "arrow", + "version": 1980, + "versionNonce": 556071835, + "isDeleted": false, + "id": "wECT1xqiaxw7G2ECloOnC", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1124.9655018387334, + "y": 535.313772100086, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 136.41075134277344, + "height": 0, + "seed": 2147214713, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354780006, + "link": null, + "locked": false, + "startBinding": { + "elementId": "ovxFkdb6JYmDVntxtXYhP", + "focus": 0.4224560986162166, + "gap": 16.109374999999943 + }, + "endBinding": { + "elementId": "S6p1P0dhiH25w1GCOnTPp", + "focus": 0.05076155392223218, + "gap": 12.6875 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 136.41075134277344, + 0 + ] + ] + }, + { + "type": "rectangle", + "version": 571, + "versionNonce": 1719794715, + "isDeleted": false, + "id": "-2EVeASmSy6Dzai2GOLcp", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 863.2155018387333, + "y": 720.4729694209747, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 130.29506185356837, + "height": 78.10175988474508, + "seed": 67551833, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "id": "S4SJSvhxNOOGmTdSWiyvI", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 939, + "versionNonce": 935307029, + "isDeleted": false, + "id": "QGA9hcyCu7CIxfyEs9-kR", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1014.1647205887334, + "y": 722.5747293057198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 131, + "height": 76, + "seed": 515180601, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [ + { + "type": "text", + "id": "GHNC0SfzsX_EaicI6lR8W" + }, + { + "id": "90Cmc39ToPGS7gEAyoJ0m", + "type": "arrow" + } + ], + "updated": 1665354057906, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 589, + "versionNonce": 1656864955, + "isDeleted": false, + "id": "GHNC0SfzsX_EaicI6lR8W", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1039.6647205887334, + "y": 735.5747293057198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 80, + "height": 50, + "seed": 499778169, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "WebHook\nServer", + "baseline": 43, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "QGA9hcyCu7CIxfyEs9-kR", + "originalText": "WebHook\nServer" + }, + { + "type": "line", + "version": 2597, + "versionNonce": 1942223989, + "isDeleted": false, + "id": "GW79IEdKEHYHBe8eryERC", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 6.239310365041726, + "x": 878.2580781443856, + "y": 743.2507145334735, + "strokeColor": "#c92a2a", + "backgroundColor": "#c92a2a", + "width": 18.43824366436583, + "height": 17.30749582081731, + "seed": 659418937, + "groupIds": [ + "s-Go0sl61eFF_N1KyhrXn", + "6rZueH22yRCRic155yH5W" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.207015016493163, + 0 + ], + [ + 14.485970010828552, + 1.639477497167701 + ], + [ + 18.43824366436583, + 8.146087016565971 + ], + [ + 16.16819789905669, + 17.30749582081731 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 2816, + "versionNonce": 400552283, + "isDeleted": false, + "id": "CKp-X9iUrQwI9UIKMEKro", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 3.0886314529654637, + "x": 895.6849336935974, + "y": 760.4526337597035, + "strokeColor": "#40c057", + "backgroundColor": "#40c057", + "width": 18.43824366436583, + "height": 17.30749582081731, + "seed": 808877271, + "groupIds": [ + "s-Go0sl61eFF_N1KyhrXn", + "6rZueH22yRCRic155yH5W" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057906, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.207015016493163, + 0 + ], + [ + 14.485970010828552, + 1.639477497167701 + ], + [ + 18.43824366436583, + 8.146087016565971 + ], + [ + 16.16819789905669, + 17.30749582081731 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 2735, + "versionNonce": 1263969749, + "isDeleted": false, + "id": "UyAUCreQoGs3b5R1XOskd", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 4.678603911496447, + "x": 877.9131086878907, + "y": 760.6056946708031, + "strokeColor": "#364fc7", + "backgroundColor": "#364fc7", + "width": 18.43824366436583, + "height": 17.30749582081731, + "seed": 1914544153, + "groupIds": [ + "s-Go0sl61eFF_N1KyhrXn", + "6rZueH22yRCRic155yH5W" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.207015016493163, + 0 + ], + [ + 14.485970010828552, + 1.639477497167701 + ], + [ + 18.43824366436583, + 8.146087016565971 + ], + [ + 16.16819789905669, + 17.30749582081731 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 2934, + "versionNonce": 925830651, + "isDeleted": false, + "id": "ko_pIRQK8MO7BhRsKb3qD", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 1.549319999052873, + "x": 896.5709399155492, + "y": 743.4814283814526, + "strokeColor": "#15aabf", + "backgroundColor": "#15aabf", + "width": 18.43824366436583, + "height": 17.30749582081731, + "seed": 2017783, + "groupIds": [ + "s-Go0sl61eFF_N1KyhrXn", + "6rZueH22yRCRic155yH5W" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 11.207015016493163, + 0 + ], + [ + 14.485970010828552, + 1.639477497167701 + ], + [ + 18.43824366436583, + 8.146087016565971 + ], + [ + 16.16819789905669, + 17.30749582081731 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "text", + "version": 2340, + "versionNonce": 1636639541, + "isDeleted": false, + "id": "C1z4kYjS5cqqozcVs0TUd", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 914.0789195264196, + "y": 748.7180603534874, + "strokeColor": "#000000", + "backgroundColor": "#364fc7", + "width": 66, + "height": 25, + "seed": 763639033, + "groupIds": [ + "6rZueH22yRCRic155yH5W" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Airflow", + "baseline": 18, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Airflow" + }, + { + "type": "arrow", + "version": 501, + "versionNonce": 598840987, + "isDeleted": false, + "id": "S4SJSvhxNOOGmTdSWiyvI", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1000.8877577080685, + "y": 651.1171875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 76.4718001322866, + "height": 66.55519805571976, + "seed": 1202091319, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": { + "elementId": "APPf-R5uuQPPULrXAaGQl", + "focus": -0.3278195400237206, + "gap": 1 + }, + "endBinding": { + "elementId": "-2EVeASmSy6Dzai2GOLcp", + "focus": -0.47296661320585337, + "gap": 2.8005838652548505 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + -76.4718001322866, + 66.55519805571976 + ] + ] + }, + { + "type": "arrow", + "version": 435, + "versionNonce": 1523381397, + "isDeleted": false, + "id": "90Cmc39ToPGS7gEAyoJ0m", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1028.9420643387334, + "y": 649.4946511807198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 49.6228133264749, + "height": 71.74441315406978, + "seed": 1476473241, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "QGA9hcyCu7CIxfyEs9-kR", + "focus": 0.284443406640884, + "gap": 1.3356649709302246 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 49.6228133264749, + 71.74441315406978 + ] + ] + }, + { + "type": "arrow", + "version": 725, + "versionNonce": 1056310715, + "isDeleted": false, + "id": "qlZi2lbDur0dqp82LYO_L", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 618.9576893387333, + "y": 279.30927731586377, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 146.6601562499999, + "height": 2.02734375, + "seed": 1645838231, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354290661, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Bxgkr0odgFyw9uf9SBh9m", + "focus": 0.012381923951955132, + "gap": 11.729810661266697 + }, + "endBinding": { + "elementId": "c28Qe0SCzKMZd3JP8Kk3b", + "focus": -0.030587254493153675, + "gap": 13.949970758483033 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + -146.6601562499999, + -2.02734375 + ] + ] + }, + { + "type": "text", + "version": 379, + "versionNonce": 599433083, + "isDeleted": false, + "id": "EdJsglDp3VgseBHeDcM06", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 298.0240955887333, + "y": 194.4634011807198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 104, + "height": 25, + "seed": 100125529, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354359520, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Web Users", + "baseline": 18, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Web Users" + }, + { + "type": "text", + "version": 629, + "versionNonce": 130473301, + "isDeleted": false, + "id": "ZTIEx8NH_vfxlwKr_MrOM", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 298.8365955887333, + "y": 507.66261993071976, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 115, + "height": 25, + "seed": 1796789593, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354315695, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Applications", + "baseline": 18, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Applications" + }, + { + "type": "text", + "version": 717, + "versionNonce": 2085277173, + "isDeleted": false, + "id": "iuEBgNrxLinbhXOGKd74q", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 298.0240955887333, + "y": 350.2759011807198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 137, + "height": 25, + "seed": 489058615, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354301139, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "lakeFS clients", + "baseline": 18, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakeFS clients" + }, + { + "type": "text", + "version": 565, + "versionNonce": 203481403, + "isDeleted": false, + "id": "6tgg9ZzYCBwT1BVPy4fTj", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 312.19140625, + "y": 543.07421875, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 130, + "height": 100, + "seed": 309086487, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354315695, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Boto S3 Router\nSpark/S3A\nKafka\nTrino\nmlflow and etc", + "baseline": 94, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "LFEQGiwWugw-bHxiMcZeO", + "originalText": "Boto S3 Router\nSpark/S3A\nKafka\nTrino\nmlflow and etc" + }, + { + "type": "arrow", + "version": 1560, + "versionNonce": 2145815675, + "isDeleted": false, + "id": "AB97GotmGG213cLmZEUpn", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 475.5631580887333, + "y": 570.0230638987974, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 138.765625, + "height": 0, + "seed": 1979912921, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354323419, + "link": null, + "locked": false, + "startBinding": { + "elementId": "LFEQGiwWugw-bHxiMcZeO", + "focus": -0.4191119063855012, + "gap": 13.871751838733303 + }, + "endBinding": { + "elementId": "_philqtA2msBgS2RN0gJq", + "focus": -0.058543712877364124, + "gap": 17.319654411266697 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 138.765625, + 0 + ] + ] + }, + { + "type": "arrow", + "version": 912, + "versionNonce": 771428981, + "isDeleted": false, + "id": "f4fx7Okaga8CbENsmpkmP", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 477.1725330887333, + "y": 419.82416426602776, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 138.92578125, + "height": 0, + "seed": 1508910009, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354311400, + "link": null, + "locked": false, + "startBinding": { + "elementId": "2ZfD2SrM-46E8aH7dhW-L", + "focus": -0.07895086353271633, + "gap": 14.141283088733303 + }, + "endBinding": { + "elementId": "9B7vsP3VbZUcmy_h0fyUT", + "focus": -0.056576449269414304, + "gap": 15.550123161266697 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 138.92578125, + 0 + ] + ] + }, + { + "type": "line", + "version": 162, + "versionNonce": 363053269, + "isDeleted": false, + "id": "EWRSqhwZ2v-na6UAj8dLs", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 530.7819080887333, + "y": 118.75636993071981, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 0, + "height": 699.48046875, + "seed": 341512663, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 699.48046875 + ] + ] + }, + { + "type": "rectangle", + "version": 1055, + "versionNonce": 1020257781, + "isDeleted": false, + "id": "c28Qe0SCzKMZd3JP8Kk3b", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 295.8950993498263, + "y": 228.874571730831, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 162.45246298042406, + "height": 97.2259675748086, + "seed": 151848981, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "qlZi2lbDur0dqp82LYO_L", + "type": "arrow" + } + ], + "updated": 1665354278569, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 838, + "versionNonce": 774463317, + "isDeleted": false, + "id": "D6Vjp_tb3fsyH-ctJK1OH", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 294.52875384721625, + "y": 242.48792166413278, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 163.66954523384825, + "height": 0, + "seed": 540471227, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 163.66954523384825, + 0 + ] + ] + }, + { + "type": "ellipse", + "version": 658, + "versionNonce": 723430523, + "isDeleted": false, + "id": "mFwVUZ5MZ2PAuQg7jQHzl", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 301.59630788569126, + "y": 234.30080117438402, + "strokeColor": "#000000", + "backgroundColor": "#fa5252", + "width": 6.354044670437319, + "height": 6.354044670437319, + "seed": 1455242613, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 703, + "versionNonce": 483175605, + "isDeleted": false, + "id": "rClxA705H32TIXIuT8Bzz", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 314.63314026398604, + "y": 234.30080117438402, + "strokeColor": "#000000", + "backgroundColor": "#fab005", + "width": 6.354044670437319, + "height": 6.354044670437319, + "seed": 1826046043, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 761, + "versionNonce": 1697211675, + "isDeleted": false, + "id": "nXSKBLbL1yLdBl6mmo2b4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 328.2425294563652, + "y": 234.87335798846848, + "strokeColor": "#000000", + "backgroundColor": "#40c057", + "width": 6.354044670437319, + "height": 6.354044670437319, + "seed": 1602209493, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 1040, + "versionNonce": 1751076373, + "isDeleted": false, + "id": "Mh9x2szX1497_0haYDzCg", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 90, + "angle": 0, + "x": 348.6019088665661, + "y": 255.2648863311597, + "strokeColor": "#000000", + "backgroundColor": "#04aaf7", + "width": 54.2680166090767, + "height": 54.2680166090767, + "seed": 778337531, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 1657, + "versionNonce": 1251215803, + "isDeleted": false, + "id": "TlUYtKTx8Yat7J4E-5VYz", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 382.9996450595104, + "y": 275.87488710290137, + "strokeColor": "#087f5b", + "backgroundColor": "#40c057", + "width": 35.85610336609899, + "height": 31.04786919388095, + "seed": 904537141, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -2.325388564058283, + -0.5267206656258894 + ], + [ + -9.619125978041163, + -7.966925549111279 + ], + [ + -14.427821284761738, + -5.522853506520872 + ], + [ + -16.020885987601655, + 3.0148697513649148 + ], + [ + -14.315022585758907, + 8.56196563161751 + ], + [ + -22.13457547355497, + 11.288130750359024 + ], + [ + -21.85344640834797, + 17.331093366033496 + ], + [ + -15.72760937019428, + 16.958642351427752 + ], + [ + -11.045595679276936, + 12.039644336042855 + ], + [ + -4.598716190115264, + 22.697473369376752 + ], + [ + -0.5275508384131768, + 23.08094364476967 + ], + [ + -4.798283119120274, + 6.098058919610197 + ], + [ + 1.7613950690442115, + 5.522853506520834 + ], + [ + 4.017369049100754, + 7.4534279964299595 + ], + [ + 11.73280006089421, + 7.4534279964299595 + ], + [ + 13.721527892544017, + 0.8066098896195579 + ], + [ + 5.362276614134478, + 1.4104653807555212 + ], + [ + 6.969224233774734, + -7.316789162669303 + ], + [ + 3.1410099260787536, + -6.790068497043413 + ], + [ + -0.30195344040758026, + -1.1724493477530322 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 1077, + "versionNonce": 1607910261, + "isDeleted": false, + "id": "U4TbV5DX0XCfbSiVt06V3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 90, + "angle": 0, + "x": 349.9472493966725, + "y": 281.89429122810816, + "strokeColor": "#000000", + "backgroundColor": "#99bcff", + "width": 53.473960938857935, + "height": 0, + "seed": 274802075, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 53.473960938857935, + 0 + ] + ] + }, + { + "type": "line", + "version": 3284, + "versionNonce": 1082202715, + "isDeleted": false, + "id": "QCBQ4txik2LHCSiGrJ3dp", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 90, + "angle": 0, + "x": 356.4199106665164, + "y": 262.28761283303623, + "strokeColor": "#000000", + "backgroundColor": "#99bcff", + "width": 37.243798848241454, + "height": 7.255009908906467, + "seed": 2111952277, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.9812141829093962, + 2.8919681956901635 + ], + [ + 5.212785296889474, + 5.318302940240559 + ], + [ + 10.84355342442342, + 6.787676566209217 + ], + [ + 19.664854934769295, + 6.921573984780089 + ], + [ + 29.959011086092513, + 5.9860900906379 + ], + [ + 35.97091461134096, + 2.582820802133418 + ], + [ + 37.243798848241454, + -0.33343592412637807 + ] + ] + }, + { + "type": "ellipse", + "version": 1101, + "versionNonce": 1467518165, + "isDeleted": false, + "id": "fLYDxkyzzy0bAgkBtuAFo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 90, + "angle": 0, + "x": 366.67630875020285, + "y": 253.50864135942476, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 19.72596765221649, + "height": 56.93834267864238, + "seed": 704625211, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "sharp", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 3489, + "versionNonce": 1673729787, + "isDeleted": false, + "id": "Vv2w5sDYg4vDitoFBaAJs", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 90, + "angle": 0, + "x": 358.34564314934124, + "y": 302.60364451298426, + "strokeColor": "#000000", + "backgroundColor": "#99bcff", + "width": 37.243798848241454, + "height": 7.489841737098479, + "seed": 1117543157, + "groupIds": [ + "R-fNoTpX4maAyiy5bwymb" + ], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354278570, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.212785296889474, + -5.490446992146268 + ], + [ + 10.84355342442342, + -7.007381641354876 + ], + [ + 19.664854934769295, + -7.145613082344516 + ], + [ + 29.959011086092513, + -6.179849230509038 + ], + [ + 35.97091461134096, + -2.666422139481399 + ], + [ + 37.243798848241454, + 0.3442286547539627 + ] + ] + }, + { + "type": "rectangle", + "version": 188, + "versionNonce": 36430203, + "isDeleted": false, + "id": "ovxFkdb6JYmDVntxtXYhP", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 910.8248768387333, + "y": 237.4009011807198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 198.0312500000001, + "height": 418.87109374999994, + "seed": 1940931643, + "groupIds": [], + "strokeSharpness": "sharp", + "boundElements": [ + { + "id": "Bv7_JtCztVbjCF9lOPXMj", + "type": "arrow" + }, + { + "id": "tqXh-I1GblNxzaOjAhp0X", + "type": "arrow" + }, + { + "id": "wECT1xqiaxw7G2ECloOnC", + "type": "arrow" + }, + { + "id": "KcyC9x5YsNNHh13B4n1nx", + "type": "arrow" + } + ], + "updated": 1665354225422, + "link": null, + "locked": false + }, + { + "type": "arrow", + "version": 158, + "versionNonce": 1923373915, + "isDeleted": false, + "id": "Bv7_JtCztVbjCF9lOPXMj", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 815.5475330887333, + "y": 417.6782449307198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 77.0703125, + "height": 0, + "seed": 909377717, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": { + "elementId": "9B7vsP3VbZUcmy_h0fyUT", + "focus": 0.010427646359565883, + "gap": 14.899095588733303 + }, + "endBinding": { + "elementId": "ovxFkdb6JYmDVntxtXYhP", + "focus": 0.1392227993770456, + "gap": 18.207031250000057 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 77.0703125, + 0 + ] + ] + }, + { + "type": "arrow", + "version": 300, + "versionNonce": 1021696981, + "isDeleted": false, + "id": "tqXh-I1GblNxzaOjAhp0X", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 811.9225330887333, + "y": 564.5649636807198, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 78.66796875, + "height": 0, + "seed": 140186005, + "groupIds": [], + "strokeSharpness": "round", + "boundElements": [], + "updated": 1665354057907, + "link": null, + "locked": false, + "startBinding": { + "elementId": "_philqtA2msBgS2RN0gJq", + "focus": -0.05883478643613427, + "gap": 11.274095588733303 + }, + "endBinding": { + "elementId": "ovxFkdb6JYmDVntxtXYhP", + "focus": -0.5621228935662265, + "gap": 20.234375000000057 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 78.66796875, + 0 + ] + ] + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": {} +} \ No newline at end of file diff --git a/v1.46/assets/img/architecture.png b/v1.46/assets/img/architecture.png new file mode 100644 index 000000000..ee6136d76 Binary files /dev/null and b/v1.46/assets/img/architecture.png differ diff --git a/v1.46/assets/img/azure_postgres_conn.png b/v1.46/assets/img/azure_postgres_conn.png new file mode 100644 index 000000000..cfd401583 Binary files /dev/null and b/v1.46/assets/img/azure_postgres_conn.png differ diff --git a/v1.46/assets/img/branching_1.png b/v1.46/assets/img/branching_1.png new file mode 100644 index 000000000..e258a4117 Binary files /dev/null and b/v1.46/assets/img/branching_1.png differ diff --git a/v1.46/assets/img/branching_2.png b/v1.46/assets/img/branching_2.png new file mode 100644 index 000000000..c2f87c57a Binary files /dev/null and b/v1.46/assets/img/branching_2.png differ diff --git a/v1.46/assets/img/branching_3.png b/v1.46/assets/img/branching_3.png new file mode 100644 index 000000000..c6b153ab5 Binary files /dev/null and b/v1.46/assets/img/branching_3.png differ diff --git a/v1.46/assets/img/branching_4.png b/v1.46/assets/img/branching_4.png new file mode 100644 index 000000000..d34bd2d6f Binary files /dev/null and b/v1.46/assets/img/branching_4.png differ diff --git a/v1.46/assets/img/branching_6.png b/v1.46/assets/img/branching_6.png new file mode 100644 index 000000000..69556c739 Binary files /dev/null and b/v1.46/assets/img/branching_6.png differ diff --git a/v1.46/assets/img/branching_7.png b/v1.46/assets/img/branching_7.png new file mode 100644 index 000000000..93bb84bca Binary files /dev/null and b/v1.46/assets/img/branching_7.png differ diff --git a/v1.46/assets/img/branching_8.png b/v1.46/assets/img/branching_8.png new file mode 100644 index 000000000..212a5435b Binary files /dev/null and b/v1.46/assets/img/branching_8.png differ diff --git a/v1.46/assets/img/catalog_export_athena_aws_ui_sql.png b/v1.46/assets/img/catalog_export_athena_aws_ui_sql.png new file mode 100644 index 000000000..25fd7b06a Binary files /dev/null and b/v1.46/assets/img/catalog_export_athena_aws_ui_sql.png differ diff --git a/v1.46/assets/img/cloudera/ClouderaManager.png b/v1.46/assets/img/cloudera/ClouderaManager.png new file mode 100644 index 000000000..fc96e7164 Binary files /dev/null and b/v1.46/assets/img/cloudera/ClouderaManager.png differ diff --git a/v1.46/assets/img/cloudera/ManagementConsole.png b/v1.46/assets/img/cloudera/ManagementConsole.png new file mode 100644 index 000000000..3c91a38f7 Binary files /dev/null and b/v1.46/assets/img/cloudera/ManagementConsole.png differ diff --git a/v1.46/assets/img/cloudera/hadoopFileSystems.png b/v1.46/assets/img/cloudera/hadoopFileSystems.png new file mode 100644 index 000000000..35501d782 Binary files /dev/null and b/v1.46/assets/img/cloudera/hadoopFileSystems.png differ diff --git a/v1.46/assets/img/cloudera/spark_on_yarn.png b/v1.46/assets/img/cloudera/spark_on_yarn.png new file mode 100644 index 000000000..1375a1c48 Binary files /dev/null and b/v1.46/assets/img/cloudera/spark_on_yarn.png differ diff --git a/v1.46/assets/img/cookies.png b/v1.46/assets/img/cookies.png new file mode 100644 index 000000000..682d5705a Binary files /dev/null and b/v1.46/assets/img/cookies.png differ diff --git a/v1.46/assets/img/create-repo-no-sn.png b/v1.46/assets/img/create-repo-no-sn.png new file mode 100644 index 000000000..1b6f5f591 Binary files /dev/null and b/v1.46/assets/img/create-repo-no-sn.png differ diff --git a/v1.46/assets/img/create_repo_azure.png b/v1.46/assets/img/create_repo_azure.png new file mode 100644 index 000000000..29761e623 Binary files /dev/null and b/v1.46/assets/img/create_repo_azure.png differ diff --git a/v1.46/assets/img/create_repo_local.png b/v1.46/assets/img/create_repo_local.png new file mode 100644 index 000000000..214d8d454 Binary files /dev/null and b/v1.46/assets/img/create_repo_local.png differ diff --git a/v1.46/assets/img/create_repo_s3.png b/v1.46/assets/img/create_repo_s3.png new file mode 100644 index 000000000..9c81864cf Binary files /dev/null and b/v1.46/assets/img/create_repo_s3.png differ diff --git a/v1.46/assets/img/csv_export_hooks_data.png b/v1.46/assets/img/csv_export_hooks_data.png new file mode 100644 index 000000000..e4f27ba9e Binary files /dev/null and b/v1.46/assets/img/csv_export_hooks_data.png differ diff --git a/v1.46/assets/img/databricks-arch.png b/v1.46/assets/img/databricks-arch.png new file mode 100644 index 000000000..35aa9fb11 Binary files /dev/null and b/v1.46/assets/img/databricks-arch.png differ diff --git a/v1.46/assets/img/databricks-install-package.png b/v1.46/assets/img/databricks-install-package.png new file mode 100644 index 000000000..116e2408b Binary files /dev/null and b/v1.46/assets/img/databricks-install-package.png differ diff --git a/v1.46/assets/img/databricks_lakefs_conf.png b/v1.46/assets/img/databricks_lakefs_conf.png new file mode 100644 index 000000000..3f586a6e5 Binary files /dev/null and b/v1.46/assets/img/databricks_lakefs_conf.png differ diff --git a/v1.46/assets/img/delete_branch_protection_rule.png b/v1.46/assets/img/delete_branch_protection_rule.png new file mode 100644 index 000000000..1086a5a98 Binary files /dev/null and b/v1.46/assets/img/delete_branch_protection_rule.png differ diff --git a/v1.46/assets/img/delta-diff-operations.png b/v1.46/assets/img/delta-diff-operations.png new file mode 100644 index 000000000..2f9fb4c1e Binary files /dev/null and b/v1.46/assets/img/delta-diff-operations.png differ diff --git a/v1.46/assets/img/delta-diff-table-icon.png b/v1.46/assets/img/delta-diff-table-icon.png new file mode 100644 index 000000000..d3f21d9a0 Binary files /dev/null and b/v1.46/assets/img/delta-diff-table-icon.png differ diff --git a/v1.46/assets/img/delta-lake/change-override.excalidraw b/v1.46/assets/img/delta-lake/change-override.excalidraw new file mode 100644 index 000000000..a212a7386 --- /dev/null +++ b/v1.46/assets/img/delta-lake/change-override.excalidraw @@ -0,0 +1,1230 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "ellipse", + "version": 1027, + "versionNonce": 647317342, + "isDeleted": false, + "id": "fe_nfDr9tTgEIvvpCd4v2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4962.066245843499, + "y": -5360.556055867614, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 105164354, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "hnGG0ORkM1R9laet6Fu-R", + "type": "arrow" + }, + { + "id": "rQ1E3oz56mEvpV-VDg2J7", + "type": "arrow" + } + ], + "updated": 1709205412170, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1021, + "versionNonce": 768337282, + "isDeleted": false, + "id": "SoEo4-k9VMyZ5uBfxyIJz", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5014.204696382217, + "y": -5340.757431640277, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 33.90399169921875, + "height": 20, + "seed": 1138333186, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412170, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Main", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Main", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "ellipse", + "version": 1096, + "versionNonce": 1366338462, + "isDeleted": false, + "id": "tC2jVLTBxRlrnJ20cLkch", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4850.969454146422, + "y": -5626.018764393218, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 368805314, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "hnGG0ORkM1R9laet6Fu-R", + "type": "arrow" + }, + { + "id": "C58w5kcGj-nrQmBTlM3Jh", + "type": "arrow" + } + ], + "updated": 1709205412170, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1019, + "versionNonce": 1006101826, + "isDeleted": false, + "id": "y_svLlAPCrmXkxzAxG-6R", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4973.38442513125, + "y": -5607.453373942603, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 117.24790954589844, + "height": 20, + "seed": 1976139138, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412170, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "table_updates", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "table_updates", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "rectangle", + "version": 2142, + "versionNonce": 1599495170, + "isDeleted": false, + "id": "HabXaoI8CH7irylkjrLYT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4799.428878125622, + "y": -5571.366127982683, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 600761090, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "mDzdjx08Tx8DERt_I6oGJ" + }, + { + "id": "bRWwXzQ71wM5xiGBRf6Fg", + "type": "arrow" + }, + { + "id": "hnGG0ORkM1R9laet6Fu-R", + "type": "arrow" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1595, + "versionNonce": 751754526, + "isDeleted": false, + "id": "mDzdjx08Tx8DERt_I6oGJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4794.428878125622, + "y": -5566.366127982683, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1136603842, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "HabXaoI8CH7irylkjrLYT", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "arrow", + "version": 2359, + "versionNonce": 776654302, + "isDeleted": false, + "id": "hnGG0ORkM1R9laet6Fu-R", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4932.327073881227, + "y": -5362.938868759379, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 93.78930268948716, + "height": 215.66890458533726, + "seed": 667889986, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "startBinding": { + "elementId": "fe_nfDr9tTgEIvvpCd4v2", + "focus": -0.2679492115275112, + "gap": 2.650562094922236 + }, + "endBinding": { + "elementId": "tC2jVLTBxRlrnJ20cLkch", + "focus": 0.1113230338672914, + "gap": 2.6099758318858157 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 93.78930268948716, + -215.66890458533726 + ] + ] + }, + { + "type": "rectangle", + "version": 1255, + "versionNonce": 2134401822, + "isDeleted": false, + "id": "kkijwArtO7QkKMC0uzt0I", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4885.859025042897, + "y": -5760.962795521386, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 626.6013076279205, + "height": 126.94807450850031, + "seed": 2102640770, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1320, + "versionNonce": 2034853314, + "isDeleted": false, + "id": "7rgUrDQRWvC514soL_7DN", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4868.4364576124635, + "y": -5742.445399831138, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 77.93992614746094, + "height": 25, + "seed": 323211970, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_1", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "text", + "version": 820, + "versionNonce": 649744222, + "isDeleted": false, + "id": "b963r4PhcYVJd0u-BDUQB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4802.081990073777, + "y": -5677.471527573025, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.03993225097656, + "height": 20, + "seed": 1110650626, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "rectangle", + "version": 2226, + "versionNonce": 1952144770, + "isDeleted": false, + "id": "Sl5OkkrPd5ze725b4mtoo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4667.254099237502, + "y": -5751.56239205704, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1194758750, + "groupIds": [ + "VN77kAFV_B4v8lW-fk7pS" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "ohSb2OurRU0qNlDleyO4Q" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1679, + "versionNonce": 397718430, + "isDeleted": false, + "id": "ohSb2OurRU0qNlDleyO4Q", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4662.254099237502, + "y": -5746.56239205704, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 95458974, + "groupIds": [ + "VN77kAFV_B4v8lW-fk7pS" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "Sl5OkkrPd5ze725b4mtoo", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 605, + "versionNonce": 1910005058, + "isDeleted": false, + "id": "loW37EPae1kVkmXVuXEKX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4598.661108823377, + "y": -5685.441551837686, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 652219102, + "groupIds": [ + "VN77kAFV_B4v8lW-fk7pS" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "text", + "version": 1937, + "versionNonce": 2086178782, + "isDeleted": false, + "id": "riWYWxJq3vRDrVWOgCCA6", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4757.9349944719015, + "y": -5893.660961115476, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 86.75991821289062, + "height": 25, + "seed": 1680093890, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_2", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_2", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "rectangle", + "version": 1150, + "versionNonce": 2132763906, + "isDeleted": false, + "id": "ENZOzINYjC5AYsv_xPm1D", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4774.080569581123, + "y": -5907.653045786014, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 622.5461620288896, + "height": 132.6602189619462, + "seed": 877955714, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1043, + "versionNonce": 280421406, + "isDeleted": false, + "id": "-XL0T2w3Vjb5h9UVEL_0b", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4695.662828790724, + "y": -5848.503605062552, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.03993225097656, + "height": 20, + "seed": 1662436190, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "rectangle", + "version": 2568, + "versionNonce": 632255682, + "isDeleted": false, + "id": "wu9O7s6wpd_4VPxn-3QCR", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4560.755357500721, + "y": -5896.000928905702, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 162054914, + "groupIds": [ + "Vlvg1MzWmG4hCc5yFv94W" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "62Yr4UKb6bJYYwjlIUFqk" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 2021, + "versionNonce": 1723904094, + "isDeleted": false, + "id": "62Yr4UKb6bJYYwjlIUFqk", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4555.755357500721, + "y": -5891.000928905702, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 324866754, + "groupIds": [ + "Vlvg1MzWmG4hCc5yFv94W" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "wu9O7s6wpd_4VPxn-3QCR", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 947, + "versionNonce": 1730645122, + "isDeleted": false, + "id": "xv5tZ1iJ5py8jiTy_uPxX", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4492.162367086598, + "y": -5829.880088686348, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 1232169602, + "groupIds": [ + "Vlvg1MzWmG4hCc5yFv94W" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "C58w5kcGj-nrQmBTlM3Jh", + "type": "arrow", + "x": -4798.600862037932, + "y": -5600.172060712383, + "width": 512.5524096461941, + "height": 0.3490622264407648, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 707598722, + "version": 942, + "versionNonce": 709293214, + "isDeleted": false, + "boundElements": null, + "updated": 1709205412171, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 512.5524096461941, + 0.3490622264407648 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "tC2jVLTBxRlrnJ20cLkch", + "focus": 0.07834831332532421, + "gap": 1 + }, + "endBinding": { + "elementId": "okug8_b1o-AK1c4zb1cHm", + "focus": -0.08446521325685712, + "gap": 1.2725227181444865 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "ellipse", + "version": 1211, + "versionNonce": 960932930, + "isDeleted": false, + "id": "okug8_b1o-AK1c4zb1cHm", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4284.86130480659, + "y": -5625.7797410291205, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 607501250, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "C58w5kcGj-nrQmBTlM3Jh", + "type": "arrow" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 2883, + "versionNonce": 1694150878, + "isDeleted": false, + "id": "YsU4X2D7I84VH5qlixNZb", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4398.895428489304, + "y": -5568.694422935351, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1290313474, + "groupIds": [ + "i9Ty7PgbjYEYjMfRwAQ_x" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "h5-GUvm1zeBixMrtqQZ-t" + }, + { + "id": "WVniNkq4JXSyroRgK0V3S", + "type": "arrow" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 2333, + "versionNonce": 32232450, + "isDeleted": false, + "id": "h5-GUvm1zeBixMrtqQZ-t", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4393.895428489304, + "y": -5563.694422935351, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 440495810, + "groupIds": [ + "i9Ty7PgbjYEYjMfRwAQ_x" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "YsU4X2D7I84VH5qlixNZb", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 1259, + "versionNonce": 1114432798, + "isDeleted": false, + "id": "tbzp6IoKEQ_09RUCrja2w", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4330.302438075181, + "y": -5502.573582715998, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 728385154, + "groupIds": [ + "i9Ty7PgbjYEYjMfRwAQ_x" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "arrow", + "version": 1429, + "versionNonce": 652491650, + "isDeleted": false, + "id": "rQ1E3oz56mEvpV-VDg2J7", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4909.624399796204, + "y": -5336.177683168648, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 738.7930016379823, + "height": 4.4436759483332935, + "seed": 1654600670, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "startBinding": { + "elementId": "fe_nfDr9tTgEIvvpCd4v2", + "focus": 0.024511903836125275, + "gap": 1 + }, + "endBinding": { + "elementId": "jIcpsn41EuSbCdnDmAuo-", + "focus": -0.02183714962544129, + "gap": 2.9915930404247533 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 738.7930016379823, + -4.4436759483332935 + ] + ] + }, + { + "type": "ellipse", + "version": 1464, + "versionNonce": 1429255582, + "isDeleted": false, + "id": "jIcpsn41EuSbCdnDmAuo-", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4167.849374154192, + "y": -5365.269027149354, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 1049167902, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "rQ1E3oz56mEvpV-VDg2J7", + "type": "arrow" + }, + { + "id": "WVniNkq4JXSyroRgK0V3S", + "type": "arrow" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "id": "WVniNkq4JXSyroRgK0V3S", + "type": "arrow", + "x": -4243.9285422714165, + "y": -5580.419412949524, + "width": 97.3017710453023, + "height": 213.499272478678, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 781890462, + "version": 1378, + "versionNonce": 1536523074, + "isDeleted": false, + "boundElements": null, + "updated": 1709205412171, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 97.3017710453023, + 213.499272478678 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "YsU4X2D7I84VH5qlixNZb", + "focus": -1.0442313323075774, + "gap": 11.724990014172363 + }, + "endBinding": { + "elementId": "jIcpsn41EuSbCdnDmAuo-", + "focus": 0.2565292854670769, + "gap": 1.995269766354479 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "rectangle", + "version": 2851, + "versionNonce": 1862554078, + "isDeleted": false, + "id": "xWTA3yWkCeWYAENX-VNAA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4184.040535592853, + "y": -5302.752519151736, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 690885698, + "groupIds": [ + "QkQ7Tm6n1EKE9giprl8Yk" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "_hCKRkwFYAExNMeSOgBD7" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 2303, + "versionNonce": 797491970, + "isDeleted": false, + "id": "_hCKRkwFYAExNMeSOgBD7", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4179.040535592853, + "y": -5297.752519151736, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1353071618, + "groupIds": [ + "QkQ7Tm6n1EKE9giprl8Yk" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "xWTA3yWkCeWYAENX-VNAA", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 1229, + "versionNonce": 2090493470, + "isDeleted": false, + "id": "WZThBioc8RIWhdjYGzJpn", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4115.44754517873, + "y": -5236.631678932381, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 1849912258, + "groupIds": [ + "QkQ7Tm6n1EKE9giprl8Yk" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "rectangle", + "version": 2255, + "versionNonce": 2101154498, + "isDeleted": false, + "id": "wxujanoYybgFtiGCKIa7g", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4972.787358498867, + "y": -5303.541197409743, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1679913310, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "oTDEG8OMQ_l-2EbLcHLeE" + } + ], + "updated": 1709205412171, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1708, + "versionNonce": 1323561566, + "isDeleted": false, + "id": "oTDEG8OMQ_l-2EbLcHLeE", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4967.787358498867, + "y": -5298.541197409743, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 535270814, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205412171, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "wxujanoYybgFtiGCKIa7g", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": {} +} \ No newline at end of file diff --git a/v1.46/assets/img/delta-lake/concurrent-file-overwrite.png b/v1.46/assets/img/delta-lake/concurrent-file-overwrite.png new file mode 100644 index 000000000..1a6638fb0 Binary files /dev/null and b/v1.46/assets/img/delta-lake/concurrent-file-overwrite.png differ diff --git a/v1.46/assets/img/delta-lake/merge-conflict.excalidraw b/v1.46/assets/img/delta-lake/merge-conflict.excalidraw new file mode 100644 index 000000000..a4c4a6e86 --- /dev/null +++ b/v1.46/assets/img/delta-lake/merge-conflict.excalidraw @@ -0,0 +1,1499 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "ellipse", + "version": 723, + "versionNonce": 2133466690, + "isDeleted": false, + "id": "ogZ6xhDLj2zCWtoP_pLWt", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4874.160408813066, + "y": -4506.505037062658, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 2002829378, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "WLwABAOJOSDvAU7Y7yFjD", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 653, + "versionNonce": 1992253150, + "isDeleted": false, + "id": "4onWMn2YDuyGxv896P6mo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4996.617165404867, + "y": -4487.939646612043, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 33.90399169921875, + "height": 20, + "seed": 463693826, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Main", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Main", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "ellipse", + "version": 1057, + "versionNonce": 1586562562, + "isDeleted": false, + "id": "LLwD7_p7LskpBfMC0seH9", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4414.71486961106, + "y": -4497.629609816566, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 951785410, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "WLwABAOJOSDvAU7Y7yFjD", + "type": "arrow" + }, + { + "id": "2UbB1KMaCo94zzVFFyzaJ", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 581, + "versionNonce": 1399575326, + "isDeleted": false, + "id": "pwYTqEEjmc69F11pHjYF_", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4761.215198764938, + "y": -4719.065058599346, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 2126958466, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "1AwG7wawIuu2khWvBnauV", + "type": "arrow" + }, + { + "id": "Zmhu_d6OClqOpd2i8SGGY", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1006, + "versionNonce": 1212478914, + "isDeleted": false, + "id": "W0IxIdBb5Dii08JD7H2ok", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5021.378123395241, + "y": -4705.650272396779, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 77.93992614746094, + "height": 25, + "seed": 1107660610, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_1", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "text", + "version": 765, + "versionNonce": 50315102, + "isDeleted": false, + "id": "-s9A1E-5y1TaIoxttsW3z", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4780.634325003079, + "y": -4700.357249151108, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 15.579986572265625, + "height": 25, + "seed": 1993758466, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "1AwG7wawIuu2khWvBnauV", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "b1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "b1", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "ellipse", + "version": 1438, + "versionNonce": 2103766402, + "isDeleted": false, + "id": "zyeeGJMekmv4sNA5NsgZ9", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4748.924636543954, + "y": -4913.101373427406, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 767043266, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "BSSZFIfUQiKNRu4Rlehv5", + "type": "arrow" + }, + { + "id": "kUzN5rRojuoaj26p1Biqe", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1382, + "versionNonce": 701244318, + "isDeleted": false, + "id": "8-AtRPpXAf5qu2DAJhnYo", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5018.254038922583, + "y": -5016.093077005808, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 86.75991821289062, + "height": 25, + "seed": 256507522, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_2", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_2", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "ellipse", + "version": 1150, + "versionNonce": 1127214402, + "isDeleted": false, + "id": "9lHLcujynHEODAGbzWpPQ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4532.408416451184, + "y": -4722.149829891901, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 1481301570, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "eAx3-tJuHqAGC3LF-JPx8", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 346, + "versionNonce": 973209566, + "isDeleted": false, + "id": "AuN4vCXle-xo0AA-UkFZA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4677.104788347109, + "y": -4688.523164504064, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.03993225097656, + "height": 20, + "seed": 1436141058, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "ellipse", + "version": 1416, + "versionNonce": 1594160386, + "isDeleted": false, + "id": "FIiTK_X32e1Ezqc3thhIL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4443.213034196717, + "y": -4911.350116757134, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 1614132674, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "2UbB1KMaCo94zzVFFyzaJ", + "type": "arrow" + }, + { + "id": "kUzN5rRojuoaj26p1Biqe", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 1770, + "versionNonce": 934433822, + "isDeleted": false, + "id": "M1n1GhbJ2APqr45TNVrlk", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4823.349082948458, + "y": -4451.587461123878, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1497214338, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "kOypLf68N3ji-dtpJ5vkC" + }, + { + "id": "WLwABAOJOSDvAU7Y7yFjD", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1223, + "versionNonce": 308479170, + "isDeleted": false, + "id": "kOypLf68N3ji-dtpJ5vkC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4818.349082948458, + "y": -4446.587461123878, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 636529986, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "M1n1GhbJ2APqr45TNVrlk", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "rectangle", + "version": 1939, + "versionNonce": 2093883486, + "isDeleted": false, + "id": "n4pOczy0p3JJCiL04xQoO", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4633.7769659940495, + "y": -4653.760927200805, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 732804354, + "groupIds": [ + "6iyWuhB12BXL7nOQpmsnr" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "SvFtSb2ysu9i39_JK-Y_p" + }, + { + "id": "eAx3-tJuHqAGC3LF-JPx8", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1391, + "versionNonce": 384375938, + "isDeleted": false, + "id": "SvFtSb2ysu9i39_JK-Y_p", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4628.7769659940495, + "y": -4648.760927200805, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1274266818, + "groupIds": [ + "6iyWuhB12BXL7nOQpmsnr" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "n4pOczy0p3JJCiL04xQoO", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 322, + "versionNonce": 2099668126, + "isDeleted": false, + "id": "wyjveP0Tl9iJsRVzaYDSn", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4564.970399483169, + "y": -4587.317445755492, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 1617026178, + "groupIds": [ + "6iyWuhB12BXL7nOQpmsnr" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "rectangle", + "version": 2119, + "versionNonce": 441859138, + "isDeleted": false, + "id": "X52u9_BiHmIFEHej-Oxlf", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4579.791605041528, + "y": -4848.161354044467, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 917961794, + "groupIds": [ + "5AgixBmPhcSGPPH16TUb-" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "mstR-FmjOT92MIwTn27yT" + }, + { + "id": "2UbB1KMaCo94zzVFFyzaJ", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1571, + "versionNonce": 174858462, + "isDeleted": false, + "id": "mstR-FmjOT92MIwTn27yT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4574.791605041528, + "y": -4843.161354044467, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 215098370, + "groupIds": [ + "5AgixBmPhcSGPPH16TUb-" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "X52u9_BiHmIFEHej-Oxlf", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 502, + "versionNonce": 982082562, + "isDeleted": false, + "id": "h3Uxd9SHfAxqlA8lGGhlh", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4510.985038530647, + "y": -4781.7178725991525, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 213013442, + "groupIds": [ + "5AgixBmPhcSGPPH16TUb-" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "text", + "version": 420, + "versionNonce": 405820702, + "isDeleted": false, + "id": "jGTLUs8NOf4XHQ71uklsv", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4617.846880675621, + "y": -4876.945196697177, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.03993225097656, + "height": 20, + "seed": 1215728514, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "kUzN5rRojuoaj26p1Biqe", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "arrow", + "version": 918, + "versionNonce": 25269186, + "isDeleted": false, + "id": "eAx3-tJuHqAGC3LF-JPx8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4494.295583776001, + "y": -4676.349554147726, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 93.69919137050329, + "height": 181.92035974070404, + "seed": 912116546, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "n4pOczy0p3JJCiL04xQoO", + "focus": -0.9562988901370194, + "gap": 22.588626946921067 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 93.69919137050329, + 181.92035974070404 + ] + ] + }, + { + "type": "rectangle", + "version": 2019, + "versionNonce": 1633536898, + "isDeleted": false, + "id": "jEpOc9gA9ECPKB7-xBSI9", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4445.877185720612, + "y": -4442.136724682132, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1580294850, + "groupIds": [ + "xYKUDi2W49YoCvXGyJL6e" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "GgXUZoq4b8dgEhIsBvKwx" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1472, + "versionNonce": 1155016094, + "isDeleted": false, + "id": "GgXUZoq4b8dgEhIsBvKwx", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4440.877185720612, + "y": -4437.136724682132, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1429977730, + "groupIds": [ + "xYKUDi2W49YoCvXGyJL6e" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "jEpOc9gA9ECPKB7-xBSI9", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 404, + "versionNonce": 811404098, + "isDeleted": false, + "id": "wb0_cbs-iPoF9sBYWHVv-", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4377.070619209731, + "y": -4375.693243236819, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 1889868354, + "groupIds": [ + "xYKUDi2W49YoCvXGyJL6e" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "arrow", + "version": 1189, + "versionNonce": 1480442334, + "isDeleted": false, + "id": "2UbB1KMaCo94zzVFFyzaJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4409.411395697797, + "y": -4858.488569181136, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 19.946068101089622, + "height": 359.8628738488396, + "seed": 1598849538, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "X52u9_BiHmIFEHej-Oxlf", + "focus": -1.187767547939438, + "gap": 14.932425499329838 + }, + "endBinding": { + "elementId": "LLwD7_p7LskpBfMC0seH9", + "focus": 0.03530906917672053, + "gap": 1 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 19.946068101089622, + 359.8628738488396 + ] + ] + }, + { + "type": "arrow", + "version": 564, + "versionNonce": 618550018, + "isDeleted": false, + "id": "kUzN5rRojuoaj26p1Biqe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4693.745276794551, + "y": -4892.283184485677, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 250.17607011271957, + "height": 0.1822173504415332, + "seed": 472969666, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "zyeeGJMekmv4sNA5NsgZ9", + "focus": -0.13173142231619034, + "gap": 3.9214480665472067 + }, + "endBinding": { + "elementId": "FIiTK_X32e1Ezqc3thhIL", + "focus": 0.19555038025911065, + "gap": 1 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 250.17607011271957, + 0.1822173504415332 + ] + ] + }, + { + "type": "text", + "version": 237, + "versionNonce": 1562760734, + "isDeleted": false, + "id": "itvK8N-Zu8RRRiLKZFlF-", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4373.438068288975, + "y": -4693.725003651696, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 239.95187377929688, + "height": 20, + "seed": 1785817474, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Merge fails due to a conflict ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Merge fails due to a conflict ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "line", + "version": 352, + "versionNonce": 1365287618, + "isDeleted": false, + "id": "WT2vugnQfYCHjuts054FT", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4387.246487652812, + "y": -4680.468472889977, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 13.118373325894026, + "height": 17.951572963169383, + "seed": 1363994946, + "groupIds": [ + "jdLwdOPNVKmDtD3-vXPaA" + ], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.118373325894026, + 17.951572963169383 + ] + ] + }, + { + "type": "line", + "version": 562, + "versionNonce": 1820418654, + "isDeleted": false, + "id": "90Zf7fZd5ulllMEm5WsuN", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4400.957250766653, + "y": -4679.711898531717, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 14.59167480468659, + "height": 15.397382463726899, + "seed": 404910338, + "groupIds": [ + "jdLwdOPNVKmDtD3-vXPaA" + ], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 14.59167480468659, + 15.397382463726899 + ] + ] + }, + { + "type": "arrow", + "version": 718, + "versionNonce": 545454722, + "isDeleted": false, + "id": "WLwABAOJOSDvAU7Y7yFjD", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4820.558902143084, + "y": -4479.856055443219, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 404.6595485926555, + "height": 2.1077378845993735, + "seed": 1114702018, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "ogZ6xhDLj2zCWtoP_pLWt", + "focus": 0.10653700306491325, + "gap": 2.3039530047365133 + }, + "endBinding": { + "elementId": "LLwD7_p7LskpBfMC0seH9", + "focus": 0.16409744857823455, + "gap": 1.5343036149306961 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 404.6595485926555, + 2.1077378845993735 + ] + ] + }, + { + "type": "rectangle", + "version": 557, + "versionNonce": 1908654594, + "isDeleted": false, + "id": "6AuEHwE3GbF_AEtNUpOi3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5038.888221585141, + "y": -5045.482805719315, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 669.6878387115825, + "height": 313.1841823032928, + "seed": 1795743746, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "BSSZFIfUQiKNRu4Rlehv5", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "rectangle", + "version": 888, + "versionNonce": 1322440478, + "isDeleted": false, + "id": "E3Z-rmUWZoIBam9yOArN6", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5037.20835969898, + "y": -4721.684812635738, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 668.0315021892181, + "height": 188.8683344243646, + "seed": 1823576002, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "1AwG7wawIuu2khWvBnauV", + "type": "arrow" + }, + { + "id": "BSSZFIfUQiKNRu4Rlehv5", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false + }, + { + "type": "arrow", + "version": 1609, + "versionNonce": 989908930, + "isDeleted": false, + "id": "1AwG7wawIuu2khWvBnauV", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4834.608650019732, + "y": -4504.732350186047, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 81.79272997921726, + "height": 168.82151191470984, + "seed": 753795970, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "E3Z-rmUWZoIBam9yOArN6", + "focus": -0.18973897987154792, + "gap": 28.084128025325754 + }, + "endBinding": { + "elementId": "-s9A1E-5y1TaIoxttsW3z", + "focus": -1.9469887223595808, + "gap": 12.238418390298648 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 81.79272997921726, + -168.82151191470984 + ] + ] + }, + { + "type": "arrow", + "version": 3003, + "versionNonce": 235308382, + "isDeleted": false, + "id": "BSSZFIfUQiKNRu4Rlehv5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4845.6801434550825, + "y": -4506.576247150813, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 107.13697027572653, + "height": 359.1605733587958, + "seed": 220611394, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "E3Z-rmUWZoIBam9yOArN6", + "focus": -0.29402235190910925, + "gap": 26.240231060559836 + }, + "endBinding": { + "elementId": "zyeeGJMekmv4sNA5NsgZ9", + "focus": 0.3130525348324614, + "gap": 3.5398723098396054 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 107.13697027572653, + -359.1605733587958 + ] + ] + }, + { + "type": "arrow", + "version": 275, + "versionNonce": 529893250, + "isDeleted": false, + "id": "Zmhu_d6OClqOpd2i8SGGY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4708.833474763065, + "y": -4698.500519625166, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 177.86420549665127, + "height": 1.169651576450633, + "seed": 62950146, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "startBinding": { + "elementId": "pwYTqEEjmc69F11pHjYF_", + "focus": -0.1487412340172701, + "gap": 1.1813995419280694 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 177.86420549665127, + 1.169651576450633 + ] + ] + }, + { + "type": "text", + "version": 944, + "versionNonce": 1946823070, + "isDeleted": false, + "id": "ku9fwnNuiqyc625XEumnj", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4781.085874121029, + "y": -4896.7100993141175, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 24.399978637695312, + "height": 25, + "seed": 121363138, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "BSSZFIfUQiKNRu4Rlehv5", + "type": "arrow" + } + ], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "b2", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "b2", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "text", + "version": 256, + "versionNonce": 1519338974, + "isDeleted": false, + "id": "wdm8lY-Pc3nWnX4OKAv9I", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5048.63663120371, + "y": -5118.105719398638, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 723.295654296875, + "height": 35, + "seed": 1131516482, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205338879, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "Merge conflicts by Multi-writers to multiple branches", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Merge conflicts by Multi-writers to multiple branches", + "lineHeight": 1.25, + "baseline": 24 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": {} +} \ No newline at end of file diff --git a/v1.46/assets/img/delta-lake/merge-conflict.png b/v1.46/assets/img/delta-lake/merge-conflict.png new file mode 100644 index 000000000..410c27035 Binary files /dev/null and b/v1.46/assets/img/delta-lake/merge-conflict.png differ diff --git a/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches-merges.png b/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches-merges.png new file mode 100644 index 000000000..c0eaf61c7 Binary files /dev/null and b/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches-merges.png differ diff --git a/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches.excalidraw b/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches.excalidraw new file mode 100644 index 000000000..4490b2808 --- /dev/null +++ b/v1.46/assets/img/delta-lake/multi-writers-with-lakefs-branches.excalidraw @@ -0,0 +1,2250 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "ellipse", + "version": 650, + "versionNonce": 2050083358, + "isDeleted": false, + "id": "I8RGguEZIqgQ2iYCMY84G", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4864.8435371630285, + "y": -6305.026070423402, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 123034946, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "17huQdfU8roWs28YiJOFQ", + "type": "arrow" + }, + { + "id": "HVrlrSxgKJ_WzZvmX9Y5N", + "type": "arrow" + }, + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 549, + "versionNonce": 1810814558, + "isDeleted": false, + "id": "_6MstH_s_f9liSx4461Ju", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4985.144203385236, + "y": -6292.842375059928, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 33.90399169921875, + "height": 20, + "seed": 1484672898, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "Main", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Main", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "ellipse", + "version": 952, + "versionNonce": 57975874, + "isDeleted": false, + "id": "_pxFluJ_cp9w75XmiVlw-", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4403.24190759143, + "y": -6302.532338264451, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 1470024926, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "17huQdfU8roWs28YiJOFQ", + "type": "arrow" + }, + { + "id": "E1zwoGpModq1EKmZdw64p", + "type": "arrow" + } + ], + "updated": 1709205160427, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 476, + "versionNonce": 1066700290, + "isDeleted": false, + "id": "Laf-YTcaNfUCqrV36HKSG", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4749.742236745308, + "y": -6523.9677870472315, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 174303006, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "xTI2SX2v3Ibfasf1nrMu2", + "type": "arrow" + }, + { + "id": "HVrlrSxgKJ_WzZvmX9Y5N", + "type": "arrow" + }, + { + "id": "Vp4_YREtsDSxxEhj5zmIt", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 902, + "versionNonce": 474270494, + "isDeleted": false, + "id": "pe8NNkP-KtTPQCWZgkty5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -5009.905161375611, + "y": -6510.553000844663, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 77.93992614746094, + "height": 25, + "seed": 870126430, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_1", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "text", + "version": 660, + "versionNonce": 905557854, + "isDeleted": false, + "id": "EeXqsSo4ns-ZCoBVD0LCL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4769.161362983448, + "y": -6505.259977598993, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 15.579986572265625, + "height": 25, + "seed": 1722156994, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "HVrlrSxgKJ_WzZvmX9Y5N", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "b1", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "b1", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "ellipse", + "version": 1333, + "versionNonce": 1522998174, + "isDeleted": false, + "id": "frI8ILH80by_7U7jKorJe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4737.451674524324, + "y": -6718.00410187529, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 1746920286, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow" + }, + { + "id": "g5hSV8QmbxT18ueyrdRKR", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1314, + "versionNonce": 1837135170, + "isDeleted": false, + "id": "wZdBpoSIj3AqXRusmeTf5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4998.286156905957, + "y": -6735.140045612948, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 86.75991821289062, + "height": 25, + "seed": 1061668766, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "writer_2", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "writer_2", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "ellipse", + "version": 1046, + "versionNonce": 810503198, + "isDeleted": false, + "id": "nzVX7_-rJ4eSTAyqWqiK4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4520.935454431553, + "y": -6527.052558339786, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 373467074, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "lIVdCf4mUsZuAfK280qjY", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "id": "Bj-ulLnFznW98y3Rqhvuw", + "type": "text", + "x": -4665.631826327479, + "y": -6493.425892951948, + "width": 127.03993225097656, + "height": 20, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 610683970, + "version": 242, + "versionNonce": 1191975006, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "text": "insert to table ", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25 + }, + { + "type": "ellipse", + "version": 1311, + "versionNonce": 296133762, + "isDeleted": false, + "id": "TE6v9B5AAJjR5aIgpNtWi", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4431.740072177087, + "y": -6716.252845205019, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 2024224158, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "P2L0-K8BUaSimEQic5kwJ", + "type": "arrow" + }, + { + "id": "g5hSV8QmbxT18ueyrdRKR", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "id": "G9By4GiKoleoOWTjfa1lw", + "type": "rectangle", + "x": -4817.706152103706, + "y": -6256.431595821762, + "width": 155.75714111328205, + "height": 110, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "seed": 1254547010, + "version": 1680, + "versionNonce": 1948435522, + "isDeleted": false, + "boundElements": [ + { + "type": "text", + "id": "RDlrbWC9cw0RtUoQ7CON4" + }, + { + "id": "17huQdfU8roWs28YiJOFQ", + "type": "arrow" + }, + { + "id": "oLB4nglhzy-mLQ4zT4MlJ", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "id": "RDlrbWC9cw0RtUoQ7CON4", + "type": "text", + "x": -4812.706152103706, + "y": -6251.431595821762, + "width": 142.6879425048828, + "height": 100, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1130590722, + "version": 1133, + "versionNonce": 1737234654, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 94, + "containerId": "G9By4GiKoleoOWTjfa1lw", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 1837, + "versionNonce": 201754078, + "isDeleted": false, + "id": "cYBcW8tBG39tjX_xiVwap", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4622.4522724539875, + "y": -6458.865588115637, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1920357378, + "groupIds": [ + "dDaO15OSW4bSJcLRyePcW" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "Q3K3Y1kUFR-G_EDVTLb7K" + }, + { + "id": "lIVdCf4mUsZuAfK280qjY", + "type": "arrow" + } + ], + "updated": 1709205168693, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1290, + "versionNonce": 590748190, + "isDeleted": false, + "id": "Q3K3Y1kUFR-G_EDVTLb7K", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4617.4522724539875, + "y": -6453.865588115637, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 240157634, + "groupIds": [ + "dDaO15OSW4bSJcLRyePcW" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205168693, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "cYBcW8tBG39tjX_xiVwap", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "id": "7vDHXet_nuaCq2V5wnHx1", + "type": "text", + "x": -4553.645705943107, + "y": -6392.422106670323, + "width": 72.01594543457031, + "height": 20, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [ + "dDaO15OSW4bSJcLRyePcW" + ], + "frameId": null, + "roundness": null, + "seed": 905098498, + "version": 221, + "versionNonce": 171654658, + "isDeleted": false, + "boundElements": null, + "updated": 1709205170678, + "link": null, + "locked": false, + "text": "0001.json", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 2005, + "versionNonce": 526776670, + "isDeleted": false, + "id": "lB-Gdq5GxDC5VH1c4nBNC", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4568.0345947947335, + "y": -6653.176246329492, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 2056296862, + "groupIds": [ + "DfNpHw7mvwumYpxFTzcfp" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "smR7X02aUeC9hzegUQT88" + }, + { + "id": "P2L0-K8BUaSimEQic5kwJ", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1458, + "versionNonce": 1830245250, + "isDeleted": false, + "id": "smR7X02aUeC9hzegUQT88", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4563.0345947947335, + "y": -6648.176246329492, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 650622430, + "groupIds": [ + "DfNpHw7mvwumYpxFTzcfp" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "lB-Gdq5GxDC5VH1c4nBNC", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 389, + "versionNonce": 2111907202, + "isDeleted": false, + "id": "EfkopdUAlSmTX0CpXq5Ri", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4499.228028283853, + "y": -6586.732764884177, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 777079326, + "groupIds": [ + "DfNpHw7mvwumYpxFTzcfp" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205173242, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "text", + "version": 316, + "versionNonce": 463835970, + "isDeleted": false, + "id": "xe_XEuPqPeNXKNJL9Tb6p", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4606.3739186559915, + "y": -6681.847925145062, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.03993225097656, + "height": 20, + "seed": 521780162, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "g5hSV8QmbxT18ueyrdRKR", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "lIVdCf4mUsZuAfK280qjY", + "type": "arrow", + "x": -4445.222993601325, + "y": -6410.958974197634, + "width": 58.69398723782342, + "height": 109.95995253947967, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "seed": 1210218910, + "version": 254, + "versionNonce": 266937950, + "isDeleted": false, + "boundElements": null, + "updated": 1709205168693, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 58.69398723782342, + 109.95995253947967 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "cYBcW8tBG39tjX_xiVwap", + "focus": -0.9617735188030769, + "gap": 21.47213773938165 + }, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "rectangle", + "version": 1955, + "versionNonce": 1512516290, + "isDeleted": false, + "id": "8ys_ZIn-JhNRsE00pwI-Z", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4434.404223700983, + "y": -6247.072599914871, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 1767858754, + "groupIds": [ + "jV9T-u8DWG2DcSQAzysjP" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "0fiGcLhM8scDP1NqbOaWY" + }, + { + "id": "E1zwoGpModq1EKmZdw64p", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 1409, + "versionNonce": 524769886, + "isDeleted": false, + "id": "0fiGcLhM8scDP1NqbOaWY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4429.404223700983, + "y": -6242.072599914871, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1501415938, + "groupIds": [ + "jV9T-u8DWG2DcSQAzysjP" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "8ys_ZIn-JhNRsE00pwI-Z", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 341, + "versionNonce": 998855810, + "isDeleted": false, + "id": "IR47HAyj6b4oE7R88Urdv", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4365.597657190102, + "y": -6180.629118469558, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 1358510530, + "groupIds": [ + "jV9T-u8DWG2DcSQAzysjP" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205181462, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "P2L0-K8BUaSimEQic5kwJ", + "type": "arrow", + "x": -4400.307428477639, + "y": -6667.746697748289, + "width": 25.39454745223884, + "height": 363.68677134764766, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "seed": 2142313346, + "version": 549, + "versionNonce": 812146334, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 25.39454745223884, + 363.68677134764766 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "lB-Gdq5GxDC5VH1c4nBNC", + "focus": -1.1589277588857154, + "gap": 14.570451418797347 + }, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "id": "g5hSV8QmbxT18ueyrdRKR", + "type": "arrow", + "x": -4682.27231477492, + "y": -6697.185912933561, + "width": 250.17607011271957, + "height": 0.1822173504415332, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "seed": 2021419614, + "version": 262, + "versionNonce": 801694274, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 250.17607011271957, + 0.1822173504415332 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "frI8ILH80by_7U7jKorJe", + "focus": -0.13173142231615237, + "gap": 3.921448066547107 + }, + "endBinding": { + "elementId": "TE6v9B5AAJjR5aIgpNtWi", + "focus": 0.19555038025903468, + "gap": 1 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "id": "0WDIjOlhzkviW-BNds7jh", + "type": "text", + "x": -4361.965106269344, + "y": -6498.627732099581, + "width": 239.95187377929688, + "height": 20, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 689578946, + "version": 135, + "versionNonce": 488956290, + "isDeleted": false, + "boundElements": null, + "updated": 1709205217523, + "link": null, + "locked": false, + "text": "Merge fails due to a conflict ", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "Merge fails due to a conflict ", + "lineHeight": 1.25 + }, + { + "id": "-menBhvfH3-WXyHUd8UCx", + "type": "line", + "x": -4375.773525633182, + "y": -6485.371201337862, + "width": 13.118373325894026, + "height": 17.951572963169383, + "angle": 0, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [ + "wn2Ivq1enxBAhYeZ8Q1f1" + ], + "frameId": null, + "roundness": { + "type": 2 + }, + "seed": 1643381122, + "version": 248, + "versionNonce": 1253822238, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + -13.118373325894026, + 17.951572963169383 + ] + ], + "lastCommittedPoint": null, + "startBinding": null, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": null + }, + { + "type": "line", + "version": 458, + "versionNonce": 721598914, + "isDeleted": false, + "id": "LuJH7OPVOwzMNk9-G1ltP", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4389.484288747022, + "y": -6484.614626979603, + "strokeColor": "#e03131", + "backgroundColor": "transparent", + "width": 14.59167480468659, + "height": 15.397382463726899, + "seed": 756159554, + "groupIds": [ + "wn2Ivq1enxBAhYeZ8Q1f1" + ], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 14.59167480468659, + 15.397382463726899 + ] + ] + }, + { + "id": "17huQdfU8roWs28YiJOFQ", + "type": "arrow", + "x": -4811.59438537688, + "y": -6279.120094837566, + "width": 406.57657464437034, + "height": 1.9340786202155869, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "seed": 1238573762, + "version": 418, + "versionNonce": 915169410, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155388, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 406.57657464437034, + 1.9340786202155869 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "I8RGguEZIqgQ2iYCMY84G", + "focus": 0.0761171461910146, + "gap": 1.8824790651980692 + }, + "endBinding": { + "elementId": "_pxFluJ_cp9w75XmiVlw-", + "focus": -0.06367547286089222, + "gap": 1.8162509856176428 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "arrow", + "version": 1392, + "versionNonce": 256467102, + "isDeleted": false, + "id": "O9RyGZ8uliaqanJxFXT6M", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4364.8497363480365, + "y": -6302.3465232926, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 169.7135555023724, + "height": 360.97302809826124, + "seed": 1255772546, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205155388, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "C6VNWiQh3UeB3CE82gqOe", + "focus": 0.08140434080565791, + "gap": 3.4684718011418063 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 169.7135555023724, + -360.97302809826124 + ] + ] + }, + { + "type": "ellipse", + "version": 1278, + "versionNonce": 1827733570, + "isDeleted": false, + "id": "C6VNWiQh3UeB3CE82gqOe", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4207.18357413668, + "y": -6711.496794865507, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 547549250, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "O9RyGZ8uliaqanJxFXT6M", + "type": "arrow" + }, + { + "id": "JlePwwVpmoikeh89UpvPo", + "type": "arrow" + } + ], + "updated": 1709205155388, + "link": null, + "locked": false + }, + { + "id": "cgeJO65hweUreVPeaKagB", + "type": "rectangle", + "x": -5027.41525956551, + "y": -6764.610223357403, + "width": 1237.3977057488414, + "height": 225.9534647176562, + "angle": 0, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1737730526, + "version": 393, + "versionNonce": 46420254, + "isDeleted": false, + "boundElements": [ + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false + }, + { + "id": "oTJTZ68UunsDfa6Bz65__", + "type": "rectangle", + "x": -5025.735397679348, + "y": -6531.796187417758, + "width": 625.2706473214296, + "height": 194.04575892857156, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "dotted", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 2098033822, + "version": 624, + "versionNonce": 926635970, + "isDeleted": false, + "boundElements": [ + { + "id": "xW0DXOgS5gQ44Umiwfsvn", + "type": "arrow" + }, + { + "id": "HVrlrSxgKJ_WzZvmX9Y5N", + "type": "arrow" + }, + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 1651, + "versionNonce": 1855487326, + "isDeleted": false, + "id": "zEGtaUtEdZYSrf4dbL4tF", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3911.67220050099, + "y": -6714.8132823367705, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 49, + "seed": 1726980894, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "JlePwwVpmoikeh89UpvPo", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 522, + "versionNonce": 1604133278, + "isDeleted": false, + "id": "FN4kZnnuFWH3WVLoVME3W", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4153.782156075877, + "y": -6675.249365694784, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 199.99989318847656, + "height": 20, + "seed": 361336670, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "JlePwwVpmoikeh89UpvPo", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "retry to insert to table ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "retry to insert to table ", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "JlePwwVpmoikeh89UpvPo", + "type": "arrow", + "x": -4155.099731157279, + "y": -6692.036355564607, + "width": 241.66386188623483, + "height": 0.5398655690451051, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 2050260382, + "version": 654, + "versionNonce": 1224949214, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155389, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 241.66386188623483, + 0.5398655690451051 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "C6VNWiQh3UeB3CE82gqOe", + "focus": -0.18998017452960755, + "gap": 1.0731331833146314 + }, + "endBinding": { + "elementId": "zEGtaUtEdZYSrf4dbL4tF", + "focus": 0.045787793630051817, + "gap": 1.7915412032686433 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "rectangle", + "version": 2689, + "versionNonce": 1326964318, + "isDeleted": false, + "id": "kZ7xyav-wF0SzTg-W48oA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4041.8004695398613, + "y": -6645.344150989828, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 958905538, + "groupIds": [ + "PZA8FLKHsg03dyKPiMiXY", + "bDtIabYEhbJPretPnaQb7" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "ceiYP_Er-lS11gbNqCEq5" + }, + { + "id": "FfVjSXw-kI_ckJ0J0_3fj", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 2142, + "versionNonce": 899825282, + "isDeleted": false, + "id": "ceiYP_Er-lS11gbNqCEq5", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4036.8004695398613, + "y": -6640.344150989828, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 1951847554, + "groupIds": [ + "PZA8FLKHsg03dyKPiMiXY", + "bDtIabYEhbJPretPnaQb7" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155389, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "kZ7xyav-wF0SzTg-W48oA", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 1061, + "versionNonce": 1554098782, + "isDeleted": false, + "id": "B1XAVEOiOsHSxYhKiQgY3", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3973.4624993511748, + "y": -6581.080343965555, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 883118750, + "groupIds": [ + "SDTkfJK_c9rlszHZCIj2k", + "bDtIabYEhbJPretPnaQb7" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205195630, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "text", + "version": 1143, + "versionNonce": 668151042, + "isDeleted": false, + "id": "eNHxsBS_MtiO9Hv3irz1l", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3974.434004652515, + "y": -6558.378230405287, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 79.07194519042969, + "height": 20, + "seed": 1012704350, + "groupIds": [ + "mCoQNulPCYMTRLSxBRgeh", + "bDtIabYEhbJPretPnaQb7" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205176595, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0002.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0002.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "FfVjSXw-kI_ckJ0J0_3fj", + "type": "arrow", + "x": -3867.157059887273, + "y": -6608.165817588203, + "width": 72.44893829240482, + "height": 307.13824837993525, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1220073950, + "version": 847, + "versionNonce": 1205755614, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155389, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 72.44893829240482, + 307.13824837993525 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "kZ7xyav-wF0SzTg-W48oA", + "focus": -1.1112951929089885, + "gap": 18.886268539306002 + }, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "ellipse", + "version": 1141, + "versionNonce": 1927415298, + "isDeleted": false, + "id": "sza_MQkoDVdKg6y1DOeBx", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3816.5167483317946, + "y": -6299.659237606094, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 51.445748465402175, + "height": 47.903921944754984, + "seed": 743475266, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "id": "E1zwoGpModq1EKmZdw64p", + "type": "arrow" + }, + { + "id": "FfVjSXw-kI_ckJ0J0_3fj", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false + }, + { + "type": "arrow", + "version": 780, + "versionNonce": 322253698, + "isDeleted": false, + "id": "E1zwoGpModq1EKmZdw64p", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4349.5204543130585, + "y": -6275.421048283134, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 531.3035993004432, + "height": 1.377464118514581, + "seed": 978902978, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1709205160760, + "link": null, + "locked": false, + "startBinding": { + "elementId": "_pxFluJ_cp9w75XmiVlw-", + "focus": 0.12887160237939754, + "gap": 2.477823519563046 + }, + "endBinding": { + "elementId": "sza_MQkoDVdKg6y1DOeBx", + "focus": -0.07242759362461478, + "gap": 1.7576857359728635 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 531.3035993004432, + 1.377464118514581 + ] + ] + }, + { + "type": "rectangle", + "version": 2707, + "versionNonce": 1441096386, + "isDeleted": false, + "id": "KPYxoOFrlBo2Sb0dZg9gc", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3828.179226003347, + "y": -6244.524053118949, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 155.75714111328205, + "height": 110, + "seed": 301284034, + "groupIds": [ + "vBqsrhAPC4z2QC277Au0l", + "GNeveevTDFVnckBqN-YoW" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "bXVBM1yeZdSlPFNIlLkyd" + } + ], + "updated": 1709205163717, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 2160, + "versionNonce": 913291906, + "isDeleted": false, + "id": "bXVBM1yeZdSlPFNIlLkyd", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3823.179226003347, + "y": -6239.524053118949, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 142.6879425048828, + "height": 100, + "seed": 314872450, + "groupIds": [ + "vBqsrhAPC4z2QC277Au0l", + "GNeveevTDFVnckBqN-YoW" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205163717, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "my-table/\n _delta_log/\n 0000.json\n \n ", + "textAlign": "left", + "verticalAlign": "top", + "containerId": "KPYxoOFrlBo2Sb0dZg9gc", + "originalText": "my-table/\n _delta_log/\n 0000.json\n \n ", + "lineHeight": 1.25, + "baseline": 94 + }, + { + "type": "text", + "version": 1079, + "versionNonce": 361983390, + "isDeleted": false, + "id": "8Xl61AApZV1BJq0TXaS4E", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3759.841255814661, + "y": -6180.260246094677, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "width": 72.01594543457031, + "height": 20, + "seed": 928182850, + "groupIds": [ + "j4jsjTrEjPq8dqu3lb09a", + "GNeveevTDFVnckBqN-YoW" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205188428, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0001.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0001.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "type": "text", + "version": 1162, + "versionNonce": 1853594626, + "isDeleted": false, + "id": "YjDxh1oKbYOKmIFMj8yFT", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -3760.8127611160016, + "y": -6157.558132534407, + "strokeColor": "#f08c00", + "backgroundColor": "transparent", + "width": 79.07194519042969, + "height": 20, + "seed": 1115642370, + "groupIds": [ + "3R4ogctVE-taTP--Wuu9s", + "GNeveevTDFVnckBqN-YoW" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205184894, + "link": null, + "locked": false, + "fontSize": 16, + "fontFamily": 1, + "text": "0002.json", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "0002.json", + "lineHeight": 1.25, + "baseline": 14 + }, + { + "id": "Dk12h2yd6vKwNh6HJeHKp", + "type": "text", + "x": -3769.3598018666926, + "y": -6324.110945718538, + "width": 129.79193115234375, + "height": 20, + "angle": 0, + "strokeColor": "#2f9e44", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1764308574, + "version": 139, + "versionNonce": 1410598082, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155389, + "link": null, + "locked": false, + "text": "Merge succeeds ", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 14, + "containerId": null, + "originalText": "Merge succeeds ", + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 792, + "versionNonce": 715989086, + "isDeleted": false, + "id": "HVrlrSxgKJ_WzZvmX9Y5N", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4824.464998791562, + "y": -6302.695820250774, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 84.15189668284711, + "height": 173.40085586770238, + "seed": 1437605470, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155389, + "link": null, + "locked": false, + "startBinding": { + "elementId": "I8RGguEZIqgQ2iYCMY84G", + "focus": 0.14746664035579127, + "gap": 1.6478971627609553 + }, + "endBinding": { + "elementId": "EeXqsSo4ns-ZCoBVD0LCL", + "focus": -2.103379796210999, + "gap": 13.268274302467944 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 84.15189668284711, + -173.40085586770238 + ] + ] + }, + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow", + "x": -4837.840230596915, + "y": -6305.734970536609, + "width": 108.28679638350604, + "height": 365.9516799068024, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 844996510, + "version": 2280, + "versionNonce": 2054823042, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155389, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 108.28679638350604, + -365.9516799068024 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "I8RGguEZIqgQ2iYCMY84G", + "focus": -0.22550627866706457, + "gap": 1 + }, + "endBinding": { + "elementId": "NHmazCsKZ8W6WlErUNsKj", + "focus": -2.0766335163406278, + "gap": 15.659499250293266 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "id": "Vp4_YREtsDSxxEhj5zmIt", + "type": "arrow", + "x": -4697.360512743435, + "y": -6503.40324807305, + "width": 177.86420549665127, + "height": 1.169651576450633, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1293681794, + "version": 72, + "versionNonce": 388326558, + "isDeleted": false, + "boundElements": null, + "updated": 1709205155389, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 177.86420549665127, + 1.169651576450633 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "Laf-YTcaNfUCqrV36HKSG", + "focus": -0.14874123401723216, + "gap": 1.1813995419279344 + }, + "endBinding": null, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "text", + "version": 840, + "versionNonce": 1674001474, + "isDeleted": false, + "id": "NHmazCsKZ8W6WlErUNsKj", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4769.612912101397, + "y": -6701.612827762002, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 24.399978637695312, + "height": 25, + "seed": 1929146818, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "-y7UzxgOlXVLOiBsmLUry", + "type": "arrow" + } + ], + "updated": 1709205155389, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "b2", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "b2", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "type": "text", + "version": 846, + "versionNonce": 210444510, + "isDeleted": false, + "id": "muXYWomAHA9tM6b8PBL2N", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": -4240.239686145225, + "y": -6700.096333849421, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 23.779983520507812, + "height": 25, + "seed": 512954050, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1709205155389, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "b3", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "b3", + "lineHeight": 1.25, + "baseline": 17 + }, + { + "id": "PAyd1QjwPqTb57jaQ9LsN", + "type": "text", + "x": -5056.295138910642, + "y": -6860.213749767095, + "width": 338.995849609375, + "height": 35, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "roundness": null, + "seed": 1829891934, + "version": 139, + "versionNonce": 1647610498, + "isDeleted": false, + "boundElements": null, + "updated": 1709205317845, + "link": null, + "locked": false, + "text": "Multi-writers Workaround", + "fontSize": 28, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "baseline": 24, + "containerId": null, + "originalText": "Multi-writers Workaround", + "lineHeight": 1.25 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": {} +} \ No newline at end of file diff --git a/v1.46/assets/img/delta-record-addition.png b/v1.46/assets/img/delta-record-addition.png new file mode 100644 index 000000000..8ce14a97d Binary files /dev/null and b/v1.46/assets/img/delta-record-addition.png differ diff --git a/v1.46/assets/img/delta-schema-change.png b/v1.46/assets/img/delta-schema-change.png new file mode 100644 index 000000000..a56c19dde Binary files /dev/null and b/v1.46/assets/img/delta-schema-change.png differ diff --git a/v1.46/assets/img/deploy/deploy-lakefs.excalidraw.png b/v1.46/assets/img/deploy/deploy-lakefs.excalidraw.png new file mode 100644 index 000000000..8dff1ec99 Binary files /dev/null and b/v1.46/assets/img/deploy/deploy-lakefs.excalidraw.png differ diff --git a/v1.46/assets/img/deploy/deploy-on-aws.excalidraw.png b/v1.46/assets/img/deploy/deploy-on-aws.excalidraw.png new file mode 100644 index 000000000..48f80f96a Binary files /dev/null and b/v1.46/assets/img/deploy/deploy-on-aws.excalidraw.png differ diff --git a/v1.46/assets/img/deploy/deploy-on-azure.excalidraw.png b/v1.46/assets/img/deploy/deploy-on-azure.excalidraw.png new file mode 100644 index 000000000..dde393808 Binary files /dev/null and b/v1.46/assets/img/deploy/deploy-on-azure.excalidraw.png differ diff --git a/v1.46/assets/img/deploy/deploy-on-gcp.excalidraw.png b/v1.46/assets/img/deploy/deploy-on-gcp.excalidraw.png new file mode 100644 index 000000000..a0ccbfdc5 Binary files /dev/null and b/v1.46/assets/img/deploy/deploy-on-gcp.excalidraw.png differ diff --git a/v1.46/assets/img/docs_logo.jpg b/v1.46/assets/img/docs_logo.jpg new file mode 100644 index 000000000..45d7defcb Binary files /dev/null and b/v1.46/assets/img/docs_logo.jpg differ diff --git a/v1.46/assets/img/docs_logo.png b/v1.46/assets/img/docs_logo.png new file mode 100644 index 000000000..000395641 Binary files /dev/null and b/v1.46/assets/img/docs_logo.png differ diff --git a/v1.46/assets/img/duckdb.png b/v1.46/assets/img/duckdb.png new file mode 100644 index 000000000..ced7e6527 Binary files /dev/null and b/v1.46/assets/img/duckdb.png differ diff --git a/v1.46/assets/img/empty_repo_list.png b/v1.46/assets/img/empty_repo_list.png new file mode 100644 index 000000000..4d3ff21ce Binary files /dev/null and b/v1.46/assets/img/empty_repo_list.png differ diff --git a/v1.46/assets/img/enterprise/enterprise-arch.excalidraw b/v1.46/assets/img/enterprise/enterprise-arch.excalidraw new file mode 100644 index 000000000..374e680d8 --- /dev/null +++ b/v1.46/assets/img/enterprise/enterprise-arch.excalidraw @@ -0,0 +1,1450 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "type": "rectangle", + "version": 422, + "versionNonce": 1311699014, + "index": "a2", + "isDeleted": false, + "id": "7eEwcdvLsyiVJQ43QZnR5", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 20, + "angle": 0, + "x": 1750.81438463707, + "y": -926.8342062515839, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 332.31903204258964, + "height": 193.2722988936914, + "seed": 1416552582, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "_6gNM-DRkJaMKQsQ2SyzQ", + "type": "arrow" + }, + { + "id": "YllS82nAwrqMo9ZSXpVb1", + "type": "arrow" + }, + { + "id": "rbM2veZIDmCophmBNfmqs", + "type": "arrow" + }, + { + "id": "Z3xFKVJN3UYuyp4DXgfvn", + "type": "arrow" + } + ], + "updated": 1714308036640, + "link": null, + "locked": false + }, + { + "id": "xnrJUlSpz2SdEBfnxJDd9", + "type": "text", + "x": 1869.9344887052414, + "y": -899.8937096483263, + "width": 79.68794250488281, + "height": 35, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 40, + "groupIds": [], + "frameId": null, + "index": "a4", + "roundness": null, + "seed": 1918401370, + "version": 131, + "versionNonce": 1366144518, + "isDeleted": false, + "boundElements": null, + "updated": 1714308045116, + "link": null, + "locked": false, + "text": "Fluffy", + "fontSize": 28, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Fluffy", + "lineHeight": 1.25 + }, + { + "id": "Gawq1ZdBkaz4K1CvR6dpE", + "type": "rectangle", + "x": 1751.8773808679393, + "y": -1222.8666077688783, + "width": 332.31903204258964, + "height": 193.2722988936914, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 20, + "groupIds": [ + "3asvjcnRT22kyHS7lR_RC" + ], + "frameId": null, + "index": "a5V", + "roundness": { + "type": 3 + }, + "seed": 768808966, + "version": 389, + "versionNonce": 910187974, + "isDeleted": false, + "boundElements": [ + { + "id": "_6gNM-DRkJaMKQsQ2SyzQ", + "type": "arrow" + }, + { + "id": "YjQgpSF3oQ4gnZyPRlRL1", + "type": "arrow" + }, + { + "id": "-iwNE2WrEhf-EWSRbcj0e", + "type": "arrow" + }, + { + "id": "UPmw7JR4shDYhAn-VLH32", + "type": "arrow" + } + ], + "updated": 1714308007790, + "link": null, + "locked": false + }, + { + "id": "0S4TIenuNuF6VNZaBg1-o", + "type": "image", + "x": 1867.9048427104228, + "y": -1146.3573032827424, + "width": 100.2641083576225, + "height": 100.2641083576225, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [ + "3asvjcnRT22kyHS7lR_RC" + ], + "frameId": null, + "index": "a6", + "roundness": null, + "seed": 201664006, + "version": 342, + "versionNonce": 1617396230, + "isDeleted": false, + "boundElements": null, + "updated": 1714307213408, + "link": null, + "locked": false, + "status": "saved", + "fileId": "42f43407d586d13ccf0d29be8715d3ed1c722167", + "scale": [ + 1, + 1 + ] + }, + { + "type": "text", + "version": 413, + "versionNonce": 1565395290, + "index": "a7", + "isDeleted": false, + "id": "0SvUAuAjTuKs2AXrLTM8v", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 40, + "angle": 0, + "x": 1873.992921059156, + "y": -1193.2973743219482, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 88.08795166015625, + "height": 35, + "seed": 1432544582, + "groupIds": [ + "3asvjcnRT22kyHS7lR_RC" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714307213408, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "lakeFS", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakeFS", + "lineHeight": 1.25 + }, + { + "type": "rectangle", + "version": 604, + "versionNonce": 686301958, + "index": "a8", + "isDeleted": false, + "id": "VlJw3MaharsUWGsLHAxL8", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 20, + "angle": 0, + "x": 1332.7380388717243, + "y": -1067.2081658138754, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 200.96411345337762, + "height": 147.57084418783415, + "seed": 1812680454, + "groupIds": [ + "Q80DkbObQsva0V-UQX9pr" + ], + "frameId": null, + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "YjQgpSF3oQ4gnZyPRlRL1", + "type": "arrow" + }, + { + "id": "YllS82nAwrqMo9ZSXpVb1", + "type": "arrow" + }, + { + "id": "UahJQLVQwuqbMTeM6d-qo", + "type": "arrow" + } + ], + "updated": 1714307461317, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 349, + "versionNonce": 176585498, + "index": "a9", + "isDeleted": false, + "id": "GeK2ok6yPGj7y77I1AXx1", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 40, + "angle": 0, + "x": 1363.3101606008545, + "y": -1018.4227437199584, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 139.8198699951172, + "height": 50, + "seed": 5313606, + "groupIds": [ + "Q80DkbObQsva0V-UQX9pr" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714307281900, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Reverse Proxy\n (Nginx)", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Reverse Proxy\n (Nginx)", + "lineHeight": 1.25 + }, + { + "id": "_6gNM-DRkJaMKQsQ2SyzQ", + "type": "arrow", + "x": 1913.5743935383239, + "y": -1018.2268453843149, + "width": 2.830819593245451, + "height": 93.82056632347599, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aA", + "roundness": { + "type": 2 + }, + "seed": 446823578, + "version": 102, + "versionNonce": 1619076186, + "isDeleted": false, + "boundElements": null, + "updated": 1714307404640, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + -2.830819593245451, + 93.82056632347599 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "Gawq1ZdBkaz4K1CvR6dpE", + "focus": 0.007119567756919515, + "gap": 11.367463490871955 + }, + "endBinding": { + "elementId": "7eEwcdvLsyiVJQ43QZnR5", + "focus": -0.05366155495896556, + "gap": 1 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "id": "YjQgpSF3oQ4gnZyPRlRL1", + "type": "arrow", + "x": 1541.7823144330268, + "y": -1021.0620537493118, + "width": 203.05221193095144, + "height": 112.83609439583154, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aC", + "roundness": { + "type": 2 + }, + "seed": 1991576282, + "version": 123, + "versionNonce": 1510353626, + "isDeleted": false, + "boundElements": null, + "updated": 1714307458120, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 203.05221193095144, + -112.83609439583154 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "VlJw3MaharsUWGsLHAxL8", + "focus": 0.25218233616941843, + "gap": 8.080162107924934 + }, + "endBinding": { + "elementId": "Gawq1ZdBkaz4K1CvR6dpE", + "focus": 0.5499057700881548, + "gap": 7.042854503961053 + }, + "startArrowhead": null, + "endArrowhead": "arrow" + }, + { + "type": "arrow", + "version": 376, + "versionNonce": 1542847366, + "index": "aD", + "isDeleted": false, + "id": "YllS82nAwrqMo9ZSXpVb1", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 1543.05591652687, + "y": -973.304206322248, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 190.63891100780506, + "height": 141.50883654523204, + "seed": 3654918, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714307453297, + "link": null, + "locked": false, + "startBinding": { + "elementId": "VlJw3MaharsUWGsLHAxL8", + "focus": -0.41390023099597545, + "gap": 9.353764201768172 + }, + "endBinding": { + "elementId": "7eEwcdvLsyiVJQ43QZnR5", + "focus": -0.6112006930068046, + "gap": 17.119557102394765 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 190.63891100780506, + 141.50883654523204 + ] + ] + }, + { + "type": "arrow", + "version": 460, + "versionNonce": 1498259162, + "index": "aE", + "isDeleted": false, + "id": "UahJQLVQwuqbMTeM6d-qo", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 1173.214626408946, + "y": -996.584375123391, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 147.33746540617403, + "height": 3.8066413573469617, + "seed": 1059135238, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714307487588, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "VlJw3MaharsUWGsLHAxL8", + "focus": 0.12934005021811398, + "gap": 12.185947056604164 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 147.33746540617403, + -3.8066413573469617 + ] + ] + }, + { + "type": "text", + "version": 295, + "versionNonce": 683785434, + "index": "aF", + "isDeleted": false, + "id": "vGEU5fYL-QHgrV6Nnkd2A", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 1169.721772806785, + "y": -1043.1820831354905, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 151.57986450195312, + "height": 25, + "seed": 2031666950, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714307493692, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "lakefs.acme.com", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakefs.acme.com", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 669, + "versionNonce": 2028106522, + "index": "aG", + "isDeleted": false, + "id": "JXoiTwHwA3Ec51XnaA8u5", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 1335.1256345812612, + "y": -1103.9476515209071, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 200.85980224609375, + "height": 50, + "seed": 1422380634, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714307580097, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Prefix based routing\n", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Prefix based routing\n", + "lineHeight": 1.25 + }, + { + "id": "YU-SehvNUMNwO46FA8giW", + "type": "text", + "x": 1549.232180578229, + "y": -891.6480359875749, + "width": 120.33992004394531, + "height": 100, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aI", + "roundness": null, + "seed": 222185050, + "version": 306, + "versionNonce": 875611078, + "isDeleted": false, + "boundElements": null, + "updated": 1714545708243, + "link": null, + "locked": false, + "text": "/saml/\n/sso/\n/api/v1/oidc/\n...", + "fontSize": 20, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "/saml/\n/sso/\n/api/v1/oidc/\n...", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 596, + "versionNonce": 170156442, + "index": "aK", + "isDeleted": false, + "id": "oeGyuD3hWXrd6Q79vLMiN", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 1543.7913471875506, + "y": -1165.1730371584104, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 182.09983825683594, + "height": 50, + "seed": 1650594374, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308072523, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Everything except \nthe below routes", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Everything except \nthe below routes", + "lineHeight": 1.25 + }, + { + "id": "z3wb6uoVDyLZgo_dpKyBl", + "type": "rectangle", + "x": 2215.0054595130678, + "y": -1337.7986665183848, + "width": 434.5793931209073, + "height": 697.3331334702658, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 20, + "groupIds": [], + "frameId": null, + "index": "aO", + "roundness": { + "type": 3 + }, + "seed": 191414106, + "version": 265, + "versionNonce": 1579222362, + "isDeleted": false, + "boundElements": null, + "updated": 1714307698531, + "link": null, + "locked": false + }, + { + "id": "K5NClpRGylEmxB_Wm-5aT", + "type": "text", + "x": 2289.414797449179, + "y": -1309.3478654727094, + "width": 301.95184326171875, + "height": 35, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aP", + "roundness": null, + "seed": 1400470662, + "version": 117, + "versionNonce": 620498394, + "isDeleted": false, + "boundElements": null, + "updated": 1714307719330, + "link": null, + "locked": false, + "text": "Third Party Providers", + "fontSize": 28, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Third Party Providers", + "lineHeight": 1.25 + }, + { + "id": "xnAyKqeLkXYZK021tYs2A", + "type": "image", + "x": 2365.901165471365, + "y": -1238.7492776405395, + "width": 145.4325718820477, + "height": 151.3685952241721, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aR", + "roundness": null, + "seed": 856504070, + "version": 306, + "versionNonce": 355214406, + "isDeleted": false, + "boundElements": [ + { + "id": "-iwNE2WrEhf-EWSRbcj0e", + "type": "arrow" + }, + { + "id": "UPmw7JR4shDYhAn-VLH32", + "type": "arrow" + } + ], + "updated": 1714308007790, + "link": null, + "locked": false, + "status": "saved", + "fileId": "01f5bbd8e4ddca1dd3d02921396d9d769ba087b4", + "scale": [ + 1, + 1 + ] + }, + { + "id": "4epq6d18H1Pi9HUPCc6dY", + "type": "image", + "x": 2368.551779999215, + "y": -841.2435079642223, + "width": 140.13134282634692, + "height": 140.13134282634692, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aU", + "roundness": null, + "seed": 1103079194, + "version": 330, + "versionNonce": 1668814534, + "isDeleted": false, + "boundElements": [ + { + "id": "Z3xFKVJN3UYuyp4DXgfvn", + "type": "arrow" + } + ], + "updated": 1714308036640, + "link": null, + "locked": false, + "status": "saved", + "fileId": "28af0d2b6f1f4003921becff31d875783d4890bb", + "scale": [ + 1, + 1 + ] + }, + { + "id": "PnXkBbkQ8Ig9XtCkVaieD", + "type": "text", + "x": 2322.2106052178233, + "y": -699.0748199926416, + "width": 216.4197998046875, + "height": 25, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aV", + "roundness": null, + "seed": 1266908954, + "version": 148, + "versionNonce": 327826758, + "isDeleted": false, + "boundElements": null, + "updated": 1714307994815, + "link": null, + "locked": false, + "text": "SSO Identity Provider", + "fontSize": 20, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "SSO Identity Provider", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 269, + "versionNonce": 878103622, + "index": "aW", + "isDeleted": false, + "id": "QTjdAhBE1pUDySrts74uo", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 2331.997540218541, + "y": -882.7712134051693, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 213.2398223876953, + "height": 25, + "seed": 862017754, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [ + { + "id": "rbM2veZIDmCophmBNfmqs", + "type": "arrow" + } + ], + "updated": 1714308030286, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "KV DB (e.g. postgres)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "KV DB (e.g. postgres)", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 369, + "versionNonce": 1260573702, + "index": "aX", + "isDeleted": false, + "id": "BhCRzPQ5ZrOOJH4GLlQxZ", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "angle": 0, + "x": 2340.5175368616074, + "y": -1079.1342910633073, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 196.1998291015625, + "height": 25, + "seed": 1439635206, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714307989094, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "Blockstore (e.g. S3)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Blockstore (e.g. S3)", + "lineHeight": 1.25 + }, + { + "type": "arrow", + "version": 374, + "versionNonce": 405390406, + "index": "aY", + "isDeleted": false, + "id": "-iwNE2WrEhf-EWSRbcj0e", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2088.299140850368, + "y": -1138.7026272061219, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 268.6262723812356, + "height": 0.3648552853771889, + "seed": 611672986, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714308023190, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Gawq1ZdBkaz4K1CvR6dpE", + "focus": -0.131149941438345, + "gap": 4.102727939839042 + }, + "endBinding": { + "elementId": "xnAyKqeLkXYZK021tYs2A", + "focus": -0.3277535041961902, + "gap": 8.97575223976105 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 268.6262723812356, + 0.3648552853771889 + ] + ] + }, + { + "type": "arrow", + "version": 1039, + "versionNonce": 1937114374, + "index": "aZ", + "isDeleted": false, + "id": "UPmw7JR4shDYhAn-VLH32", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2087.172907681361, + "y": -1126.1826647261519, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 271.31743208119497, + "height": 144.26068438890047, + "seed": 295832454, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714545494086, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Gawq1ZdBkaz4K1CvR6dpE", + "focus": -0.4852460198889117, + "gap": 2.9764947708317777 + }, + "endBinding": { + "elementId": "WHdKqWgNktR-2_FDZRCeI", + "focus": -0.23420082794532118, + "gap": 17.88202860013439 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 271.31743208119497, + 144.26068438890047 + ] + ] + }, + { + "type": "arrow", + "version": 1011, + "versionNonce": 1544616838, + "index": "aa", + "isDeleted": false, + "id": "rbM2veZIDmCophmBNfmqs", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2086.182076054847, + "y": -854.8814542820674, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 272.84001846872115, + "height": 96.89347749161016, + "seed": 1905304070, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714545494087, + "link": null, + "locked": false, + "startBinding": { + "elementId": "7eEwcdvLsyiVJQ43QZnR5", + "focus": 0.22863684675511883, + "gap": 3.048659375187526 + }, + "endBinding": { + "elementId": "WHdKqWgNktR-2_FDZRCeI", + "focus": 0.17760024048881923, + "gap": 17.350273839121883 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 272.84001846872115, + -96.89347749161016 + ] + ] + }, + { + "type": "arrow", + "version": 548, + "versionNonce": 2143601626, + "index": "ab", + "isDeleted": false, + "id": "Z3xFKVJN3UYuyp4DXgfvn", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2080.7789417058734, + "y": -834.105838706363, + "strokeColor": "#0c8599", + "backgroundColor": "#eaddd7", + "width": 269.68130838476236, + "height": 35.1211459394126, + "seed": 6537926, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1714308045104, + "link": null, + "locked": false, + "startBinding": { + "elementId": "7eEwcdvLsyiVJQ43QZnR5", + "focus": -0.2134039460510788, + "gap": 1 + }, + "endBinding": { + "elementId": "4epq6d18H1Pi9HUPCc6dY", + "focus": 0.20616072996064502, + "gap": 18.091529908579332 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 269.68130838476236, + 35.1211459394126 + ] + ] + }, + { + "id": "SdPJKHOVhZhqFnlMA5tJO", + "type": "text", + "x": 1228.4676373570853, + "y": -989.6839333135869, + "width": 25.239990234375, + "height": 25, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "groupIds": [], + "frameId": null, + "index": "ac", + "roundness": null, + "seed": 924095366, + "version": 47, + "versionNonce": 873928218, + "isDeleted": false, + "boundElements": null, + "updated": 1714308235438, + "link": null, + "locked": false, + "text": "[1]", + "fontSize": 20, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[1]", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 218, + "versionNonce": 244381446, + "index": "ad", + "isDeleted": false, + "id": "C2dCS0ZX7ODBYix-TVr1K", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 1415.517464366353, + "y": -911.5266491522183, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 34.05998229980469, + "height": 25, + "seed": 1245057222, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308251260, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "[2]", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[2]", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 325, + "versionNonce": 242231002, + "index": "ae", + "isDeleted": false, + "id": "JD-FjoYseTKpSeV_sISoB", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 1896.1240231565876, + "y": -1256.7142549400035, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 33.43998718261719, + "height": 25, + "seed": 1167838406, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308283752, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "[3]", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[3]", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 291, + "versionNonce": 1401298906, + "index": "af", + "isDeleted": false, + "id": "PY2IOIshxbmyJGWhH0R2I", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 1894.804382824698, + "y": -718.6392961957064, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 32.61997985839844, + "height": 25, + "seed": 1175851590, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308290945, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "[4]", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[4]", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 585, + "versionNonce": 1938804570, + "index": "ag", + "isDeleted": false, + "id": "FAnl_jQpTBU04XntUl5Tl", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2512.2204326385317, + "y": -965.2341822337299, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 32.17997741699219, + "height": 25, + "seed": 277624346, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308334206, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "[5]", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[5]", + "lineHeight": 1.25 + }, + { + "type": "text", + "version": 296, + "versionNonce": 1877833882, + "index": "ah", + "isDeleted": false, + "id": "bangvYBm9G4mP9YJ9wG0J", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 30, + "angle": 0, + "x": 2517.166552796947, + "y": -788.0557322867609, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "width": 32.61997985839844, + "height": 25, + "seed": 711020998, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1714308338074, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 1, + "text": "[6]", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "[6]", + "lineHeight": 1.25 + }, + { + "id": "WHdKqWgNktR-2_FDZRCeI", + "type": "image", + "x": 2376.3723683626904, + "y": -1035.8263413657496, + "width": 126.50653833186759, + "height": 144.49464751192338, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 80, + "groupIds": [], + "frameId": null, + "index": "aj", + "roundness": null, + "seed": 2029135834, + "version": 168, + "versionNonce": 470249094, + "isDeleted": false, + "boundElements": [ + { + "id": "UPmw7JR4shDYhAn-VLH32", + "type": "arrow" + }, + { + "id": "rbM2veZIDmCophmBNfmqs", + "type": "arrow" + } + ], + "updated": 1714545494086, + "link": null, + "locked": false, + "status": "saved", + "fileId": "8a6fb526480e2312a4c8f4e4503f7c8edca88962", + "scale": [ + 1, + 1 + ] + }, + { + "id": "zEOgmohkYz4YoN-KD02kD", + "type": "text", + "x": 1927.847385099101, + "y": -1009.3837441073794, + "width": 54.0999755859375, + "height": 75, + "angle": 0, + "strokeColor": "#0c8599", + "backgroundColor": "transparent", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 80, + "groupIds": [], + "frameId": null, + "index": "ak", + "roundness": null, + "seed": 1417480838, + "version": 118, + "versionNonce": 512455686, + "isDeleted": false, + "boundElements": null, + "updated": 1714545701359, + "link": null, + "locked": false, + "text": "RBAC\nSTS\n...", + "fontSize": 20, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "RBAC\nSTS\n...", + "lineHeight": 1.25 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "42f43407d586d13ccf0d29be8715d3ed1c722167": { + "mimeType": "image/svg+xml", + "id": "42f43407d586d13ccf0d29be8715d3ed1c722167", + "dataURL": "", + "created": 1714307151956, + "lastRetrieved": 1715236891767 + }, + "01f5bbd8e4ddca1dd3d02921396d9d769ba087b4": { + "mimeType": "image/png", + "id": "01f5bbd8e4ddca1dd3d02921396d9d769ba087b4", + "dataURL": "", + "created": 1714307782740, + "lastRetrieved": 1715236891767 + }, + "28af0d2b6f1f4003921becff31d875783d4890bb": { + "mimeType": "image/png", + "id": "28af0d2b6f1f4003921becff31d875783d4890bb", + "dataURL": "", + "created": 1714307898564, + "lastRetrieved": 1715236891767 + }, + "8a6fb526480e2312a4c8f4e4503f7c8edca88962": { + "mimeType": "image/png", + "id": "8a6fb526480e2312a4c8f4e4503f7c8edca88962", + "dataURL": "", + "created": 1714545483871, + "lastRetrieved": 1715236891767 + } + } +} \ No newline at end of file diff --git a/v1.46/assets/img/enterprise/enterprise-arch.png b/v1.46/assets/img/enterprise/enterprise-arch.png new file mode 100644 index 000000000..a7f37c7f2 Binary files /dev/null and b/v1.46/assets/img/enterprise/enterprise-arch.png differ diff --git a/v1.46/assets/img/experiment_branch_model.png b/v1.46/assets/img/experiment_branch_model.png new file mode 100644 index 000000000..727075730 Binary files /dev/null and b/v1.46/assets/img/experiment_branch_model.png differ diff --git a/v1.46/assets/img/flare_existing_ticket.png b/v1.46/assets/img/flare_existing_ticket.png new file mode 100644 index 000000000..cbead4ac7 Binary files /dev/null and b/v1.46/assets/img/flare_existing_ticket.png differ diff --git a/v1.46/assets/img/flare_new_ticket.png b/v1.46/assets/img/flare_new_ticket.png new file mode 100644 index 000000000..55a52b201 Binary files /dev/null and b/v1.46/assets/img/flare_new_ticket.png differ diff --git a/v1.46/assets/img/gc-sample-commits.png b/v1.46/assets/img/gc-sample-commits.png new file mode 100644 index 000000000..14a301950 Binary files /dev/null and b/v1.46/assets/img/gc-sample-commits.png differ diff --git a/v1.46/assets/img/gc_rules_from_ui.png b/v1.46/assets/img/gc_rules_from_ui.png new file mode 100644 index 000000000..8a60f1ab9 Binary files /dev/null and b/v1.46/assets/img/gc_rules_from_ui.png differ diff --git a/v1.46/assets/img/glue_export_hook_result_log.png b/v1.46/assets/img/glue_export_hook_result_log.png new file mode 100644 index 000000000..fa7e7d7e0 Binary files /dev/null and b/v1.46/assets/img/glue_export_hook_result_log.png differ diff --git a/v1.46/assets/img/grafana.png b/v1.46/assets/img/grafana.png new file mode 100644 index 000000000..7d8f4164a Binary files /dev/null and b/v1.46/assets/img/grafana.png differ diff --git a/v1.46/assets/img/graveler1.png b/v1.46/assets/img/graveler1.png new file mode 100644 index 000000000..96e913972 Binary files /dev/null and b/v1.46/assets/img/graveler1.png differ diff --git a/v1.46/assets/img/graveler2.png b/v1.46/assets/img/graveler2.png new file mode 100644 index 000000000..a7e7ed669 Binary files /dev/null and b/v1.46/assets/img/graveler2.png differ diff --git a/v1.46/assets/img/iso-env-commit.png b/v1.46/assets/img/iso-env-commit.png new file mode 100644 index 000000000..14c7f0d05 Binary files /dev/null and b/v1.46/assets/img/iso-env-commit.png differ diff --git a/v1.46/assets/img/iso-env-create-repo.png b/v1.46/assets/img/iso-env-create-repo.png new file mode 100644 index 000000000..332c338df Binary files /dev/null and b/v1.46/assets/img/iso-env-create-repo.png differ diff --git a/v1.46/assets/img/iso-env-df-counts.png b/v1.46/assets/img/iso-env-df-counts.png new file mode 100644 index 000000000..e0d528da2 Binary files /dev/null and b/v1.46/assets/img/iso-env-df-counts.png differ diff --git a/v1.46/assets/img/iso-env-ex-repo.png b/v1.46/assets/img/iso-env-ex-repo.png new file mode 100644 index 000000000..4bc700514 Binary files /dev/null and b/v1.46/assets/img/iso-env-ex-repo.png differ diff --git a/v1.46/assets/img/iso-env-example-repo.png b/v1.46/assets/img/iso-env-example-repo.png new file mode 100644 index 000000000..69d2bd460 Binary files /dev/null and b/v1.46/assets/img/iso-env-example-repo.png differ diff --git a/v1.46/assets/img/iso-env-objects.png b/v1.46/assets/img/iso-env-objects.png new file mode 100644 index 000000000..e6dac980c Binary files /dev/null and b/v1.46/assets/img/iso-env-objects.png differ diff --git a/v1.46/assets/img/iso-env-two-branches.png b/v1.46/assets/img/iso-env-two-branches.png new file mode 100644 index 000000000..ec35c467c Binary files /dev/null and b/v1.46/assets/img/iso-env-two-branches.png differ diff --git a/v1.46/assets/img/iso-env-upload-object.png b/v1.46/assets/img/iso-env-upload-object.png new file mode 100644 index 000000000..a52efed25 Binary files /dev/null and b/v1.46/assets/img/iso-env-upload-object.png differ diff --git a/v1.46/assets/img/iso_env_dev_test_branching.png b/v1.46/assets/img/iso_env_dev_test_branching.png new file mode 100644 index 000000000..4f3ece156 Binary files /dev/null and b/v1.46/assets/img/iso_env_dev_test_branching.png differ diff --git a/v1.46/assets/img/iso_env_myrepo.png b/v1.46/assets/img/iso_env_myrepo.png new file mode 100644 index 000000000..d15d79570 Binary files /dev/null and b/v1.46/assets/img/iso_env_myrepo.png differ diff --git a/v1.46/assets/img/iso_env_sampledata.png b/v1.46/assets/img/iso_env_sampledata.png new file mode 100644 index 000000000..4dac1d150 Binary files /dev/null and b/v1.46/assets/img/iso_env_sampledata.png differ diff --git a/v1.46/assets/img/iso_env_testenv_branch.png b/v1.46/assets/img/iso_env_testenv_branch.png new file mode 100644 index 000000000..47877e1ca Binary files /dev/null and b/v1.46/assets/img/iso_env_testenv_branch.png differ diff --git a/v1.46/assets/img/job_branching_model.png b/v1.46/assets/img/job_branching_model.png new file mode 100644 index 000000000..31ea3d45d Binary files /dev/null and b/v1.46/assets/img/job_branching_model.png differ diff --git a/v1.46/assets/img/lakeFSArchitecture.excalidraw b/v1.46/assets/img/lakeFSArchitecture.excalidraw new file mode 100644 index 000000000..36bc1a59c --- /dev/null +++ b/v1.46/assets/img/lakeFSArchitecture.excalidraw @@ -0,0 +1,2205 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://excalidraw.com", + "elements": [ + { + "id": "fKdWdkNM9XYc-7n1H_JyG", + "type": "rectangle", + "x": 358.7510170252085, + "y": 3361.6559190850235, + "width": 389.40315290385337, + "height": 461.69072919178734, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aV", + "roundness": { + "type": 3 + }, + "seed": 1911862587, + "version": 901, + "versionNonce": 1470497424, + "isDeleted": false, + "boundElements": [ + { + "id": "94vLrbjNaZ-Is-8OTvFrb", + "type": "arrow" + }, + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow" + }, + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow" + } + ], + "updated": 1731847128060, + "link": null, + "locked": false + }, + { + "id": "rc8yXewY6HEPU-FwSwn3-", + "type": "rectangle", + "x": 882.1160590331633, + "y": 3401.0912156421628, + "width": 181.98388158898558, + "height": 93.00907374489543, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aVG", + "roundness": { + "type": 3 + }, + "seed": 1701992539, + "version": 383, + "versionNonce": 1428039067, + "isDeleted": false, + "boundElements": [ + { + "id": "HCNSYE_W_xRA4UTqgTuuu", + "type": "arrow" + }, + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow" + } + ], + "updated": 1722538040803, + "link": null, + "locked": false + }, + { + "id": "iw67kxGsUMx8mEWqnsFSX", + "type": "text", + "x": 408.4219476901487, + "y": 3312.2646968477534, + "width": 290.91986206173897, + "height": 35, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aW", + "roundness": null, + "seed": 1345709877, + "version": 488, + "versionNonce": 1720203920, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "text": "Applications / Clients", + "fontSize": 28, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Applications / Clients", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "C1-kKCZJAGayLq8AFHcP_", + "type": "text", + "x": 1267.0388624342286, + "y": 3282.233183635602, + "width": 179.22793579101562, + "height": 35, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aWG", + "roundness": null, + "seed": 413344987, + "version": 188, + "versionNonce": 1377980475, + "isDeleted": false, + "boundElements": [], + "updated": 1722537743081, + "link": null, + "locked": false, + "text": "Object Store", + "fontSize": 28, + "fontFamily": 1, + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Object Store", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "9OomwBL5b71Vw2A8hnvQG", + "type": "text", + "x": 1224.8463054463277, + "y": 3587.374912933678, + "width": 165.03779590129852, + "height": 52.303226761469986, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aWV", + "roundness": null, + "seed": 386188827, + "version": 524, + "versionNonce": 1295073589, + "isDeleted": false, + "boundElements": [], + "updated": 1722540974228, + "link": null, + "locked": false, + "text": "lakeFS Managed\nBucket/s*", + "fontSize": 20.921290704587996, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "lakeFS Managed\nBucket/s*", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "9amrjmm6vzKNvV1txfde1", + "type": "text", + "x": 1236.8717429018543, + "y": 3529.01430153436, + "width": 61.157129574892735, + "height": 30.444620954311894, + "angle": 0, + "strokeColor": "#258c81", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aWd", + "roundness": null, + "seed": 235752443, + "version": 564, + "versionNonce": 615590805, + "isDeleted": false, + "boundElements": [ + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow" + } + ], + "updated": 1722537938459, + "link": null, + "locked": false, + "text": "Metadata\nFiles", + "fontSize": 12.177848381724752, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Metadata\nFiles", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "ToMIIJgI6tA6N828WY6rq", + "type": "text", + "x": 1318.8706249429092, + "y": 3530.2828956536414, + "width": 32.62775346636772, + "height": 30.44462095431188, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aWh", + "roundness": null, + "seed": 785567637, + "version": 675, + "versionNonce": 904819125, + "isDeleted": false, + "boundElements": [], + "updated": 1722538759880, + "link": null, + "locked": false, + "text": "Data\nFiles", + "fontSize": 12.177848381724752, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Data\nFiles", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "SSwnkbraAHrWf4k26Yw0H", + "type": "text", + "x": 1250.5732655789477, + "y": 3827.328135003164, + "width": 105.2199496626854, + "height": 50, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aWl", + "roundness": null, + "seed": 1034420533, + "version": 503, + "versionNonce": 518870299, + "isDeleted": false, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "text": "Unmanaged\nBucket/s", + "fontSize": 20, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Unmanaged\nBucket/s", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "xluZVa0RqMjTBXzbb4Q1-", + "type": "image", + "x": 419.965696581165, + "y": 3384.0780312077163, + "width": 81.20410969829025, + "height": 41.95545667744996, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aX", + "roundness": null, + "seed": 1509193627, + "version": 614, + "versionNonce": 1019373712, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0e322a8749f5fd6440745c039816dc9c40aad84a", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "HkpOPASwk5EvC4_E-VpJp", + "type": "image", + "x": 552.1739989997923, + "y": 3347.066295811333, + "width": 126.30894192327354, + "height": 126.30894192327354, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aY", + "roundness": null, + "seed": 198202651, + "version": 1290, + "versionNonce": 1736890000, + "isDeleted": false, + "boundElements": [ + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow" + }, + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow" + } + ], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "97a54ef0c7419b7b3fb8526c7386b19214357d1d", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "4YsWiagkSZ8dcCTLG9cME", + "type": "image", + "x": 403.4820510628298, + "y": 3752.6177834090313, + "width": 115.98831381729387, + "height": 52.91966817914034, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "aa", + "roundness": null, + "seed": 134767253, + "version": 762, + "versionNonce": 1536001168, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "bc3358bef265f112f724f733549d4cd7de1dee8a", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "kcyl4yj8L_MmIInlnhzHu", + "type": "image", + "x": 548.9271827941639, + "y": 3738.8980800186364, + "width": 148.48822663420256, + "height": 66.81970198539115, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "ab", + "roundness": null, + "seed": 218255285, + "version": 664, + "versionNonce": 915305104, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "ec273749eb4f3f53fb9f9368bfa3b7c3af3d1c37", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "XC4ZfmwenvusZshR_P-vg", + "type": "image", + "x": 924.799911833121, + "y": 3408.648705162707, + "width": 97.38623046777609, + "height": 76.28588053309127, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "ac", + "roundness": null, + "seed": 111869403, + "version": 393, + "versionNonce": 204275739, + "isDeleted": false, + "boundElements": [], + "updated": 1722538040803, + "link": null, + "locked": false, + "status": "saved", + "fileId": "57c766831721e9fb1cd3ac887423f3539a447a55", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "PLXjmnHvBPwOF5s_2M80N", + "type": "image", + "x": 1195.0597551231497, + "y": 3273.658229707978, + "width": 57.78143547920078, + "height": 57.78143547920078, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "ad", + "roundness": null, + "seed": 1661221877, + "version": 242, + "versionNonce": 1515296795, + "isDeleted": false, + "boundElements": [], + "updated": 1722537740595, + "link": null, + "locked": false, + "status": "saved", + "fileId": "14425b5cfc24549457fe531a77b482c2336c0d1d", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "ellipse", + "version": 413, + "versionNonce": 973815451, + "index": "am", + "isDeleted": false, + "id": "UgXMbNZ2cISgBrh6ismcD", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1214.1092737197202, + "y": 3426.8532767163338, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 185.48977829962644, + "height": 39.201122455943676, + "seed": 1418848981, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 530, + "versionNonce": 106559931, + "index": "amG", + "isDeleted": false, + "id": "_hfeDSwLW7xKar-modPKg", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1237.1477084934322, + "y": 3716.930821703292, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 127.96422399257345, + "height": 27.043760905302562, + "seed": 1436675061, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 469, + "versionNonce": 776064149, + "index": "an", + "isDeleted": false, + "id": "cwJIg6NqVo2NMUoXhDnjW", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1400.4484126719815, + "y": 3445.719602817568, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 27.435039617214706, + "height": 131.1158512677214, + "seed": 219662389, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -27.435039617214706, + 131.1158512677214 + ] + ] + }, + { + "type": "line", + "version": 586, + "versionNonce": 1846086235, + "index": "anG", + "isDeleted": false, + "id": "2z_yZfH_SMtR9BrRtmG8E", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1365.6978827081718, + "y": 3729.9461738227797, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 18.926668558261248, + "height": 90.45316844089105, + "seed": 1052424667, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -18.926668558261248, + 90.45316844089105 + ] + ] + }, + { + "type": "line", + "version": 673, + "versionNonce": 687780667, + "index": "ao", + "isDeleted": false, + "id": "H_XtdgYlVr7iUFJpqCbWp", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1212.980854385111, + "y": 3446.065940141501, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 27.435039617214724, + "height": 131.11585126772144, + "seed": 1740306837, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 27.435039617214724, + 131.11585126772144 + ] + ] + }, + { + "type": "line", + "version": 790, + "versionNonce": 2060257019, + "index": "aoG", + "isDeleted": false, + "id": "CFRTG3kjWCUrlOVzJ7Vpo", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1236.3692434706413, + "y": 3730.185102281819, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 18.92666855826126, + "height": 90.45316844089108, + "seed": 717923669, + "groupIds": [], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 18.92666855826126, + 90.45316844089108 + ] + ] + }, + { + "type": "line", + "version": 648, + "versionNonce": 1910862325, + "index": "ap", + "isDeleted": false, + "id": "akl7vry0vw3MofnYpEndn", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1241.0102979672567, + "y": 3576.4837884948356, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 132.9629216786985, + "height": 9.835979999683579, + "seed": 1571880693, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 64.33734309428377, + 9.835979999683579 + ], + [ + 132.9629216786985, + 1.3938745221653432 + ] + ] + }, + { + "type": "line", + "version": 767, + "versionNonce": 321417115, + "index": "apG", + "isDeleted": false, + "id": "dZYE85DV9lUebPHuRCjgl", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1255.70597474399, + "y": 3820.156737982185, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 91.72741079519767, + "height": 6.785568236718933, + "seed": 949798523, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 44.384538373350125, + 6.785568236718933 + ], + [ + 91.72741079519767, + 0.961595152072412 + ] + ] + }, + { + "type": "line", + "version": 1251, + "versionNonce": 517641179, + "index": "aq", + "isDeleted": false, + "id": "Fr7P_CVOlfsTYH98ZGC7W", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1401.550970267136, + "y": 3447.863697266036, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 204.20793772300001, + "height": 73.92223842015714, + "seed": 629102677, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722536698420, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.41304386281733, + -6.992816983523688 + ], + [ + -8.914834167083862, + -46.345262208693256 + ], + [ + -56.84501538608395, + -71.12383284278236 + ], + [ + -123.82021026334682, + -73.92223842015714 + ], + [ + -171.6030768976081, + -53.21233210229683 + ], + [ + -195.7948938601827, + -6.2553848968735215 + ], + [ + -188.10231038835357, + -0.4688874539394121 + ] + ] + }, + { + "type": "line", + "version": 1368, + "versionNonce": 692962363, + "index": "aqV", + "isDeleted": false, + "id": "nya0fBWtyDxdIUTsvDnGM", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1366.458506438301, + "y": 3731.4253247753554, + "strokeColor": "#1e1e1e", + "backgroundColor": "#12b886", + "width": 140.87736005396957, + "height": 50.996890297369326, + "seed": 1096102581, + "groupIds": [], + "frameId": null, + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1722538174899, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.803924287309746, + -4.824149379127804 + ], + [ + -6.150095421272407, + -31.972303641918348 + ], + [ + -39.21579047863243, + -49.066348348330976 + ], + [ + -85.42010921675873, + -50.996890297369326 + ], + [ + -118.38417605130402, + -36.709703611302615 + ], + [ + -135.07343576665983, + -4.315415552496223 + ], + [ + -129.76653700655336, + -0.32347237531487866 + ] + ] + }, + { + "id": "rV6rckVoN-g4RRvph38Hu", + "type": "image", + "x": 451.890835890738, + "y": 3695.0442749081635, + "width": 167.7464959390759, + "height": 46.59624887196553, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "as", + "roundness": null, + "seed": 2014252859, + "version": 851, + "versionNonce": 524394640, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "00b5aa438f1425e928c19864c99e6812ef49bd10", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "type": "line", + "version": 688, + "versionNonce": 122856315, + "index": "au", + "isDeleted": false, + "id": "Vz387B1b2vmibhxvnqQSH", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1272.6330163883117, + "y": 3476.5593270274376, + "strokeColor": "#1e1e1e", + "backgroundColor": "#2ce5b4", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 1643133019, + "groupIds": [ + "KDDtU5RMPJKd3ckJPcy7u" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536755118, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 572, + "versionNonce": 633764891, + "index": "av", + "isDeleted": false, + "id": "LmVPBMi0EdUrxG5bIS9xR", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1281.2718300799863, + "y": 3490.260528414957, + "strokeColor": "#1e1e1e", + "backgroundColor": "#2ce5b4", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 94270715, + "groupIds": [ + "KDDtU5RMPJKd3ckJPcy7u" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722536755118, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 797, + "versionNonce": 882216149, + "index": "aw", + "isDeleted": false, + "id": "ev_I2v8T0dl-snNInpQe2", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1340.3672103751042, + "y": 3477.9419639990383, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 2016632891, + "groupIds": [ + "dRFmKGMiTbRgGwfpJBDTA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538773231, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 681, + "versionNonce": 854988539, + "index": "ax", + "isDeleted": false, + "id": "nURCUYMNMyP0_7-yyrBkJ", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1349.0060240667788, + "y": 3491.643165386558, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 1600001269, + "groupIds": [ + "dRFmKGMiTbRgGwfpJBDTA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538773231, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 885, + "versionNonce": 376624187, + "index": "axG", + "isDeleted": false, + "id": "XYysRAd24zX_Q2p2eGnra", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1331.0303939180938, + "y": 3759.3131073009517, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 998622907, + "groupIds": [ + "irGlI1gUH6Ec0F0oKCLIA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 769, + "versionNonce": 1577050357, + "index": "axV", + "isDeleted": false, + "id": "d48yd9wY4bhWwtyxNljv4", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1339.6692076097684, + "y": 3773.0143086884714, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 2108380789, + "groupIds": [ + "irGlI1gUH6Ec0F0oKCLIA" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "type": "line", + "version": 914, + "versionNonce": 1797428443, + "index": "axd", + "isDeleted": false, + "id": "O0Ik_-buSPvejH_LvCcMc", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1287.1445715076575, + "y": 3759.4733474492523, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 28.98577851804393, + "height": 42.9811349199005, + "seed": 392454165, + "groupIds": [ + "hhRkWUwlJ4OviPrDNswcV" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 8.907146199098388, + 15.427629767282772 + ], + [ + 8.907146199098388, + 42.9811349199005 + ], + [ + -20.07863231894554, + 42.9811349199005 + ], + [ + -20.07863231894554, + 1.3633462798771523 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 798, + "versionNonce": 1034025557, + "index": "axl", + "isDeleted": false, + "id": "1ITqAjtgxj2u9TJWayiE8", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1295.783385199332, + "y": 3773.174548836772, + "strokeColor": "#1971c2", + "backgroundColor": "#a5d8ff", + "width": 13.070403643350538, + "height": 13.168462466172253, + "seed": 1201739707, + "groupIds": [ + "hhRkWUwlJ4OviPrDNswcV" + ], + "frameId": null, + "roundness": null, + "boundElements": [], + "updated": 1722538776295, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -13.070403643350538, + 0 + ], + [ + -9.54192475998025, + -13.168462466172253 + ] + ] + }, + { + "id": "HCNSYE_W_xRA4UTqgTuuu", + "type": "arrow", + "x": 1063.9421887202243, + "y": 3447.669888493214, + "width": 185.72806155786225, + "height": 53.8549954573391, + "angle": 0, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b00", + "roundness": { + "type": 2 + }, + "seed": 1298328277, + "version": 228, + "versionNonce": 1686590101, + "isDeleted": false, + "boundElements": [ + { + "type": "text", + "id": "9JxM9dyoRp9meqG8llmiS" + } + ], + "updated": 1722538052045, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 67.67232637227676, + 37.36154794061849 + ], + [ + 185.72806155786225, + 53.8549954573391 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "rc8yXewY6HEPU-FwSwn3-", + "focus": -0.5176202515159919, + "gap": 1 + }, + "endBinding": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow" + }, + { + "id": "9JxM9dyoRp9meqG8llmiS", + "type": "text", + "x": 1091.4385315145344, + "y": 3475.0314364338324, + "width": 80.35196715593338, + "height": 20, + "angle": 0, + "strokeColor": "#258c81", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b00V", + "roundness": null, + "seed": 364298389, + "version": 13, + "versionNonce": 145362939, + "isDeleted": false, + "boundElements": [], + "updated": 1722538051132, + "link": null, + "locked": false, + "text": "Metadata", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "HCNSYE_W_xRA4UTqgTuuu", + "originalText": "Metadata", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "GyPIp862EuIWsednp_LJE", + "type": "arrow", + "x": 749.1541699290619, + "y": 3449.289653386231, + "width": 130.59118001605634, + "height": 0.2223057387436711, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0G", + "roundness": { + "type": 2 + }, + "seed": 52134741, + "version": 1003, + "versionNonce": 846712464, + "isDeleted": false, + "boundElements": [], + "updated": 1731847150799, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 130.59118001605634, + 0.2223057387436711 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": -0.6209306908899385, + "gap": 1, + "fixedPoint": null + }, + "endBinding": { + "elementId": "rc8yXewY6HEPU-FwSwn3-", + "focus": -0.04447413942394212, + "gap": 2.3707090880450323, + "fixedPoint": null + }, + "startArrowhead": "arrow", + "endArrowhead": "arrow" + }, + { + "id": "94vLrbjNaZ-Is-8OTvFrb", + "type": "arrow", + "x": 752.8003605164386, + "y": 3749.072631682589, + "width": 471.93596538921497, + "height": 2.3734459621564383, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0H", + "roundness": { + "type": 2 + }, + "seed": 382577883, + "version": 1016, + "versionNonce": 1653985904, + "isDeleted": false, + "boundElements": [ + { + "type": "text", + "id": "K2M3nmYuNkeXvtPaSRB-X" + } + ], + "updated": 1731847156943, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 471.93596538921497, + 2.3734459621564383 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": 0.6710626336478095, + "gap": 4.646190587376623, + "fixedPoint": null + }, + "endBinding": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow" + }, + { + "id": "K2M3nmYuNkeXvtPaSRB-X", + "type": "text", + "x": 940.0115674424673, + "y": 3697.1162259565835, + "width": 42.895988285541534, + "height": 20, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0HG", + "roundness": null, + "seed": 1105348949, + "version": 7, + "versionNonce": 2025606677, + "isDeleted": false, + "boundElements": [], + "updated": 1722538767196, + "link": null, + "locked": false, + "text": "Data", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "94vLrbjNaZ-Is-8OTvFrb", + "originalText": "Data", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "FoG-M0XjPajxjfIcaEGnO", + "type": "arrow", + "x": 751.3083437605845, + "y": 3557.286409892364, + "width": 476.85489131014253, + "height": 1.0651492733136365, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0HV", + "roundness": { + "type": 2 + }, + "seed": 915786645, + "version": 1343, + "versionNonce": 958418064, + "isDeleted": false, + "boundElements": [ + { + "type": "text", + "id": "e1j3hXwZh4R5fHWNy3zFH" + } + ], + "updated": 1731847128060, + "link": null, + "locked": false, + "points": [ + [ + 0, + 0 + ], + [ + 476.85489131014253, + 1.0651492733136365 + ] + ], + "lastCommittedPoint": null, + "startBinding": { + "elementId": "fKdWdkNM9XYc-7n1H_JyG", + "focus": -0.1541714895390519, + "gap": 3.1541738315226464 + }, + "endBinding": { + "elementId": "9amrjmm6vzKNvV1txfde1", + "focus": -0.9221739442877142, + "gap": 8.708507831127235 + }, + "startArrowhead": "arrow", + "endArrowhead": "arrow" + }, + { + "id": "e1j3hXwZh4R5fHWNy3zFH", + "type": "text", + "x": 944.103324640474, + "y": 3548.844679102245, + "width": 42.895988285541534, + "height": 20, + "angle": 0, + "strokeColor": "#1971c2", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0Hl", + "roundness": null, + "seed": 696045493, + "version": 47, + "versionNonce": 942598901, + "isDeleted": false, + "boundElements": [], + "updated": 1722538756147, + "link": null, + "locked": false, + "text": "Data", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "FoG-M0XjPajxjfIcaEGnO", + "originalText": "Data", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "WdkrYra56assCyd-YXdgZ", + "type": "image", + "x": 446.1287137782343, + "y": 3557.9775185064696, + "width": 217.4295667190742, + "height": 57.98121779175312, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0O", + "roundness": null, + "seed": 66171035, + "version": 569, + "versionNonce": 626986640, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "052b11aa8b22d6eabf95831f5c83abbbdebc45ef", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "sFbdpiCJ6bbj4yMSWgtQQ", + "type": "image", + "x": 403.06042421214914, + "y": 3630.0467361993547, + "width": 113.06387301869914, + "height": 43.812250794745914, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0P", + "roundness": null, + "seed": 1476771035, + "version": 501, + "versionNonce": 564051088, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "733eb5ed9a01ba64388cba865a5ff133b3a04561", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "z5AUyrD2Xi7HOoCOe7774", + "type": "image", + "x": 594.2253081303182, + "y": 3633.2216686746606, + "width": 94.972116571869, + "height": 34.955015127146225, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0Q", + "roundness": null, + "seed": 849362939, + "version": 519, + "versionNonce": 364423824, + "isDeleted": false, + "boundElements": [], + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "8a405086d877b8b98996d7386da4ece965e92454", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "9OvAOafyzEx2eTcf6a6sZ", + "type": "text", + "x": 1185.3806807731926, + "y": 3331.483415371421, + "width": 275.1998291015625, + "height": 20, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "#ffffff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0R", + "roundness": null, + "seed": 1762599413, + "version": 57, + "versionNonce": 1717000277, + "isDeleted": false, + "boundElements": [], + "updated": 1722538651923, + "link": null, + "locked": false, + "text": "S3 / Azure Blob / GCS / MinIO ...", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "S3 / Azure Blob / GCS / MinIO ...", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "Fhakw2nF56ggm_i2A1Hlz", + "type": "text", + "x": 367.03081660045905, + "y": 3856.102039602422, + "width": 595.9036865234375, + "height": 20, + "angle": 0, + "strokeColor": "#1e1e1e", + "backgroundColor": "#a5d8ff", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0S", + "roundness": null, + "seed": 1798373467, + "version": 792, + "versionNonce": 622779504, + "isDeleted": false, + "boundElements": [], + "updated": 1731847183579, + "link": null, + "locked": false, + "text": "* All Data & Metadata is always stored within your Account and Network", + "fontSize": 16, + "fontFamily": 1, + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "* All Data & Metadata is always stored within your Account and Network", + "autoResize": true, + "lineHeight": 1.25 + }, + { + "id": "o8X_Fx326X8JiZIkuuFA9", + "type": "image", + "x": 484.73042132190164, + "y": 3441.4926633490163, + "width": 119.0231773445256, + "height": 47.53260544515676, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#1e6b61", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0U", + "roundness": null, + "seed": 865759344, + "version": 847, + "versionNonce": 207935632, + "isDeleted": false, + "boundElements": null, + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "b001d66de9fa144d4d413562a3a62181d0d4450e", + "scale": [ + 1, + 1 + ], + "crop": null + }, + { + "id": "JaEdbiO1wbDmtQyyd6isX", + "type": "image", + "x": 369.14936862099376, + "y": 3505.4835047322053, + "width": 176.6006152010927, + "height": 44.388175607862905, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#1e6b61", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0W", + "roundness": null, + "seed": 585527920, + "version": 961, + "versionNonce": 2036024976, + "isDeleted": false, + "boundElements": null, + "updated": 1731847128061, + "link": null, + "locked": false, + "status": "saved", + "fileId": "f6104a267b941b1e71e4188461fabc63c1406c02", + "scale": [ + 1, + 1 + ], + "crop": { + "x": 14.061882697898454, + "y": 17.14293450507074, + "width": 1181.0609396595476, + "height": 296.85706549492926, + "naturalWidth": 1208, + "naturalHeight": 314 + } + }, + { + "id": "TXSTOQI5F5qlrOUNCuuHb", + "type": "image", + "x": 559.6305978501075, + "y": 3496.5944271721455, + "width": 170.64923206272238, + "height": 60.146860481123454, + "angle": 0, + "strokeColor": "transparent", + "backgroundColor": "#1e6b61", + "fillStyle": "solid", + "strokeWidth": 4, + "strokeStyle": "solid", + "roughness": 2, + "opacity": 100, + "groupIds": [], + "frameId": null, + "index": "b0X", + "roundness": null, + "seed": 208157808, + "version": 490, + "versionNonce": 1143701616, + "isDeleted": false, + "boundElements": null, + "updated": 1731847139613, + "link": null, + "locked": false, + "status": "saved", + "fileId": "0909b69b3afd8b5bbaf69e31cbd2868cc931bde5", + "scale": [ + 1, + 1 + ], + "crop": null + } + ], + "appState": { + "gridSize": 20, + "gridStep": 5, + "gridModeEnabled": false, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "0e322a8749f5fd6440745c039816dc9c40aad84a": { + "mimeType": "image/png", + "id": "0e322a8749f5fd6440745c039816dc9c40aad84a", + "dataURL": "", + "created": 1722534560073, + "lastRetrieved": 1722534560073 + }, + "97a54ef0c7419b7b3fb8526c7386b19214357d1d": { + "mimeType": "image/png", + "id": "97a54ef0c7419b7b3fb8526c7386b19214357d1d", + "dataURL": "", + "created": 1722534694523, + "lastRetrieved": 1722534694523 + }, + "bc3358bef265f112f724f733549d4cd7de1dee8a": { + "mimeType": "image/png", + "id": "bc3358bef265f112f724f733549d4cd7de1dee8a", + "dataURL": "", + "created": 1722534760628, + "lastRetrieved": 1722534760628 + }, + "ec273749eb4f3f53fb9f9368bfa3b7c3af3d1c37": { + "mimeType": "image/png", + "id": "ec273749eb4f3f53fb9f9368bfa3b7c3af3d1c37", + "dataURL": "", + "created": 1722534903542, + "lastRetrieved": 1722534903542 + }, + "57c766831721e9fb1cd3ac887423f3539a447a55": { + "mimeType": "image/png", + "id": "57c766831721e9fb1cd3ac887423f3539a447a55", + "dataURL": "", + "created": 1722535141187, + "lastRetrieved": 1722535141187 + }, + "14425b5cfc24549457fe531a77b482c2336c0d1d": { + "mimeType": "image/png", + "id": "14425b5cfc24549457fe531a77b482c2336c0d1d", + "dataURL": "", + "created": 1722535283692, + "lastRetrieved": 1722535283692 + }, + "00b5aa438f1425e928c19864c99e6812ef49bd10": { + "mimeType": "image/png", + "id": "00b5aa438f1425e928c19864c99e6812ef49bd10", + "dataURL": "", + "created": 1722536352800, + "lastRetrieved": 1722536352800 + }, + "052b11aa8b22d6eabf95831f5c83abbbdebc45ef": { + "mimeType": "image/png", + "id": "052b11aa8b22d6eabf95831f5c83abbbdebc45ef", + "dataURL": "", + "created": 1722538330971, + "lastRetrieved": 1722538330971 + }, + "733eb5ed9a01ba64388cba865a5ff133b3a04561": { + "mimeType": "image/png", + "id": "733eb5ed9a01ba64388cba865a5ff133b3a04561", + "dataURL": "", + "created": 1722538387275, + "lastRetrieved": 1722538387275 + }, + "8a405086d877b8b98996d7386da4ece965e92454": { + "mimeType": "image/png", + "id": "8a405086d877b8b98996d7386da4ece965e92454", + "dataURL": "", + "created": 1722538515722, + "lastRetrieved": 1722538515722 + }, + "b001d66de9fa144d4d413562a3a62181d0d4450e": { + "mimeType": "image/png", + "id": "b001d66de9fa144d4d413562a3a62181d0d4450e", + "dataURL": "", + "created": 1731846113674, + "lastRetrieved": 1731846113674 + }, + "f6104a267b941b1e71e4188461fabc63c1406c02": { + "mimeType": "image/png", + "id": "f6104a267b941b1e71e4188461fabc63c1406c02", + "dataURL": "", + "created": 1731846238642, + "lastRetrieved": 1731846238642 + }, + "0909b69b3afd8b5bbaf69e31cbd2868cc931bde5": { + "mimeType": "image/png", + "id": "0909b69b3afd8b5bbaf69e31cbd2868cc931bde5", + "dataURL": "", + "created": 1731846356834, + "lastRetrieved": 1731846356834 + } + } +} \ No newline at end of file diff --git a/v1.46/assets/img/lakeFSArchitecture.png b/v1.46/assets/img/lakeFSArchitecture.png new file mode 100644 index 000000000..8e86ecd18 Binary files /dev/null and b/v1.46/assets/img/lakeFSArchitecture.png differ diff --git a/v1.46/assets/img/lakeFS_integration.png b/v1.46/assets/img/lakeFS_integration.png new file mode 100644 index 000000000..55b6f7dfe Binary files /dev/null and b/v1.46/assets/img/lakeFS_integration.png differ diff --git a/v1.46/assets/img/lakectl-local/axolotl.png b/v1.46/assets/img/lakectl-local/axolotl.png new file mode 100644 index 000000000..573ce5851 Binary files /dev/null and b/v1.46/assets/img/lakectl-local/axolotl.png differ diff --git a/v1.46/assets/img/lakectl-local/code_repo.png b/v1.46/assets/img/lakectl-local/code_repo.png new file mode 100644 index 000000000..f40a392a4 Binary files /dev/null and b/v1.46/assets/img/lakectl-local/code_repo.png differ diff --git a/v1.46/assets/img/lakectl-local/experiment-branch.png b/v1.46/assets/img/lakectl-local/experiment-branch.png new file mode 100644 index 000000000..1454bc3ea Binary files /dev/null and b/v1.46/assets/img/lakectl-local/experiment-branch.png differ diff --git a/v1.46/assets/img/lakectl-local/lakefs-commit-git-commit-id.png b/v1.46/assets/img/lakectl-local/lakefs-commit-git-commit-id.png new file mode 100644 index 000000000..bb26e668f Binary files /dev/null and b/v1.46/assets/img/lakectl-local/lakefs-commit-git-commit-id.png differ diff --git a/v1.46/assets/img/lakectl-local/lakefs-repo-with-train-dataset.png b/v1.46/assets/img/lakectl-local/lakefs-repo-with-train-dataset.png new file mode 100644 index 000000000..44e7a7ce1 Binary files /dev/null and b/v1.46/assets/img/lakectl-local/lakefs-repo-with-train-dataset.png differ diff --git a/v1.46/assets/img/lakectl-local/tes-and-train-dataset.png b/v1.46/assets/img/lakectl-local/tes-and-train-dataset.png new file mode 100644 index 000000000..0e39a30fe Binary files /dev/null and b/v1.46/assets/img/lakectl-local/tes-and-train-dataset.png differ diff --git a/v1.46/assets/img/lakefs-logo-with-text.png b/v1.46/assets/img/lakefs-logo-with-text.png new file mode 100644 index 000000000..b6a2d0ea2 Binary files /dev/null and b/v1.46/assets/img/lakefs-logo-with-text.png differ diff --git a/v1.46/assets/img/lakefs-release-asset.png b/v1.46/assets/img/lakefs-release-asset.png new file mode 100644 index 000000000..4ebf17c5b Binary files /dev/null and b/v1.46/assets/img/lakefs-release-asset.png differ diff --git a/v1.46/assets/img/lakefs_table.png b/v1.46/assets/img/lakefs_table.png new file mode 100644 index 000000000..9d43a7ee0 Binary files /dev/null and b/v1.46/assets/img/lakefs_table.png differ diff --git a/v1.46/assets/img/login.png b/v1.46/assets/img/login.png new file mode 100644 index 000000000..f552ccde3 Binary files /dev/null and b/v1.46/assets/img/login.png differ diff --git a/v1.46/assets/img/logo_large.png b/v1.46/assets/img/logo_large.png new file mode 100644 index 000000000..45d7c34e7 Binary files /dev/null and b/v1.46/assets/img/logo_large.png differ diff --git a/v1.46/assets/img/logos/airbyte.png b/v1.46/assets/img/logos/airbyte.png new file mode 100644 index 000000000..e51abf099 Binary files /dev/null and b/v1.46/assets/img/logos/airbyte.png differ diff --git a/v1.46/assets/img/logos/airflow.png b/v1.46/assets/img/logos/airflow.png new file mode 100644 index 000000000..8a20330a4 Binary files /dev/null and b/v1.46/assets/img/logos/airflow.png differ diff --git a/v1.46/assets/img/logos/apache_hive.png b/v1.46/assets/img/logos/apache_hive.png new file mode 100644 index 000000000..f34db575d Binary files /dev/null and b/v1.46/assets/img/logos/apache_hive.png differ diff --git a/v1.46/assets/img/logos/apache_iceberg.png b/v1.46/assets/img/logos/apache_iceberg.png new file mode 100644 index 000000000..6a51b96bb Binary files /dev/null and b/v1.46/assets/img/logos/apache_iceberg.png differ diff --git a/v1.46/assets/img/logos/apache_kafka.png b/v1.46/assets/img/logos/apache_kafka.png new file mode 100644 index 000000000..ad54e4d03 Binary files /dev/null and b/v1.46/assets/img/logos/apache_kafka.png differ diff --git a/v1.46/assets/img/logos/apache_spark.png b/v1.46/assets/img/logos/apache_spark.png new file mode 100644 index 000000000..71436b467 Binary files /dev/null and b/v1.46/assets/img/logos/apache_spark.png differ diff --git a/v1.46/assets/img/logos/athena.png b/v1.46/assets/img/logos/athena.png new file mode 100644 index 000000000..5525cda55 Binary files /dev/null and b/v1.46/assets/img/logos/athena.png differ diff --git a/v1.46/assets/img/logos/cli.png b/v1.46/assets/img/logos/cli.png new file mode 100644 index 000000000..bb01402b4 Binary files /dev/null and b/v1.46/assets/img/logos/cli.png differ diff --git a/v1.46/assets/img/logos/cloudera.png b/v1.46/assets/img/logos/cloudera.png new file mode 100644 index 000000000..43d5a43dc Binary files /dev/null and b/v1.46/assets/img/logos/cloudera.png differ diff --git a/v1.46/assets/img/logos/databricks.png b/v1.46/assets/img/logos/databricks.png new file mode 100644 index 000000000..a2d91399e Binary files /dev/null and b/v1.46/assets/img/logos/databricks.png differ diff --git a/v1.46/assets/img/logos/dbt.png b/v1.46/assets/img/logos/dbt.png new file mode 100644 index 000000000..b57bf3734 Binary files /dev/null and b/v1.46/assets/img/logos/dbt.png differ diff --git a/v1.46/assets/img/logos/delta-lake.png b/v1.46/assets/img/logos/delta-lake.png new file mode 100644 index 000000000..fd88bd136 Binary files /dev/null and b/v1.46/assets/img/logos/delta-lake.png differ diff --git a/v1.46/assets/img/logos/dremio.png b/v1.46/assets/img/logos/dremio.png new file mode 100644 index 000000000..4dca2f4b7 Binary files /dev/null and b/v1.46/assets/img/logos/dremio.png differ diff --git a/v1.46/assets/img/logos/duckdb.png b/v1.46/assets/img/logos/duckdb.png new file mode 100644 index 000000000..6521b586d Binary files /dev/null and b/v1.46/assets/img/logos/duckdb.png differ diff --git a/v1.46/assets/img/logos/git.png b/v1.46/assets/img/logos/git.png new file mode 100644 index 000000000..51f4ae540 Binary files /dev/null and b/v1.46/assets/img/logos/git.png differ diff --git a/v1.46/assets/img/logos/glue.png b/v1.46/assets/img/logos/glue.png new file mode 100644 index 000000000..a74437323 Binary files /dev/null and b/v1.46/assets/img/logos/glue.png differ diff --git a/v1.46/assets/img/logos/huggingface.png b/v1.46/assets/img/logos/huggingface.png new file mode 100644 index 000000000..49e2841dd Binary files /dev/null and b/v1.46/assets/img/logos/huggingface.png differ diff --git a/v1.46/assets/img/logos/kubeflow.png b/v1.46/assets/img/logos/kubeflow.png new file mode 100644 index 000000000..b0d8f97df Binary files /dev/null and b/v1.46/assets/img/logos/kubeflow.png differ diff --git a/v1.46/assets/img/logos/python.png b/v1.46/assets/img/logos/python.png new file mode 100644 index 000000000..14997956a Binary files /dev/null and b/v1.46/assets/img/logos/python.png differ diff --git a/v1.46/assets/img/logos/r.png b/v1.46/assets/img/logos/r.png new file mode 100644 index 000000000..339de0674 Binary files /dev/null and b/v1.46/assets/img/logos/r.png differ diff --git a/v1.46/assets/img/logos/red_hat_openshift_ai.png b/v1.46/assets/img/logos/red_hat_openshift_ai.png new file mode 100644 index 000000000..c3df4c435 Binary files /dev/null and b/v1.46/assets/img/logos/red_hat_openshift_ai.png differ diff --git a/v1.46/assets/img/logos/sagemaker.png b/v1.46/assets/img/logos/sagemaker.png new file mode 100644 index 000000000..06998e349 Binary files /dev/null and b/v1.46/assets/img/logos/sagemaker.png differ diff --git a/v1.46/assets/img/logos/trino_presto.png b/v1.46/assets/img/logos/trino_presto.png new file mode 100644 index 000000000..b00bbeeef Binary files /dev/null and b/v1.46/assets/img/logos/trino_presto.png differ diff --git a/v1.46/assets/img/logos/vertex_ai.png b/v1.46/assets/img/logos/vertex_ai.png new file mode 100644 index 000000000..edd60e5fc Binary files /dev/null and b/v1.46/assets/img/logos/vertex_ai.png differ diff --git a/v1.46/assets/img/metastore-S3.svg b/v1.46/assets/img/metastore-S3.svg new file mode 100644 index 000000000..e84d6a29c --- /dev/null +++ b/v1.46/assets/img/metastore-S3.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/v1.46/assets/img/metastore-lakefs-dev.svg b/v1.46/assets/img/metastore-lakefs-dev.svg new file mode 100644 index 000000000..fc24a78d6 --- /dev/null +++ b/v1.46/assets/img/metastore-lakefs-dev.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/v1.46/assets/img/metastore-lakefs.svg b/v1.46/assets/img/metastore-lakefs.svg new file mode 100644 index 000000000..82b2e0e4a --- /dev/null +++ b/v1.46/assets/img/metastore-lakefs.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/v1.46/assets/img/mirroring/arch.png b/v1.46/assets/img/mirroring/arch.png new file mode 100644 index 000000000..dc6af51ef Binary files /dev/null and b/v1.46/assets/img/mirroring/arch.png differ diff --git a/v1.46/assets/img/object_added.png b/v1.46/assets/img/object_added.png new file mode 100644 index 000000000..6db24985e Binary files /dev/null and b/v1.46/assets/img/object_added.png differ diff --git a/v1.46/assets/img/open_core.png b/v1.46/assets/img/open_core.png new file mode 100644 index 000000000..a9097a6d9 Binary files /dev/null and b/v1.46/assets/img/open_core.png differ diff --git a/v1.46/assets/img/pipeline_branching_model.png b/v1.46/assets/img/pipeline_branching_model.png new file mode 100644 index 000000000..3e94918cb Binary files /dev/null and b/v1.46/assets/img/pipeline_branching_model.png differ diff --git a/v1.46/assets/img/promotion_workflow.png b/v1.46/assets/img/promotion_workflow.png new file mode 100644 index 000000000..46ae5a4a4 Binary files /dev/null and b/v1.46/assets/img/promotion_workflow.png differ diff --git a/v1.46/assets/img/pull-request-merged.png b/v1.46/assets/img/pull-request-merged.png new file mode 100644 index 000000000..7308e479f Binary files /dev/null and b/v1.46/assets/img/pull-request-merged.png differ diff --git a/v1.46/assets/img/pull-request-open.png b/v1.46/assets/img/pull-request-open.png new file mode 100644 index 000000000..4a098d476 Binary files /dev/null and b/v1.46/assets/img/pull-request-open.png differ diff --git a/v1.46/assets/img/pull-request-review.png b/v1.46/assets/img/pull-request-review.png new file mode 100644 index 000000000..847fd9c70 Binary files /dev/null and b/v1.46/assets/img/pull-request-review.png differ diff --git a/v1.46/assets/img/quickstart/axolotl.png b/v1.46/assets/img/quickstart/axolotl.png new file mode 100644 index 000000000..d8265aca4 Binary files /dev/null and b/v1.46/assets/img/quickstart/axolotl.png differ diff --git a/v1.46/assets/img/quickstart/create-quickstart-repo.png b/v1.46/assets/img/quickstart/create-quickstart-repo.png new file mode 100644 index 000000000..c9a40dea1 Binary files /dev/null and b/v1.46/assets/img/quickstart/create-quickstart-repo.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-editor-02.png b/v1.46/assets/img/quickstart/duckdb-editor-02.png new file mode 100644 index 000000000..a726c8f58 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-editor-02.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-editor-03.png b/v1.46/assets/img/quickstart/duckdb-editor-03.png new file mode 100644 index 000000000..101a01557 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-editor-03.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-editor-04.png b/v1.46/assets/img/quickstart/duckdb-editor-04.png new file mode 100644 index 000000000..c3a2949d6 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-editor-04.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-editor-05.png b/v1.46/assets/img/quickstart/duckdb-editor-05.png new file mode 100644 index 000000000..a323ec70c Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-editor-05.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-editor-06.png b/v1.46/assets/img/quickstart/duckdb-editor-06.png new file mode 100644 index 000000000..7911ca146 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-editor-06.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-main-01.png b/v1.46/assets/img/quickstart/duckdb-main-01.png new file mode 100644 index 000000000..78b269785 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-main-01.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-main-02.png b/v1.46/assets/img/quickstart/duckdb-main-02.png new file mode 100644 index 000000000..2f853d8c1 Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-main-02.png differ diff --git a/v1.46/assets/img/quickstart/duckdb-main-03.png b/v1.46/assets/img/quickstart/duckdb-main-03.png new file mode 100644 index 000000000..ef220f55a Binary files /dev/null and b/v1.46/assets/img/quickstart/duckdb-main-03.png differ diff --git a/v1.46/assets/img/quickstart/empty-repo-list.png b/v1.46/assets/img/quickstart/empty-repo-list.png new file mode 100644 index 000000000..762bd6b2d Binary files /dev/null and b/v1.46/assets/img/quickstart/empty-repo-list.png differ diff --git a/v1.46/assets/img/quickstart/hooks-00.png b/v1.46/assets/img/quickstart/hooks-00.png new file mode 100644 index 000000000..a7ceefaf8 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-00.png differ diff --git a/v1.46/assets/img/quickstart/hooks-01.png b/v1.46/assets/img/quickstart/hooks-01.png new file mode 100644 index 000000000..831e2b4f9 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-01.png differ diff --git a/v1.46/assets/img/quickstart/hooks-02.png b/v1.46/assets/img/quickstart/hooks-02.png new file mode 100644 index 000000000..dbe505036 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-02.png differ diff --git a/v1.46/assets/img/quickstart/hooks-03.png b/v1.46/assets/img/quickstart/hooks-03.png new file mode 100644 index 000000000..1d42e5bea Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-03.png differ diff --git a/v1.46/assets/img/quickstart/hooks-04.png b/v1.46/assets/img/quickstart/hooks-04.png new file mode 100644 index 000000000..7e43657b2 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-04.png differ diff --git a/v1.46/assets/img/quickstart/hooks-05.png b/v1.46/assets/img/quickstart/hooks-05.png new file mode 100644 index 000000000..a7db42737 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-05.png differ diff --git a/v1.46/assets/img/quickstart/hooks-06.png b/v1.46/assets/img/quickstart/hooks-06.png new file mode 100644 index 000000000..e9ddfacbc Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-06.png differ diff --git a/v1.46/assets/img/quickstart/hooks-07.png b/v1.46/assets/img/quickstart/hooks-07.png new file mode 100644 index 000000000..ed64375a5 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-07.png differ diff --git a/v1.46/assets/img/quickstart/hooks-08.png b/v1.46/assets/img/quickstart/hooks-08.png new file mode 100644 index 000000000..ae3fb52f0 Binary files /dev/null and b/v1.46/assets/img/quickstart/hooks-08.png differ diff --git a/v1.46/assets/img/quickstart/lakectl-local-01.png b/v1.46/assets/img/quickstart/lakectl-local-01.png new file mode 100644 index 000000000..26606f25d Binary files /dev/null and b/v1.46/assets/img/quickstart/lakectl-local-01.png differ diff --git a/v1.46/assets/img/quickstart/lakectl-local-02.png b/v1.46/assets/img/quickstart/lakectl-local-02.png new file mode 100644 index 000000000..c562be450 Binary files /dev/null and b/v1.46/assets/img/quickstart/lakectl-local-02.png differ diff --git a/v1.46/assets/img/quickstart/lakefs-login-screen.png b/v1.46/assets/img/quickstart/lakefs-login-screen.png new file mode 100644 index 000000000..06262989c Binary files /dev/null and b/v1.46/assets/img/quickstart/lakefs-login-screen.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-repo.png b/v1.46/assets/img/quickstart/quickstart-repo.png new file mode 100644 index 000000000..e25c5e180 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-repo.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-01.png b/v1.46/assets/img/quickstart/quickstart-step-01.png new file mode 100644 index 000000000..1e8cfb944 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-01.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-02.png b/v1.46/assets/img/quickstart/quickstart-step-02.png new file mode 100644 index 000000000..31c44b173 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-02.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-03.png b/v1.46/assets/img/quickstart/quickstart-step-03.png new file mode 100644 index 000000000..d48e82b52 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-03.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-04.png b/v1.46/assets/img/quickstart/quickstart-step-04.png new file mode 100644 index 000000000..aeb51993e Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-04.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-05.png b/v1.46/assets/img/quickstart/quickstart-step-05.png new file mode 100644 index 000000000..6953cb96e Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-05.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-06.png b/v1.46/assets/img/quickstart/quickstart-step-06.png new file mode 100644 index 000000000..769f598e7 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-06.png differ diff --git a/v1.46/assets/img/quickstart/quickstart-step-07.png b/v1.46/assets/img/quickstart/quickstart-step-07.png new file mode 100644 index 000000000..4411ac026 Binary files /dev/null and b/v1.46/assets/img/quickstart/quickstart-step-07.png differ diff --git a/v1.46/assets/img/quickstart/repo-contents.png b/v1.46/assets/img/quickstart/repo-contents.png new file mode 100644 index 000000000..c40006d3d Binary files /dev/null and b/v1.46/assets/img/quickstart/repo-contents.png differ diff --git a/v1.46/assets/img/quickstart/repo-list.png b/v1.46/assets/img/quickstart/repo-list.png new file mode 100644 index 000000000..43ef606b3 Binary files /dev/null and b/v1.46/assets/img/quickstart/repo-list.png differ diff --git a/v1.46/assets/img/quickstart/user_config.png b/v1.46/assets/img/quickstart/user_config.png new file mode 100644 index 000000000..0645eb07d Binary files /dev/null and b/v1.46/assets/img/quickstart/user_config.png differ diff --git a/v1.46/assets/img/rds_conn.png b/v1.46/assets/img/rds_conn.png new file mode 100644 index 000000000..cb96a3ebc Binary files /dev/null and b/v1.46/assets/img/rds_conn.png differ diff --git a/v1.46/assets/img/red-hat/OpenShiftAIDemoArchitecture.png b/v1.46/assets/img/red-hat/OpenShiftAIDemoArchitecture.png new file mode 100644 index 000000000..946c5aaba Binary files /dev/null and b/v1.46/assets/img/red-hat/OpenShiftAIDemoArchitecture.png differ diff --git a/v1.46/assets/img/red-hat/OpenShiftDeploymentArchitecture.png b/v1.46/assets/img/red-hat/OpenShiftDeploymentArchitecture.png new file mode 100644 index 000000000..c1ae3a5ba Binary files /dev/null and b/v1.46/assets/img/red-hat/OpenShiftDeploymentArchitecture.png differ diff --git a/v1.46/assets/img/reference_arch1.png b/v1.46/assets/img/reference_arch1.png new file mode 100644 index 000000000..224450ca6 Binary files /dev/null and b/v1.46/assets/img/reference_arch1.png differ diff --git a/v1.46/assets/img/reference_arch2.png b/v1.46/assets/img/reference_arch2.png new file mode 100644 index 000000000..00a910b2d Binary files /dev/null and b/v1.46/assets/img/reference_arch2.png differ diff --git a/v1.46/assets/img/repo-created.png b/v1.46/assets/img/repo-created.png new file mode 100644 index 000000000..bfe6de10a Binary files /dev/null and b/v1.46/assets/img/repo-created.png differ diff --git a/v1.46/assets/img/repo_create.png b/v1.46/assets/img/repo_create.png new file mode 100644 index 000000000..55d315e4a Binary files /dev/null and b/v1.46/assets/img/repo_create.png differ diff --git a/v1.46/assets/img/reproduce-commit-history.png b/v1.46/assets/img/reproduce-commit-history.png new file mode 100644 index 000000000..61dddf0f7 Binary files /dev/null and b/v1.46/assets/img/reproduce-commit-history.png differ diff --git a/v1.46/assets/img/rollback-commit-history.png b/v1.46/assets/img/rollback-commit-history.png new file mode 100644 index 000000000..e27c7c682 Binary files /dev/null and b/v1.46/assets/img/rollback-commit-history.png differ diff --git a/v1.46/assets/img/rollback-copy-id.png b/v1.46/assets/img/rollback-copy-id.png new file mode 100644 index 000000000..bc416ec17 Binary files /dev/null and b/v1.46/assets/img/rollback-copy-id.png differ diff --git a/v1.46/assets/img/rollback-revert-commit.png b/v1.46/assets/img/rollback-revert-commit.png new file mode 100644 index 000000000..153d3ccb5 Binary files /dev/null and b/v1.46/assets/img/rollback-revert-commit.png differ diff --git a/v1.46/assets/img/route53.png b/v1.46/assets/img/route53.png new file mode 100644 index 000000000..702c3fdc8 Binary files /dev/null and b/v1.46/assets/img/route53.png differ diff --git a/v1.46/assets/img/s3_branch.png b/v1.46/assets/img/s3_branch.png new file mode 100644 index 000000000..1a85a7f36 Binary files /dev/null and b/v1.46/assets/img/s3_branch.png differ diff --git a/v1.46/assets/img/s3gatewayvsclientdataflow.png b/v1.46/assets/img/s3gatewayvsclientdataflow.png new file mode 100644 index 000000000..9431c18cc Binary files /dev/null and b/v1.46/assets/img/s3gatewayvsclientdataflow.png differ diff --git a/v1.46/assets/img/scim/entra-id-provisioning-setup.png b/v1.46/assets/img/scim/entra-id-provisioning-setup.png new file mode 100644 index 000000000..06268da31 Binary files /dev/null and b/v1.46/assets/img/scim/entra-id-provisioning-setup.png differ diff --git a/v1.46/assets/img/scim/lakefs_cloud_scim_configuration.png b/v1.46/assets/img/scim/lakefs_cloud_scim_configuration.png new file mode 100644 index 000000000..39f97b863 Binary files /dev/null and b/v1.46/assets/img/scim/lakefs_cloud_scim_configuration.png differ diff --git a/v1.46/assets/img/scim/lakefs_cloud_scim_settings.png b/v1.46/assets/img/scim/lakefs_cloud_scim_settings.png new file mode 100644 index 000000000..abd6a3167 Binary files /dev/null and b/v1.46/assets/img/scim/lakefs_cloud_scim_settings.png differ diff --git a/v1.46/assets/img/setup.png b/v1.46/assets/img/setup.png new file mode 100644 index 000000000..725a2c992 Binary files /dev/null and b/v1.46/assets/img/setup.png differ diff --git a/v1.46/assets/img/setup_done.png b/v1.46/assets/img/setup_done.png new file mode 100644 index 000000000..db7b8c661 Binary files /dev/null and b/v1.46/assets/img/setup_done.png differ diff --git a/v1.46/assets/img/stream_branching_model.png b/v1.46/assets/img/stream_branching_model.png new file mode 100644 index 000000000..efe5200cd Binary files /dev/null and b/v1.46/assets/img/stream_branching_model.png differ diff --git a/v1.46/assets/img/svix_play.png b/v1.46/assets/img/svix_play.png new file mode 100644 index 000000000..d60fecb7f Binary files /dev/null and b/v1.46/assets/img/svix_play.png differ diff --git a/v1.46/assets/img/unity-delta-sharing-create-catalog.png b/v1.46/assets/img/unity-delta-sharing-create-catalog.png new file mode 100644 index 000000000..b3af5a2ac Binary files /dev/null and b/v1.46/assets/img/unity-delta-sharing-create-catalog.png differ diff --git a/v1.46/assets/img/unity-delta-sharing-provider.png b/v1.46/assets/img/unity-delta-sharing-provider.png new file mode 100644 index 000000000..a9f4d9383 Binary files /dev/null and b/v1.46/assets/img/unity-delta-sharing-provider.png differ diff --git a/v1.46/assets/img/unity-delta-sharing-schema-per-branch.png b/v1.46/assets/img/unity-delta-sharing-schema-per-branch.png new file mode 100644 index 000000000..66d2533e2 Binary files /dev/null and b/v1.46/assets/img/unity-delta-sharing-schema-per-branch.png differ diff --git a/v1.46/assets/img/unity-delta-sharing-shares.png b/v1.46/assets/img/unity-delta-sharing-shares.png new file mode 100644 index 000000000..76afd5c1d Binary files /dev/null and b/v1.46/assets/img/unity-delta-sharing-shares.png differ diff --git a/v1.46/assets/img/unity_export_hook_result_log.png b/v1.46/assets/img/unity_export_hook_result_log.png new file mode 100644 index 000000000..c954195ff Binary files /dev/null and b/v1.46/assets/img/unity_export_hook_result_log.png differ diff --git a/v1.46/assets/img/unity_exported_table_columns.png b/v1.46/assets/img/unity_exported_table_columns.png new file mode 100644 index 000000000..c31356695 Binary files /dev/null and b/v1.46/assets/img/unity_exported_table_columns.png differ diff --git a/v1.46/assets/img/user_branching_model.png b/v1.46/assets/img/user_branching_model.png new file mode 100644 index 000000000..01cf4ff9e Binary files /dev/null and b/v1.46/assets/img/user_branching_model.png differ diff --git a/v1.46/assets/img/waving-axolotl-transparent.gif b/v1.46/assets/img/waving-axolotl-transparent.gif new file mode 100644 index 000000000..2e225ad2d Binary files /dev/null and b/v1.46/assets/img/waving-axolotl-transparent.gif differ diff --git a/v1.46/assets/img/wrapper.png b/v1.46/assets/img/wrapper.png new file mode 100644 index 000000000..2ff764da0 Binary files /dev/null and b/v1.46/assets/img/wrapper.png differ diff --git a/v1.46/assets/js/cookieconsent-init.js b/v1.46/assets/js/cookieconsent-init.js new file mode 100644 index 000000000..abfe64944 --- /dev/null +++ b/v1.46/assets/js/cookieconsent-init.js @@ -0,0 +1,80 @@ +import "https://cdn.jsdelivr.net/gh/orestbida/cookieconsent@v3.0.0/dist/cookieconsent.umd.js"; + +CookieConsent.run({ + guiOptions: { + consentModal: { + layout: "box inline", + position: "middle center", + equalWeightButtons: false, + }, + preferencesModal: { + layout: "box", + position: "middle center", + equalWeightButtons: false, + flipButtons: false, + }, + }, + autoShow: true, + manageScriptTags: true, + revision: 1, + autoClearCookies: true, + disablePageInteraction: true, + categories: { + necessary: { + enabled: true, + readOnly: true, + }, + analytics: { + enabled: true, + readOnly: false, + }, + }, + language: { + default: "en", + translations: { + en: { + consentModal: { + title: + '
Hello fellow axolotl, it\'s cookie time!
', + description: + 'Our website uses essential cookies to ensure its proper operation and tracking cookies to understand how you interact with it. The latter will be set only after consent. Please see our privacy policy.', + acceptAllBtn: "Accept all", + showPreferencesBtn: "Settings", + revisionMessage: + "

Dear user, terms and conditions have changed since the last time you visited!", + }, + preferencesModal: { + title: "Cookie settings", + acceptAllBtn: "Accept all", + acceptNecessaryBtn: "Reject all", + savePreferencesBtn: "Save current selection", + closeIconLabel: "Close", + sections: [ + { + title: "Cookie usage", + description: + 'Our website uses essential cookies to ensure its proper operation and tracking cookies to understand how you interact with it. The latter will be set only after consent. Please see our privacy policy.', + }, + { + title: "Strictly necessary cookies", + description: + "These cookies are strictly necessary for the website to function. They are usually set to handle only your actions in response to a service request, such as setting your privacy preferences, navigating between pages, and setting your preferred version. You can set your browser to block these cookies or to alert you to their presence, but some parts of the website will not function without them. These cookies do not store any personally identifiable information.", + linkedCategory: "necessary", + }, + { + title: "Analytics & Performance cookies", + description: + "These cookies are used for analytics and performance metering purposes. They are used to collect information about how visitors use our website, which helps us improve it over time. They do not collect any information that identifies a visitor. The information collected is aggregated and anonymous.", + linkedCategory: "analytics", + }, + { + title: "More information", + description: + 'For more information about cookie usage, privacy, and how we use the data we collect, please refer to our privacy policy and terms of use.', + }, + ], + }, + }, + }, + }, +}); diff --git a/v1.46/assets/js/copy-code.js b/v1.46/assets/js/copy-code.js new file mode 100644 index 000000000..d64200213 --- /dev/null +++ b/v1.46/assets/js/copy-code.js @@ -0,0 +1,22 @@ +$(() => { + let copyCodeContainer = $("
" + + "
"); + $("div.highlighter-rouge").prepend(copyCodeContainer); + $("div.highlighter-rouge .copy-code-button").click(function() { + const tempTextArea = document.createElement('textarea'); + tempTextArea.textContent = $(this).parent().parent().find("code").text() + document.body.appendChild(tempTextArea); + const selection = document.getSelection(); + selection.removeAllRanges(); + tempTextArea.select(); + document.execCommand('copy'); + selection.removeAllRanges(); + document.body.removeChild(tempTextArea); + $(this).removeClass("fa-copy").removeClass("far").addClass("fa").addClass("fa-check"); + const that = this; + setTimeout(function() { + $(that).addClass("fa-copy").addClass("far").removeClass("fa").removeClass("fa-check"); + }, 300); + }); +}); diff --git a/v1.46/assets/js/feedback.js b/v1.46/assets/js/feedback.js new file mode 100644 index 000000000..b9b1f6e26 --- /dev/null +++ b/v1.46/assets/js/feedback.js @@ -0,0 +1,36 @@ +$(() => { + const close = () => { + $('#is-helpful-ty').dialog('close'); + $(document).off("click", close); + $(window).off("scroll", close); + } + const showThankYou = (elm) => { + $('#is-helpful-ty').dialog({ + width: 200, + position: {of: elm, my: 'top right', at: 'left-50 bottom+50', collision: "fit"}, + open: () => setTimeout(() => { + $(document).on("click", close); + $(window).on("scroll", close); + }, 1), + show: {effect: "drop"}, + hide: {effect: "drop"} + }) + setTimeout(close, 10000); + } + + const sendFeedbackEvent = (elm) => { + if (window.dataLayer === undefined) return; // GA4 is not loaded on this page + const isHelpful = elm.id.substr("page-helpful-".length) === "yes"; + window.dataLayer.push("event", "click_docs_feedback", { + "page_title": window.document.title, + "reaction": isHelpful ? "thumbs_up" : "thumbs_down", + }); + } + + $(".page-helpful-btn").on("click", (e) => { + const elm = e.target; + sendFeedbackEvent(elm); + showThankYou(elm); + }); + $('#is-helpful-ty').on("click", (e) => $(e.target).prop("tagName") === 'A') +}) diff --git a/v1.46/assets/js/just-the-docs.js b/v1.46/assets/js/just-the-docs.js new file mode 100644 index 000000000..5b08d714e --- /dev/null +++ b/v1.46/assets/js/just-the-docs.js @@ -0,0 +1,518 @@ +(function (jtd, undefined) { + +// Event handling + +jtd.addEvent = function(el, type, handler) { + if (el.attachEvent) el.attachEvent('on'+type, handler); else el.addEventListener(type, handler); +} +jtd.removeEvent = function(el, type, handler) { + if (el.detachEvent) el.detachEvent('on'+type, handler); else el.removeEventListener(type, handler); +} +jtd.onReady = function(ready) { + // in case the document is already rendered + if (document.readyState!='loading') ready(); + // modern browsers + else if (document.addEventListener) document.addEventListener('DOMContentLoaded', ready); + // IE <= 8 + else document.attachEvent('onreadystatechange', function(){ + if (document.readyState=='complete') ready(); + }); +} + +// Show/hide mobile menu + +function initNav() { + jtd.addEvent(document, 'click', function(e){ + var target = e.target; + while (target && !(target.classList && target.classList.contains('nav-list-expander'))) { + target = target.parentNode; + } + if (target) { + e.preventDefault(); + const active = target.parentNode.classList.toggle('active'); + const passive = target.parentNode.classList.toggle('passive'); + if (active && passive) target.parentNode.classList.toggle('passive'); + target.ariaPressed = active; + } + }); + + const siteNav = document.getElementById('site-nav'); + const mainHeader = document.getElementById('main-header'); + const menuButton = document.getElementById('menu-button'); + + disableHeadStyleSheet(); + + jtd.addEvent(menuButton, 'click', function(e){ + e.preventDefault(); + + if (menuButton.classList.toggle('nav-open')) { + siteNav.classList.add('nav-open'); + mainHeader.classList.add('nav-open'); + menuButton.ariaPressed = true; + } else { + siteNav.classList.remove('nav-open'); + mainHeader.classList.remove('nav-open'); + menuButton.ariaPressed = false; + } + }); +} + +// The page-specific + + + + + + + + + + + + + + + + + + + +lakeFS Cloud | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS Cloud + + +

+ +

+ + + What is lakeFS cloud? + + +

+ + +

lakeFS Cloud is a single tenant, fully-managed lakeFS solution, providing high availability, auto-scaling, support and production-grade features.

+

+ + + Why did we build lakeFS cloud? + + +

+ + +

We built lakeFS cloud for three main reasons:

+
    +
  1. We wanted to provide organizations with the benefits of lakeFS without the need to manage it, saving them the investment in infrastructure and work related to installation, upgrades, uptime and scale .
  2. +
  3. We wanted to provide lakeFS cloud users with security that meets their needs, with SSO, SCIM, and RBAC.
  4. +
  5. We wanted to provide additional functionality that reduces friction and allows fast implementation of version controlled data/ML/AI pipelines throughout their data lifecycle.
  6. +
+

+ + + What is the value of using lakeFS Cloud as a managed service? + + +

+ + +

The main advantages of using lakeFS Cloud, the lakeFS managed service are:

+
    +
  1. No installation required, no cloud costs and devops efforts on installing and maintaining a lakeFS installation.
  2. +
  3. All lakeFS services are managed and run by us, including Managed Garbage Collection.
  4. +
  5. lakeFS cloud is highly available and includes a commitment to an uptime SLA.
  6. +
  7. lakeFS cloud auto scales according to your needs. See lakeFS cloud scalability model for more details.
  8. +
  9. Upgrades are done transparently on your lakeFS cloud environment
  10. +
  11. The lakeFS team is committed to supporting you with an SLA for both issues and product enhancements.
  12. +
+

+ + + Which security features does lakeFS Cloud provide? + + +

+ + +
    +
  1. lakeFS Cloud is SOC2 Type II compliant. Contact us to get the certification.
  2. +
  3. lakeFS Cloud version controls your data, without accessing it, using pre-signed URLs! Read more here.
  4. +
  5. When using lakeFS Cloud, you are provided with a rich Role-Based Access Control functionality that allows for fine-grained control by associating permissions with users and groups, granting them specific actions on specific resources. This ensures data security and compliance within an organization.
  6. +
  7. To easily manage users and groups, lakeFS Cloud provides SSO integration (including support for SAML, OIDC, AD FS, Okta, and Azure AD), supporting existing credentials from a trusted provider, eliminating separate logins.
  8. +
  9. lakeFS Cloud supports SCIM for automatically provisioning and deprovisioning users and group memberships to allow organizations to maintain a single source of truth for their user database.
  10. +
  11. STS Auth offers temporary, secure logins using an Identity Provider, simplifying user access and enhancing security.
  12. +
  13. Authentication with AWS IAM Roles allows authentication using AWS IAM roles instead of lakeFS credentials, removing the need to maintain static credentials for lakeFS Enterprise users running on AWS.
  14. +
  15. Auditing provides a detailed action log of events happening within lakeFS, including who performed which action, on which resource - and when.
  16. +
  17. Private-Link support to ensure network security by only allowing access to your lakeFS Cloud installation from your cloud accounts
  18. +
+

+ + + What additional functionality does lakeFS Cloud provide? + + +

+ + +

Using lakeFS cloud is not just a secure and managed way of using lakeFS OSS; it is much more than that. With lakeFS Cloud you enjoy:

+
    +
  1. lakeFS Mount allows users to virtually mount a remote lakeFS repository onto a local directory. Once mounted, users can access the data as if it resides on their local filesystem, using any tool, library, or framework that reads from a local filesystem.
  2. +
  3. lakeFS Metadata Search - Allows a granular search API to filter and query versioned objects based on attached metadata. This is especially useful for machine learning environments to filter by labels and file attributes
  4. +
  5. lakeFS for Databricks - Provides a turnkey solution for Databricks customers for analytics, machine learning and business intelligence use cases including full support for Delta Lake tables, Unity Catalog, MLFlow and the rest of the Databricks product suite.
  6. +
  7. lakeFS for Snowflake - Provides full integration into the Snowflake ecosystem, including full support for Iceberg managed tables.
  8. +
  9. lakeFS Cross Cloud - Allows central management of repositories that span across multiple cloud providers including Azure, AWS, GCP and on-prem environments.
  10. +
  11. Transactional Mirroring - allows replicating lakeFS repositories into consistent read-only copies in remote locations.
  12. +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
FeatureOSSCloud
Format-agnostic data version control
Cloud-agnostic
Zero Clone copy for isolated environment
Atomic Data Promotion (via merges)
Data stays in one place
Configurable Garbage Collection
Data CI/CD using lakeFS hooks
Integrates with your data stack
Role Based Access Control (RBAC) 
Single Sign On (SSO) 
SCIM Support 
IAM Roles 
Mount Capability 
Audit Logs 
Transactional Mirroring (cross-region) 
Managed Service (auto updates, scaling) 
Managed Garbage Collection 
SOC2 Compliant 
Support SLA 
+

+ + + How lakeFS Cloud interacts with your infrastructure + + +

+ + +

Treeverse hosts and manages a dedicated lakeFS instance that interfaces with data held in your object store, such as S3.

+ +
flowchart TD
+    U[Users] --> LFC
+
+    subgraph Your Infrastructure
+    IAMM[lakeFS Managed GC IAM Role] --> ObjectStore[Client's Object Store]
+    IAMA[lakeFS Application IAM Role] --> ObjectStore
+    end
+
+    subgraph Treeverse's Infrastructure
+    MGC[Managed Garbage Collection] --> EMR[Elastic Map Reduce]
+    EMR --> IAMM
+    MGC --> CP
+    CP[Control Plane]
+    LFC --> CP
+
+        subgraph Client's Tenant
+        LFC[lakeFS Cloud] --> DB[Refstore Database]
+        end
+
+    LFC --> IAMC[lakeFS Connector IAM Role]
+    IAMC -->|ExternalID| IAMA
+    end
+
+

+ + + Setting up lakeFS Cloud + + +

+ +

+ + + AWS / Azure + + +

+ + +

Please follow the self-service setup wizard on lakeFS Cloud.

+

+ + + GCP + + +

+ + +

Please contact us for onboarding instructions.

+

+ + + lakeFS Cloud Scalability Model + + +

+ + +

By default, a lakeFS Cloud installation supports:

+
    +
  • 1,500 read operations/second across all branches on all repositories within a region
  • +
  • 1,500 write operations per second across all branches on all repositories within a region
  • +
+ +

This limit can be increased by contacting support.

+ +

Each lakeFS branch can sustain up to a maximum of 1,000 write operations/second and 3,000 read operations per second. +This scales horizontally, so for example, with 10 concurrent branches, a repository could sustain 10k writes/second and 30k reads/second, assuming load is distributed evenly between them.

+ +

Reading committed data (e.g. from a commit ID or tag) could be scaled up horizontally to any desired capacity, and defaults to ~5,000 reads/second.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/cloud/managed-gc.html b/v1.46/cloud/managed-gc.html new file mode 100644 index 000000000..ad6fbb0a9 --- /dev/null +++ b/v1.46/cloud/managed-gc.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/private-link.html b/v1.46/cloud/private-link.html new file mode 100644 index 000000000..bc047fd43 --- /dev/null +++ b/v1.46/cloud/private-link.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/scim.html b/v1.46/cloud/scim.html new file mode 100644 index 000000000..1d45bb316 --- /dev/null +++ b/v1.46/cloud/scim.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/sso.html b/v1.46/cloud/sso.html new file mode 100644 index 000000000..52716caca --- /dev/null +++ b/v1.46/cloud/sso.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/standalone-gc.html b/v1.46/cloud/standalone-gc.html new file mode 100644 index 000000000..60636debe --- /dev/null +++ b/v1.46/cloud/standalone-gc.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/unity-delta-sharing-m0-users.html b/v1.46/cloud/unity-delta-sharing-m0-users.html new file mode 100644 index 000000000..e8e7839a6 --- /dev/null +++ b/v1.46/cloud/unity-delta-sharing-m0-users.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/cloud/unity-delta-sharing.html b/v1.46/cloud/unity-delta-sharing.html new file mode 100644 index 000000000..e8e7839a6 --- /dev/null +++ b/v1.46/cloud/unity-delta-sharing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/commitment.html b/v1.46/commitment.html new file mode 100644 index 000000000..41dce22e2 --- /dev/null +++ b/v1.46/commitment.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/contributing.html b/v1.46/contributing.html new file mode 100644 index 000000000..f88e239f5 --- /dev/null +++ b/v1.46/contributing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/data_lifecycle_management/ci.html b/v1.46/data_lifecycle_management/ci.html new file mode 100644 index 000000000..2971ba3d0 --- /dev/null +++ b/v1.46/data_lifecycle_management/ci.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/data_lifecycle_management/data-devenv.html b/v1.46/data_lifecycle_management/data-devenv.html new file mode 100644 index 000000000..66f7a21a2 --- /dev/null +++ b/v1.46/data_lifecycle_management/data-devenv.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/data_lifecycle_management/production.html b/v1.46/data_lifecycle_management/production.html new file mode 100644 index 000000000..8896605e8 --- /dev/null +++ b/v1.46/data_lifecycle_management/production.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/aws.html b/v1.46/deploy/aws.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/deploy/aws.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/azure.html b/v1.46/deploy/azure.html new file mode 100644 index 000000000..3ab302607 --- /dev/null +++ b/v1.46/deploy/azure.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/docker.html b/v1.46/deploy/docker.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/deploy/docker.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/gcp.html b/v1.46/deploy/gcp.html new file mode 100644 index 000000000..e0a39904c --- /dev/null +++ b/v1.46/deploy/gcp.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/index.html b/v1.46/deploy/index.html new file mode 100644 index 000000000..da83e1cb5 --- /dev/null +++ b/v1.46/deploy/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/k8s.html b/v1.46/deploy/k8s.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/deploy/k8s.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploy/onprem.html b/v1.46/deploy/onprem.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/deploy/onprem.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/db.html b/v1.46/deploying-aws/db.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/deploying-aws/db.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/index.html b/v1.46/deploying-aws/index.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/deploying-aws/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/install.html b/v1.46/deploying-aws/install.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/deploying-aws/install.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/lb_dns.html b/v1.46/deploying-aws/lb_dns.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/deploying-aws/lb_dns.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/monitor.md b/v1.46/deploying-aws/monitor.md new file mode 100644 index 000000000..2abb5887c --- /dev/null +++ b/v1.46/deploying-aws/monitor.md @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/offboarding.html b/v1.46/deploying-aws/offboarding.html new file mode 100644 index 000000000..ad6a95fec --- /dev/null +++ b/v1.46/deploying-aws/offboarding.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying-aws/upgrade.html b/v1.46/deploying-aws/upgrade.html new file mode 100644 index 000000000..63831114b --- /dev/null +++ b/v1.46/deploying-aws/upgrade.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/deploying/install.html b/v1.46/deploying/install.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/deploying/install.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/downloads.html b/v1.46/downloads.html new file mode 100644 index 000000000..b03e665ad --- /dev/null +++ b/v1.46/downloads.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/enterprise/architecture.html b/v1.46/enterprise/architecture.html new file mode 100644 index 000000000..903e66759 --- /dev/null +++ b/v1.46/enterprise/architecture.html @@ -0,0 +1,748 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Architecture | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Architecture + + +

+ + +

The lakeFS Enterprise software consists of two components:

+
    +
  1. lakeFS Open Source: treeverse/lakeFS. See Github releases for release info.
  2. +
  3. A proprietary component called Fluffy which includes lakeFS’ Enterprise features.
  4. +
+ +

img.png

+ +

[1] Any user request to lakeFS via Browser or Programmatic access (SDK, HTTP +API, lakectl).

+ +

[2] Reverse Proxy (e.g. NGINX, Traefik, K8S Ingress): will handle user requests +and proxy between lakeFS server and fluffy server based on the path prefix +while maintaining the same host.

+ +

[3] lakeFS server - the main lakeFS service.

+ +

[4] fluffy server - service that is responsible for the Enterprise features., +it is separated by ports for security reasons.

+ +
    +
  1. SSO auth (i.e Browser login via Azure AD, Okta, Auth0), default port 8000.
  2. +
  3. RBAC authorization, default port 9000.
  4. +
+ +

[5] The KV Store - Where metadata is stored used both by lakeFS and fluffy.

+ +

[6] SSO IdP - Identity provider (e.g. Azure AD, Okta, JumpCloud). fluffy +implements SAML and Oauth2 protocols.

+ +

For more details and pricing, please contact sales.

+ +

Note: Setting up lakeFS enterprise with an SSO IdP (OIDC, SAML or LDAP) requires +configuring access from the IdP too.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/configuration.html b/v1.46/enterprise/configuration.html new file mode 100644 index 000000000..2ebfbab64 --- /dev/null +++ b/v1.46/enterprise/configuration.html @@ -0,0 +1,933 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Configuration Reference | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Enterprise Configuration Reference + + +

+ + +

Working with lakeFS Enterprise involve configuring both lakeFS and Fluffy. You can find the extended configuration references for both components below.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. lakeFS Configuration
  2. +
  3. Fluffy Server Configuration
  4. +
+ +
+

+ + + lakeFS Configuration + + +

+ + +

See the full lakeFS Server Configuration

+

+ + + Fluffy Server Configuration + + +

+ + +

Configuring Fluffy using a YAML configuration file and/or environment variables. +The configuration file’s location can be set with the ‘–config’ flag. If not specified, the first file found in the following order will be used:

+
    +
  1. ./config.yaml
  2. +
  3. $HOME/fluffy/config.yaml
  4. +
  5. /etc/fluffy/config.yaml
  6. +
  7. $HOME/.fluffy.yaml
  8. +
+ +

Configuration items can be controlled by environment variables, see below.

+

+ + + Reference + + +

+ + +

This reference uses . to denote the nesting of values.

+ +
    +
  • logging.format (one of ["json", "text"] : "text") - Format to output log message in
  • +
  • logging.level (one of ["TRACE", "DEBUG", "INFO", "WARN", "ERROR", "NONE"] : "INFO") - Logging level to output
  • +
  • +

    logging.audit_log_level (one of ["TRACE", "DEBUG", "INFO", "WARN", "ERROR", "NONE"] : "DEBUG") - Audit logs level to output.

    + +

    Note: In case you configure this field to be lower than the main logger level, you won’t be able to get the audit logs

    +
  • +
  • logging.output (string : "-") - A path or paths to write logs to. A - means the standard output, = means the standard error.
  • +
  • logging.file_max_size_mb (int : 100) - Output file maximum size in megabytes.
  • +
  • logging.files_keep (int : 0) - Number of log files to keep, default is all.
  • +
  • logging.trace_request_headers (bool : false) - If set to true and logging level is set to TRACE, logs request headers.
  • +
  • listen_address (string : "0.0.0.0:8000") - A <host>:<port> structured string representing the address to listen on
  • +
  • database - Configuration section for the Fluffy key-value store database. The database must be shared between lakeFS & Fluffy +
      +
    • database.type (string ["postgres"|"dynamodb"|"cosmosdb"|"local"] : ) - Fluffy database type
    • +
    • database.postgres - Configuration section when using database.type="postgres" +
        +
      • database.postgres.connection_string (string : "postgres://localhost:5432/postgres?sslmode=disable") - PostgreSQL connection string to use
      • +
      • database.postgres.max_open_connections (int : 25) - Maximum number of open connections to the database
      • +
      • database.postgres.max_idle_connections (int : 25) - Maximum number of connections in the idle connection pool
      • +
      • database.postgres.connection_max_lifetime (duration : 5m) - Sets the maximum amount of time a connection may be reused (valid units: ns|us|ms|s|m|h)
      • +
      +
    • +
    • database.dynamodb - Configuration section when using database.type="dynamodb" +
        +
      • database.dynamodb.table_name (string : "kvstore") - Table used to store the data
      • +
      • +

        database.dynamodb.scan_limit (int : 1025) - Maximal number of items per page during scan operation

        + +

        Note: Refer to the following AWS documentation for further information

        +
      • +
      • database.dynamodb.endpoint (string : ) - Endpoint URL for database instance
      • +
      • database.dynamodb.aws_region (string : ) - AWS Region of database instance
      • +
      • database.dynamodb.aws_profile (string : ) - AWS named profile to use
      • +
      • database.dynamodb.aws_access_key_id (string : ) - AWS access key ID
      • +
      • database.dynamodb.aws_secret_access_key (string : ) - AWS secret access key
      • +
      • Note: endpoint aws_region aws_access_key_id aws_secret_access_key are not required and used mainly for experimental purposes when working with DynamoDB with different AWS credentials.
      • +
      • database.dynamodb.health_check_interval (duration : 0s) - Interval to run health check for the DynamoDB instance (won’t run if equal to 0).
      • +
      +
    • +
    • database.cosmosdb - Configuration section when using database.type="cosmosdb" +
        +
      • database.cosmosdb.key (string : "") - If specified, will + be used to authenticate to the CosmosDB account. Otherwise, Azure SDK + default authentication (with env vars) will be used.
      • +
      • database.cosmosdb.endpoint (string : "") - CosmosDB account endpoint, e.g. https://<account>.documents.azure.com/.
      • +
      • database.cosmosdb.database (string : "") - CosmosDB database name.
      • +
      • database.cosmosdb.container (string : "") - CosmosDB container name.
      • +
      • database.cosmosdb.throughput (int32 : ) - CosmosDB container’s RU/s. If not set - the default CosmosDB container throughput is used.
      • +
      • +

        database.cosmosdb.autoscale (bool : false) - If set, CosmosDB container throughput is autoscaled (See CosmosDB docs for minimum throughput requirement). Otherwise, uses “Manual” mode (Docs).

        + +
          +
        • database.local - Configuration section when using database.type="local" +
            +
          • database.local.path (string : "~/fluffy/metadata") - Local path on the filesystem to store embedded KV metadata
          • +
          • database.local.sync_writes (bool: true) - Ensure each write is written to the disk. Disable to increase performance
          • +
          • database.local.prefetch_size (int: 256) - How many items to prefetch when iterating over embedded KV records
          • +
          • database.local.enable_logging (bool: false) - Enable trace logging for local driver
          • +
          +
        • +
        +
      • +
      +
    • +
    +
  • +
  • auth - Configuration section for the Fluffy authentication services, like SAML or OIDC. +
      +
    • auth.encrypt.secret_key (string : required) - Same value given to lakeFS. A random (cryptographically safe) generated string that is used for encryption and HMAC signing
    • +
    • auth.logout_redirect_url (string : "/auth/login") - The address to redirect to after a successful logout, e.g. login.
    • +
    • auth.post_login_redirect_url (string : '') - Required when SAML is enabled. The address to redirect after a successful login. For most common configurations, setting to / will redirect to lakeFS homepage.
    • +
    • auth.serve_listen_address (string : '') - If set, an endpoint serving RBAC requests binds to this address.
    • +
    • auth.serve_disable_authentication (bool : false) - Unsafe. Disables authentication to the RBAC server.
    • +
    • auth.ldap +
        +
      • auth.ldap.server_endpoint (string : required) - The LDAP server address, e.g. ‘ldaps://ldap.company.com:636’
      • +
      • auth.ldap.bind_dn (string : required) - The bind string, e.g. ‘uid=,ou=Users,o=,dc=,dc=com'
      • +
      • auth.ldap.bind_password (string : required) - The password for the user to bind.
      • +
      • auth.ldap.username_attribute (string : required) - The user name attribute, e.g. ‘uid’
      • +
      • auth.ldap.user_base_dn (string : required) - The search request base dn, e.g. ‘ou=Users,o=,dc=,dc=com'
      • +
      • auth.ldap.user_filter (string : required) - The search request user filter, e.g. ‘(objectClass=inetOrgPerson)’
      • +
      • auth.ldap.connection_timeout_seconds (int : required) - The timeout for a single connection
      • +
      • auth.ldap.request_timeout_seconds (int : required) - The timeout for a single request
      • +
      +
    • +
    • auth.saml Configuration section for SAML +
        +
      • auth.saml.enabled (bool : false) - Enables SAML Authentication.
      • +
      • auth.saml.sp_root_url (string : '') - The base lakeFS-URL, e.g. ‘https://'
      • +
      • auth.saml.sp_x509_key_path (string : '') - The path to the private key, e.g ‘/etc/saml_certs/rsa_saml_private.cert’
      • +
      • auth.saml.sp_x509_cert_path (string : '') - The path to the public key, ‘/etc/saml_certs/rsa_saml_public.pem’
      • +
      • auth.saml.sp_sign_request (bool : 'false') SPSignRequest some IdP require the SLO request to be signed
      • +
      • auth.saml.sp_signature_method (string : '') SPSignatureMethod optional valid signature values depending on the IdP configuration, e.g. ‘http://www.w3.org/2001/04/xmldsig-more#rsa-sha256’
      • +
      • auth.saml.idp_metadata_url (string : '') - The URL for the metadata server, e.g. ‘https:///federationmetadata/2007-06/federationmetadata.xml'
      • +
      • auth.saml.idp_skip_verify_tls_cert (bool : false) - Insecure skip verification of the IdP TLS certificate, like when signed by a private CA
      • +
      • auth.saml.idp_authn_name_id_format (string : 'urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified') - The format used in the NameIDPolicy for authentication requests
      • +
      • auth.saml.idp_request_timeout (duration : '10s') The timeout for remote authentication requests.
      • +
      • auth.saml.external_user_id_claim_name (string : '') - The claim name to use as the user identifier with an IdP mostly for logout
      • +
      +
    • +
    • auth.oidc Configuration section for OIDC +
        +
      • auth.oidc.enabled (bool : false) - Enables OIDC Authentication.
      • +
      • auth.oidc.url (string : '') - The OIDC provider url, e.g. ‘https://oidc-provider-url.com/’
      • +
      • auth.oidc.client_id (string : '') - The application’s ID.
      • +
      • auth.oidc.client_secret (string : '') - The application’s secret.
      • +
      • auth.oidc.callback_base_url (string : '') - A default callback address of the Fluffy server.
      • +
      • auth.oidc.callback_base_urls (string[] : '[]')
      • +
      +
    • +
    • Note: You may configure a list of URLs that the OIDC provider may redirect to. This allows lakeFS to be accessed from multiple hostnames while retaining federated auth capabilities. +If the provider redirects to a URL not in this list, the login will fail. This property and callback_base_url are mutually exclusive. +
        +
      • auth.oidc.authorize_endpoint_query_parameters (bool : map[string]string) - key/value parameters that are passed to a provider’s authorization endpoint.
      • +
      • auth.oidc.logout_endpoint_query_parameters (string[] : '[]') - The query parameters that will be used to redirect the user to the OIDC provider after logout, e.g. ‘[returnTo, https:///oidc/login]'
      • +
      • auth.oidc.logout_client_id_query_parameter (string : '') - The claim name that represents the client identifier in the OIDC provider
      • +
      • auth.oidc.additional_scope_claims (string[] : '[]') - Specifies optional requested permissions, other than openid and profile that are being used.
      • +
      +
    • +
    • auth.cache Configuration section for RBAC service cache +
        +
      • auth.cache.enabled (bool : true) - Enables RBAC service cache
      • +
      • auth.cache.size (int : 1024) - Number of users, policies and credentials to cache.
      • +
      • auth.cache.ttl (duration : 20s) - Cache items time to live expiry.
      • +
      • auth.cache.jitter (duration : 3s) - Cache items time to live jitter.
      • +
      +
    • +
    • auth.external - Configuration section for the external authentication methods +
        +
      • auth.external.aws_auth - Configuration section for authenticating to lakeFS using AWS presign get-caller-identity request: External Principals AWS Auth +
          +
        • auth.external.aws_auth.enabled (bool : false) - If true, external principals API will be enabled, e.g auth service and login api’s.
        • +
        • auth.external.aws_auth.get_caller_identity_max_age (duration : 15m) - The maximum age in seconds for the GetCallerIdentity request to be valid, the max is 15 minutes enforced by AWS, smaller TTL can be set.
        • +
        • auth.authentication_api.external_principals_enabled (bool : false) - If true, external principals API will be enabled, e.g auth service and login api’s.
        • +
        • auth.external.aws_auth.valid_sts_hosts ([]string) - The default are all the valid AWS STS hosts (sts.amazonaws.com, sts.us-east-2.amazonaws.com etc).
        • +
        • auth.external.aws_auth.required_headers (map[string]string : ) - Headers that must be present by the client when doing login request (e.g X-LakeFS-Server-ID: <lakefs.ingress.domain>).
        • +
        • auth.external.aws_auth.optional_headers (map[string]string : ) - Optional headers that can be present by the client when doing login request.
        • +
        • auth.external.aws_auth.http_client.timeout (duration : 10s) - The timeout for the HTTP client used to communicate with AWS STS.
        • +
        • auth.external.aws_auth.http_client.skip_verify (bool : false) - Skip SSL verification with AWS STS.
        • +
        +
      • +
      +
    • +
    +
  • +
+

+ + + Using Environment Variables + + +

+ + +

All the configuration variables can be set or overridden using environment variables. +To set an environment variable, prepend FLUFFY_ to its name, convert it to upper case, and replace . with _:

+ +

For example, logging.format becomes FLUFFY_LOGGING_FORMAT, auth.saml.enabled becomes FLUFFY_AUTH_SAML_ENABLED, etc.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/getstarted/index.html b/v1.46/enterprise/getstarted/index.html new file mode 100644 index 000000000..c7799a875 --- /dev/null +++ b/v1.46/enterprise/getstarted/index.html @@ -0,0 +1,738 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Get Started | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Get Started with lakeFS Enterprise + + +

+ +

+ + + Request a Trial License + + +

+ + +

To start a 30-day free trial, please contact us. You will be granted with a token that +allows downloading a Docker image that includes lakeFS Enterprise features.

+

+ + + Install lakeFS Enterprise + + +

+ + +

lakeFS Enterprise offers two installation methods:

+
    +
  • Quickstart: The fastest way to get started with lakeFS Enterprise. Intended for testing and evaluation.
  • +
  • Production deployment: Run lakeFS Enterprise on a Kubernetes cluster on AWS, Azure, GCP, or on-prem.
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/getstarted/install.html b/v1.46/enterprise/getstarted/install.html new file mode 100644 index 000000000..3fdcd71f0 --- /dev/null +++ b/v1.46/enterprise/getstarted/install.html @@ -0,0 +1,1277 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Install | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Install + + +

+ + +

For production deployments of lakeFS Enterprise, follow this guide.

+ + +

+ + + lakeFS Enterprise Architecture + + +

+ + +

We recommend to review the lakeFS Enterprise architecture to understand the components you will be deploying.

+

+ + + Deploy lakeFS Enterprise on Kubernetes + + +

+ + +

The guide is using the lakeFS Helm Chart to deploy a fully functional lakeFS Enterprise.

+ +

The guide includes example configurations, follow the steps below and adjust the example configurations according to:

+
    +
  • The platform you run on: among the platform supported by lakeFS
  • +
  • Type of KV store you use
  • +
  • Your SSO IdP and protocol
  • +
+

+ + + Prerequisites + + +

+ + +
    +
  1. You have a Kubernetes cluster running in one of the platforms supported by lakeFS.
  2. +
  3. Helm is installed
  4. +
  5. Access to download dockerhub/fluffy from Docker Hub. Contact us to gain access to lakeFS Enterprise features.
  6. +
  7. A KV Database that will be shared by lakeFS and Fluffy. The available options are dependent in your deployment platform.
  8. +
  9. A proxy server configured to route traffic between the lakeFS and Fluffy servers, see Reverse Proxy in lakeFS Enterprise architecture.
  10. +
+

+ + + Optional + + +

+ +
    +
  1. Access to configure your SSO IdP supported by lakeFS Enterprise.
  2. +
+ +
+

You can install lakeFS Enterprise without configuring SSO and still benefit from all other lakeFS Enterprise features.

+
+

+ + + Add the lakeFS Helm Chart + + +

+ + +
    +
  • Add the lakeFS Helm repository with helm repo add lakefs https://charts.lakefs.io
  • +
  • The chart contains a values.yaml file you can customize to suit your needs as you follow this guide. Use helm show values lakefs/lakefs to see the default values.
  • +
  • While customizing your values.yaml file, note to configure fluffy.image.privateRegistry.secretToken with the token Docker Hub token you received.
  • +
+

+ + + Authentication Configuration + + +

+ + +

Authentication in lakeFS Enterprise is handled by the Fluffy SSO service which runs side-by-side to lakeFS. This section explains +what Fluffy configurations are required for configuring the SSO service. See this configuration reference for additional Fluffy configurations.

+ +

See SSO for lakeFS Enterprise for the supported identity providers and protocols.

+ +

The examples below include example configuration for each of the supported SSO protocols. Note the IdP-specific details you’ll need to +replace with your IdP details.

+ +
+ +
+ +

The following values file will run lakeFS Enterprise with OIDC integration.

+ +
+

The full OIDC configurations explained here.

+
+ +
lakefsConfig: |
+  logging:
+      level: "INFO"
+  blockstore:
+    type: s3
+  auth:
+    oidc:
+      # the claim that's provided by the OIDC provider (e.g Okta) that will be used as the username according to OIDC provider claims provided after successful authentication
+      friendly_name_claim_name: "<some-oidc-provider-claim-name>"
+      default_initial_groups: ["Developers", "Admins"]
+      # if true then the value of friendly_name_claim_name will be refreshed during each login to maintain the latest value
+      # and the the claim value (i.e user name) will be stored in the lakeFS database
+      persist_friendly_name: true
+    ui_config:
+      login_cookie_names:
+        - internal_auth_session
+        - oidc_auth_session
+ingress:
+  enabled: true
+  ingressClassName: <class-name>
+  hosts:
+    # the ingress that will be created for lakeFS
+    - host: <lakefs.acme.com>
+      paths:
+       - /
+
+##################################################
+########### lakeFS enterprise - FLUFFY ###########
+##################################################
+
+fluffy:
+  enabled: true
+  image:
+    repository: treeverse/fluffy
+    pullPolicy: IfNotPresent
+    privateRegistry:
+      enabled: true
+      secretToken: <dockerhub-token-fluffy-image>
+  fluffyConfig: |
+    logging:
+      format: "json"
+      level: "INFO"
+    auth:
+      logout_redirect_url: https://oidc-provider-url.com/logout/example
+      oidc:
+        enabled: true
+        url: https://oidc-provider-url.com/
+        client_id: <oidc-client-id>
+        callback_base_url: https://<lakefs.acme.com>
+        # the claim name that represents the client identifier in the OIDC provider (e.g Okta)
+        logout_client_id_query_parameter: client_id
+        # the query parameters that will be used to redirect the user to the OIDC provider (e.g Okta) after logout
+        logout_endpoint_query_parameters:
+          - returnTo
+          - https://<lakefs.acme.com>/oidc/login
+  secrets:
+    create: true
+  sso:
+    enabled: true
+    oidc:
+      enabled: true
+      # secret given by the OIDC provider (e.g auth0, Okta, etc) store in kind: Secret
+      client_secret: <oidc-client-secret>
+  rbac:
+    enabled: true
+
+useDevPostgres: true
+
+ +
+
+ +

The following values file will run lakeFS Enterprise with SAML using Azure AD as the IdP. +
+You can use this example configuration to configure Active Directory Federation Services (AD FS) with SAML.

+ +
+

The full SAML configurations explained here.

+
+

+ + + Azure App Configuration + + +

+ + +
    +
  1. Create an Enterprise Application with SAML toolkit - see Azure quickstart
  2. +
  3. Add users: App > Users and groups: Attach users and roles from their existing AD users +list - only attached users will be able to login to lakeFS.
  4. +
  5. Configure SAML: App > Single sign-on > SAML: +
      +
    1. Entity ID: Add 2 ID’s, lakefs-url + lakefs-url/saml/metadata (e.g. https://lakefs.acme.com and https://lakefs.acme.com/saml/metadata)
    2. +
    3. Reply URL: lakefs-url/saml (e.g. https://lakefs.acme.com/saml)
    4. +
    5. Sign on URL: lakefs-url/sso/login-saml (e.g. https://lakefs.acme.com/sso/login-saml)
    6. +
    7. Relay State (Optional, controls where to redirect after login): /
    8. +
    +
  6. +
+

+ + + SAML Configuration + + +

+ + +
    +
  1. Configure SAML application in your IdP (i.e Azure AD) and replace the required parameters into the values.yaml below.
  2. +
  3. To generate certificates keypair use: `openssl req -x509 -newkey rsa:2048 -keyout myservice.key -out myservice.cert -days 365 -nodes -subj “/CN=lakefs.acme.com” -
  4. +
+ +
secrets:
+  authEncryptSecretKey: "some random secret string"
+
+lakefsConfig: |
+  logging:
+      level: "DEBUG"
+  blockstore:
+    type: local
+  auth:
+    cookie_auth_verification:
+      # claim name to use for friendly name in lakeFS UI
+      friendly_name_claim_name: displayName
+      external_user_id_claim_name: http://schemas.xmlsoap.org/ws/2005/05/identity/claims/name
+      default_initial_groups:
+        - "Developers"
+    encrypt:
+      secret_key: shared-secrey-key
+    ui_config:
+      login_cookie_names:
+        - internal_auth_session
+        - saml_auth_session
+ingress:
+  enabled: true
+  ingressClassName: <class-name>
+  annotations: {}
+  hosts:
+    - host: <lakefs.acme.com>
+      paths:
+       - /
+
+fluffy:
+  enabled: true
+  image:
+    repository: treeverse/fluffy
+    pullPolicy: IfNotPresent
+    privateRegistry:
+      enabled: true
+      secretToken: <dockerhub-token-fluffy-image>
+  fluffyConfig: |
+    logging:
+      format: "json"
+      level: "DEBUG"
+    auth:
+      # redirect after logout
+      logout_redirect_url: https://<lakefs.acme.com>
+      saml:
+        sp_sign_request: false
+        sp_signature_method: "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"
+        idp_metadata_url: https://login.microsoftonline.com/<...>/federationmetadata/2007-06/federationmetadata.xml?appid=<app-id>
+        # the default id format urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified
+        # idp_authn_name_id_format: "urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified"
+        external_user_id_claim_name: http://schemas.xmlsoap.org/ws/2005/05/identity/claims/name
+        idp_skip_verify_tls_cert: true
+  secrets:
+    create: true
+  sso:
+    enabled: true
+    saml:
+      enabled: true
+      createSecret: true
+      lakeFSServiceProviderIngress: https://<lakefs.acme.com>
+      certificate:
+        # certificate and private key for the SAML service provider to sign outgoing SAML requests
+        saml_rsa_public_cert: |
+          -----BEGIN CERTIFICATE-----
+          ...
+          -----END CERTIFICATE-----
+        saml_rsa_private_key: |
+          -----BEGIN PRIVATE KEY-----
+          ...
+          -----END PRIVATE KEY-----
+  rbac:
+    enabled: true
+
+ +
+
+ +

The following values file will run lakeFS Enterprise with LDAP.

+ +
+

The full LDAP configurations explained here.

+
+ +
lakefsConfig: |
+  logging:
+      level: "INFO"
+  blockstore:
+    type: local
+  auth:
+    remote_authenticator:
+      enabled: true
+      # RBAC group for first time users
+      default_user_group: "Developers"
+    ui_config:
+      login_cookie_names:
+        - internal_auth_session
+
+ingress:
+  enabled: true
+  ingressClassName: <class-name>
+  hosts:
+    - host: <lakefs.acme.com>
+      paths:
+       - /
+
+fluffy:
+  enabled: true
+  image:
+    privateRegistry:
+      enabled: true
+      secretToken: <dockerhub-token-fluffy-image>
+  fluffyConfig: |
+    logging:
+      level: "INFO"
+    auth:
+      post_login_redirect_url: /
+      ldap:
+        server_endpoint: 'ldaps://ldap.company.com:636'
+        bind_dn: uid=<bind-user-name>,ou=Users,o=<org-id>,dc=<company>,dc=com
+        username_attribute: uid
+        user_base_dn: ou=Users,o=<org-id>,dc=<company>,dc=com
+        user_filter: (objectClass=inetOrgPerson)
+        connection_timeout_seconds: 15
+        request_timeout_seconds: 17
+  secrets:
+    create: true
+  sso:
+    enabled: true
+    ldap:
+      enabled: true
+      bind_password: <ldap bind password>
+  rbac:
+    enabled: true
+
+useDevPostgres: true
+
+ +
+
+ +

See additional examples on GitHub we provide for each authentication method (oidc, adfs, ldap, rbac, IAM etc).

+

+ + + Database Configuration + + +

+ + +

In this section, you will learn how to configure lakeFS Enterprise to work with the KV Database you created (see prerequisites.

+ +

Notes:

+
    +
  • By default, the lakeFS Helm chart comes with useDevPostgres: true, you should change it to useDevPostgres: false for Fluffy to work with your KV Database and be suitable for production needs.
  • +
  • The KV database is shared between lakeFS and Fluffy, and therefore both services must use the same configuration.
  • +
  • See fluffy and lakeFS database configuration.
  • +
+ +

The database configuration structure between lakeFS and fluffy can be set directly via fluffyConfig, via K8S Secret Kind, and lakefsConfig or via environment variables.

+ +
+ +
+ +

This example uses Postgres as KV Database. lakeFS is configured via lakefsConfig and Fluffy via environment with the same database configuration.

+ +
useDevPostgres: false
+lakefsConfig: |
+  database:
+    type: postgres
+    postgres:
+      connection_string: <postgres connection string>
+
+fluffy:
+  extraEnvVars:
+    - name: FLUFFY_DATABASE_TYPE
+      value: postgres
+    - name: FLUFFY_DATABASE_POSTGRES_CONNECTION_STRING
+      value: '<postgres connection string>'
+
+
+
+ +
+ +

This example uses DynamoDB as KV Database.

+ +

+# disable dev postgres
+useDevPostgres: false
+
+lakefsConfig: |
+  database:
+    type: dynamodb
+    dynamodb:
+      table_name: <table>
+      aws_profile: <profile>
+fluffyConfig: |
+    database:
+      type: dynamodb
+      dynamodb:
+        table_name: <table>
+        aws_profile: <profile>
+        aws_region: <region>
+
+
+
+ +

This example uses Postgres as KV Database. The chart will create a kind: Secret holding the database connection string, and the lakeFS and Fluffy will use it.

+ +
useDevPostgres: false
+secrets:
+  authEncryptSecretKey: shared-key-hello
+  databaseConnectionString: <postgres connection string>
+
+lakefsConfig: |
+  database:
+    type: postgres
+fluffyConfig: |
+  database:
+    type: postgres
+
+
+

+ + + Install the lakeFS Helm Chart + + +

+ + +

After populating your values.yaml file with the relevant configuration, in the desired K8S namespace run helm install lakefs lakefs/lakefs -f values.yaml

+

+ + + Access the lakeFS UI + + +

+ + +

In your browser go to the to the Ingress host to access lakeFS UI.

+

+ + + Log Collection + + +

+ + +

The recommended practice for collecting logs would be sending them to the container std (default configuration) +and letting an external service to collect them to a sink. An example for logs collector would be fluentbit +that can collect container logs, format them and ship them to a target like S3.

+ +

There are 2 kinds of logs, regular logs like an API error or some event description used for debugging +and audit_logs that are describing a user action (i.e create branch). +The distinction between regular logs and audit_logs is in the boolean field log_audit. +lakeFS and fluffy share the same configuration structure under logging.* section in the config.

+

+ + + Advanced Deployment Configurations + + +

+ + +

The following example demonstrates a scenario where you need to configure an HTTP proxy for lakeFS and Fluffy, TLS certificates for the Ingress and extending the K8S manifests without forking the Helm chart.

+ +
ingress:
+  enabled: true
+  ingressClassName: <class-name>
+  # configure TLS certificate for the Ingress
+  tls:
+    - hosts:
+      - lakefs.acme.com
+      secretName: somesecret
+  hosts:
+    - host: lakefs.acme.com
+      paths:
+       - /
+
+# configure proxy for lakeFS
+extraEnvVars:
+  - name: HTTP_PROXY
+    value: 'http://my.company.proxy:8081'
+  - name: HTTPS_PROXY
+    value: 'http://my.company.proxy:8081'
+
+fluffy:
+  # configure proxy for fluffy
+  extraEnvVars:
+    - name: HTTP_PROXY
+      value: 'http://my.company.proxy:8081'
+    - name: HTTPS_PROXY
+      value: 'http://my.company.proxy:8081'
+
+# advanced: extra manifests to extend the K8S resources
+extraManifests:
+  - apiVersion: v1
+    kind: ConfigMap
+    metadata:
+      name: '-extra-config'
+    data:
+      config.yaml: my-data
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/getstarted/migrate-from-oss.html b/v1.46/enterprise/getstarted/migrate-from-oss.html new file mode 100644 index 000000000..049a3a9a3 --- /dev/null +++ b/v1.46/enterprise/getstarted/migrate-from-oss.html @@ -0,0 +1,739 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Migrate from lakeFS OSS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Migrate From lakeFS Open-Source to lakeFS Enterprise + + +

+ + +

To migrate from lakeFS Open Source to lakeFS Enterprise, follow the steps below:

+ +
    +
  1. Make sure you have the Fluffy Docker token, if not contact us to gain access to Fluffy. You will be granted with a token that enables downloading dockerhub/fluffy from Docker Hub.
  2. +
  3. Sanity Test (Optional): Install a new test lakeFS Enterprise before moving your current production setup. Test the setup > login > Create repository etc. Once everything seems to work, delete and cleanup the test setup and we will move to the migration process.
  4. +
  5. Follow lakeFS Enterprise installation guide +
      +
    1. Make sure that you meet the prerequisites
    2. +
    3. Update your existing values.yaml file for your deployment
    4. +
    +
  6. +
  7. DB Migration: we are going to use the same DB for both lakeFS and Fluffy, so we need to migrate the DB schema.
  8. +
  9. Make sure to SSH / exec into the lakeFS server (old pre-upgrade version), the point is to use the same lakefs configuration file when running a migration. +
      +
    1. If upgrading lakefs version do this or skip to the next step: Install the new lakeFS binary, if not use the existing one (the one you are running).
    2. +
    3. Run the command: LAKEFS_AUTH_UI_CONFIG_RBAC=internal lakefs migrate up (use the new binary if upgrading lakeFS version).
    4. +
    5. You should expect to see a log message saying Migration completed successfully.
    6. +
    7. During this short db migration process please make sure not to make any policy / RBAC related changes.
    8. +
    +
  10. +
  11. Once the migration completed - Upgrade your helm release with the modified values.yaml and the new version and run helm ugprade.
  12. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/getstarted/quickstart.html b/v1.46/enterprise/getstarted/quickstart.html new file mode 100644 index 000000000..a21aa7687 --- /dev/null +++ b/v1.46/enterprise/getstarted/quickstart.html @@ -0,0 +1,1090 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Quickstart | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Quickstart + + +

+ + +

Follow these quickstarts to try out lakeFS Enterprise.

+ +
+

⚠️ lakeFS Enterprise Quickstarts are not suitable for production use-cases.

+
+
+

+ + + Table of contents + + +

+ + +
    +
  1. lakeFS Enterprise Sample
  2. +
  3. Docker Quickstart
  4. +
  5. Kubernetes Helm Chart Quickstart
  6. +
+ +
+

+ + + lakeFS Enterprise Sample + + +

+ + +

The lakeFS Enterprise Sample is the quickest way to experience the value of lakeFS Enterprise features in a containerized environment. This Docker-based setup is ideal if you want +to easily interact with lakeFS without the hassle of integration and experiment with lakeFS without writing code.

+ +

By running the lakeFS Enterprise Sample, you will be getting a ready-to-use environment including +the following containers:

+
    +
  • lakeFS
  • +
  • Fluffy (includes lakeFS Enterprise features)
  • +
  • Postgres: used by lakeFS and Fluffy as a shared KV store
  • +
  • MinIO container: used as the storage connected to lakeFS
  • +
  • Jupyter notebooks setup: Pre-populated with notebooks that demonstrate lakeFS Enterprise’ capabilities
  • +
  • Apache Spark: this is useful for interacting with data you’ll manage with lakeFS
  • +
+ +

Checkout the RBAC demo notebook to see lakeFS Enterprise Role-Based Access Control capabilities in action.

+

+ + + Docker Quickstart + + +

+ +

+ + + Prerequisites + + +

+ + +
    +
  1. You have installed Docker Compose version 2.23.1 or higher on your machine.
  2. +
  3. Access to download dockerhub/fluffy from Docker Hub. Contact us to gain access to Fluffy.
  4. +
  5. With the token you’ve been granted, login locally to Docker Hub with docker login -u externallakefs -p <TOKEN>.
  6. +
+ +


+The quickstart docker-compose files below create a lakeFS server that’s connected to a local blockstore and spin up the following containers:

+
    +
  • lakeFS
  • +
  • Fluffy (includes lakeFS Enterprise features)
  • +
  • Postgres: used by lakeFS and Fluffy as a shared KV store
  • +
+ +

You can choose from the the following options:

+
    +
  1. Recommended: A fully functional lakeFS Enterprise setup without SSO support
  2. +
  3. Advanced: A fully functional lakeFS Enterprise setup including SSO support with OIDC integration configured
  4. +
+ +
+

If you can postpone the evaluation of the SSO integration, we suggest starting without it to speed up overall testing. The SSO integration requires additional configurations and is best addressed later.

+
+ +


+
+ +
+ +
    +
  1. Create a docker-compose.yaml file with the following content
  2. +
  3. Run docker compose up in the same directory as the docker-compose.yaml file.
  4. +
  5. In your browser, go to http://localhost:8080 to access lakeFS UI.
  6. +
+ +
version: "3"
+services:
+  lakefs:
+    image: "treeverse/lakefs:1.25.0"
+    command: "RUN"
+    ports:
+      - "8080:8080"
+    depends_on:
+      - "postgres"
+    environment:
+      - LAKEFS_LISTEN_ADDRESS=0.0.0.0:8080
+      - LAKEFS_LOGGING_LEVEL=DEBUG
+      - LAKEFS_AUTH_ENCRYPT_SECRET_KEY="random_secret"
+      - LAKEFS_AUTH_API_ENDPOINT=http://fluffy:9000/api/v1
+      - LAKEFS_AUTH_API_SUPPORTS_INVITES=true
+      - LAKEFS_AUTH_UI_CONFIG_RBAC=internal
+      - LAKEFS_AUTH_AUTHENTICATION_API_ENDPOINT=http://localhost:8000/api/v1
+      - LAKEFS_AUTH_AUTHENTICATION_API_EXTERNAL_PRINCIPALS_ENABLED=true
+      - LAKEFS_DATABASE_TYPE=postgres
+      - LAKEFS_DATABASE_POSTGRES_CONNECTION_STRING=postgres://lakefs:lakefs@postgres/postgres?sslmode=disable
+      - LAKEFS_BLOCKSTORE_TYPE=local
+      - LAKEFS_BLOCKSTORE_LOCAL_PATH=/home/lakefs
+      - LAKEFS_BLOCKSTORE_LOCAL_IMPORT_ENABLED=true
+    entrypoint: ["/app/wait-for", "postgres:5432", "--", "/app/lakefs", "run"]
+    configs:
+      - source: lakefs.yaml
+        target: /etc/lakefs/config.yaml
+  postgres:
+    image: "postgres:11"
+    ports:
+      - "5433:5432"
+    environment:
+      POSTGRES_USER: lakefs
+      POSTGRES_PASSWORD: lakefs
+
+  fluffy:
+    image: "${FLUFFY_REPO:-treeverse}/fluffy:${TAG:-0.4.4}"
+    command: "${COMMAND:-run}"
+    ports:
+      - "8000:8000"
+      - "9000:9000"
+    depends_on:
+      - "postgres"
+    environment:
+      - FLUFFY_LOGGING_LEVEL=DEBUG
+      - FLUFFY_DATABASE_TYPE=postgres
+      - FLUFFY_DATABASE_POSTGRES_CONNECTION_STRING=postgres://lakefs:lakefs@postgres/postgres?sslmode=disable
+      - FLUFFY_AUTH_ENCRYPT_SECRET_KEY="random_secret"
+      - FLUFFY_AUTH_SERVE_LISTEN_ADDRESS=0.0.0.0:9000
+      - FLUFFY_LISTEN_ADDRESS=0.0.0.0:8000
+      - FLUFFY_AUTH_SERVE_DISABLE_AUTHENTICATION=true
+      - FLUFFY_AUTH_POST_LOGIN_REDIRECT_URL=http://localhost:8080/
+    entrypoint: [ "/app/wait-for", "postgres:5432", "--", "/app/fluffy" ]
+
+configs:
+  lakefs.yaml:
+    content: |
+      auth:
+        ui_config:
+          login_cookie_names:
+            - internal_auth_session
+
+ +
+
+ +

This setup uses OIDC as the SSO authentication method thus requiring a valid OIDC configuration.

+
    +
  1. Create a docker-compose.yaml with the content below.
  2. +
  3. Create a .env file with the configurations below in the same directory as the docker-compose.yaml, docker compose will automatically use that.
  4. +
  5. Run docker compose up in the same directory as the docker-compose.yaml file.
  6. +
  7. Validate the OIDC configuration: +
      +
    • In your browser, go to http://localhost:8080 to access lakeFS UI
    • +
    • Complete the Setup process, and login with your Admin credentials
    • +
    • Logout and try to login again, you will be redirected to the OIDC login page.
    • +
    +
  8. +
+ +

.env

+ +
FLUFFY_AUTH_OIDC_CLIENT_ID=
+FLUFFY_AUTH_OIDC_CLIENT_SECRET=
+# The name of the query parameter that is used to pass the client ID to the logout endpoint of the SSO provider, i.e client_id
+FLUFFY_AUTH_OIDC_LOGOUT_CLIENT_ID_QUERY_PARAMETER=
+FLUFFY_AUTH_OIDC_URL=https://my-sso.com/
+FLUFFY_AUTH_LOGOUT_REDIRECT_URL=https://my-sso.com/logout
+# Optional: display a friendly name in the lakeFS UI by specifying which claim from the provider to show (i.e name, nickname, email etc)
+LAKEFS_AUTH_OIDC_FRIENDLY_NAME_CLAIM_NAME=
+
+ +

docker-compose.yaml

+ +
version: "3"
+services:
+  lakefs:
+    image: "treeverse/lakefs:1.25.0"
+    command: "RUN"
+    ports:
+      - "8080:8080"
+    depends_on:
+      - "postgres"
+    environment:
+      - LAKEFS_LISTEN_ADDRESS=0.0.0.0:8080
+      - LAKEFS_LOGGING_LEVEL=DEBUG
+      - LAKEFS_AUTH_ENCRYPT_SECRET_KEY="random_secret"
+      - LAKEFS_AUTH_API_ENDPOINT=http://fluffy:9000/api/v1
+      - LAKEFS_AUTH_API_SUPPORTS_INVITES=true
+      - LAKEFS_AUTH_UI_CONFIG_LOGIN_URL=http://localhost:8000/oidc/login
+      - LAKEFS_AUTH_UI_CONFIG_LOGOUT_URL=http://localhost:8000/oidc/logout
+      - LAKEFS_AUTH_UI_CONFIG_RBAC=internal
+      - LAKEFS_AUTH_AUTHENTICATION_API_ENDPOINT=http://localhost:8000/api/v1
+      - LAKEFS_AUTH_AUTHENTICATION_API_EXTERNAL_PRINCIPALS_ENABLED=true
+      - LAKEFS_DATABASE_TYPE=postgres
+      - LAKEFS_DATABASE_POSTGRES_CONNECTION_STRING=postgres://lakefs:lakefs@postgres/postgres?sslmode=disable
+      - LAKEFS_BLOCKSTORE_TYPE=local
+      - LAKEFS_BLOCKSTORE_LOCAL_PATH=/home/lakefs
+      - LAKEFS_BLOCKSTORE_LOCAL_IMPORT_ENABLED=true
+      - LAKEFS_AUTH_OIDC_FRIENDLY_NAME_CLAIM_NAME=${LAKEFS_AUTH_OIDC_FRIENDLY_NAME_CLAIM_NAME}
+    entrypoint: ["/app/wait-for", "postgres:5432", "--", "/app/lakefs", "run"]
+    configs:
+      - source: lakefs.yaml
+        target: /etc/lakefs/config.yaml
+  postgres:
+    image: "postgres:11"
+    ports:
+      - "5433:5432"
+    environment:
+      POSTGRES_USER: lakefs
+      POSTGRES_PASSWORD: lakefs
+
+  fluffy:
+    image: "${FLUFFY_REPO:-treeverse}/fluffy:${TAG:-0.4.4}"
+    command: "${COMMAND:-run}"
+    ports:
+      - "8000:8000"
+      - "9000:9000"
+    depends_on:
+      - "postgres"
+    environment:
+      - FLUFFY_LOGGING_LEVEL=DEBUG
+      - FLUFFY_DATABASE_TYPE=postgres
+      - FLUFFY_DATABASE_POSTGRES_CONNECTION_STRING=postgres://lakefs:lakefs@postgres/postgres?sslmode=disable
+      - FLUFFY_AUTH_ENCRYPT_SECRET_KEY="random_secret"
+      - FLUFFY_AUTH_SERVE_LISTEN_ADDRESS=0.0.0.0:9000
+      - FLUFFY_LISTEN_ADDRESS=0.0.0.0:8000
+      - FLUFFY_AUTH_SERVE_DISABLE_AUTHENTICATION=true
+      - FLUFFY_AUTH_LOGOUT_REDIRECT_URL=${FLUFFY_AUTH_LOGOUT_REDIRECT_URL}
+      - FLUFFY_AUTH_POST_LOGIN_REDIRECT_URL=http://localhost:8080/
+      - FLUFFY_AUTH_OIDC_ENABLED=true
+      - FLUFFY_AUTH_OIDC_URL=${FLUFFY_AUTH_OIDC_URL}
+      - FLUFFY_AUTH_OIDC_CLIENT_ID=${FLUFFY_AUTH_OIDC_CLIENT_ID}
+      - FLUFFY_AUTH_OIDC_CLIENT_SECRET=${FLUFFY_AUTH_OIDC_CLIENT_SECRET}
+      - FLUFFY_AUTH_OIDC_CALLBACK_BASE_URL=http://localhost:8000
+      - FLUFFY_AUTH_OIDC_LOGOUT_CLIENT_ID_QUERY_PARAMETER=${FLUFFY_AUTH_OIDC_LOGOUT_CLIENT_ID_QUERY_PARAMETER}
+    entrypoint: [ "/app/wait-for", "postgres:5432", "--", "/app/fluffy" ]
+    configs:
+      - source: fluffy.yaml
+        target: /etc/fluffy/config.yaml
+
+ #This tweak is unfortunate but also necessary. logout_endpoint_query_parameters is a list
+ #of strings which isn't parsed nicely as env vars.
+configs:
+  lakefs.yaml:
+    content: |
+      auth:
+        ui_config:
+          login_cookie_names:
+            - internal_auth_session
+            - oidc_auth_session
+        oidc:
+          # friendly_name_claim_name: "name"
+          default_initial_groups:
+            - Admins
+
+  fluffy.yaml:
+    content: |
+      auth:
+        oidc:
+          logout_endpoint_query_parameters:
+            - returnTo
+            - http://localhost:8080/oidc/login
+
+ +
+
+

+ + + Kubernetes Helm Chart Quickstart + + +

+ + +

In order to use lakeFS Enterprise and Fluffy, we provided out of the box setup, see lakeFS Helm chart configuration.

+ +

The values below create a fully functional lakeFS Enterprise setup without SSO support. The created setup is connected to a local blockstore, and spins up the following pods:

+
    +
  • lakeFS
  • +
  • Fluffy (includes lakeFS Enterprise features)
  • +
  • Postgres: used by lakeFS and Fluffy as a shared KV store
  • +
+ +
+

If you can postpone the evaluation of the SSO integration, we suggest starting without it to speed up overall testing. The SSO integration requires additional configurations and is best addressed later. To +try lakeFS Enterprise SSO capability on a Kubernetes cluster, check out the production deployment guide.

+
+

+ + + Prerequisites + + +

+ + +
    +
  1. You have a Kubernetes cluster running in one of the platforms supported by lakeFS.
  2. +
  3. Helm is installed
  4. +
  5. Access to download dockerhub/fluffy from Docker Hub. Contact us to gain access to Fluffy.
  6. +
+

+ + + Instructions + + +

+ + +
    +
  1. Add the lakeFS Helm repository with helm repo add lakefs https://charts.lakefs.io
  2. +
  3. Create a values.yaml file with the following content and make sure to replace <fluffy-docker-registry-token> with the token Docker Hub token you recieved, <lakefs.acme.com> and <ingress-class-name>.
  4. +
  5. In the desired K8S namespace run helm install lakefs lakefs/lakefs -f values.yaml
  6. +
  7. In your browser go to the Ingress host to access lakeFS UI.
  8. +
+ +
lakefsConfig: |
+  logging:
+      level: "DEBUG"
+  blockstore:
+    type: local
+ingress:
+  enabled: true
+  ingressClassName: <ingress-class-name>
+  annotations: {}
+  hosts:
+    - host: <lakefs.acme.com>
+      paths:
+       - /
+fluffy:
+  enabled: true
+  image:
+    privateRegistry:
+      enabled: true
+      secretToken: <fluffy-docker-registry-token>
+  fluffyConfig: |
+    logging:
+      level: "DEBUG"
+  secrets:
+    create: true
+  sso:
+    enabled: false
+  rbac:
+    enabled: true
+
+# useDevPostgres is true by default and will override any other db configuration, set false for configuring your own db
+useDevPostgres: true
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/index.html b/v1.46/enterprise/index.html new file mode 100644 index 000000000..1260b6354 --- /dev/null +++ b/v1.46/enterprise/index.html @@ -0,0 +1,863 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakeFS Enterprise | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS Enterprise + + +

+ +

+ + + What is lakeFS Enterprise? + + +

+ + +

lakeFS Enterprise is a commercially-supported version of lakeFS, offering additional features and functionalities that meet the needs of organizations from a production-grade system.

+

+ + + Why did we build lakeFS Enterprise? + + +

+ + +

lakeFS Enterprise was built for organizations that require the support, security standards and features required of a production-grade system and +are not using public clouds, hence they cannot use lakeFS Cloud.

+

+ + + What is the value of using lakeFS Enterprise? + + +

+ + +
    +
  1. Support: the lakeFS team is committed to supporting you under an SLA for both issues and product enhancements.
  2. +
  3. Security: Full support for a suite of security features and additional lakeFS functionality.
  4. +
+

+ + + What security features does lakeFS Enterprise provide? + + +

+ + +

With lakeFS Enterprise you’ll receive access to the security package containing the following features:

+
    +
  1. A rich Role-Based Access Control permission system that allows for fine-grained control by associating permissions with users and groups, granting them specific actions on specific resources. This ensures data security and compliance within an organization.
  2. +
  3. To easily manage users and groups, lakeFS Enterprise provides SSO integration (including support for SAML, OIDC, ADFS, Okta, and Azure AD), supporting existing credentials from a trusted provider, eliminating separate logins.
  4. +
  5. lakeFS Enterprise supports SCIM for automatically provisioning and deprovisioning users and group memberships to allow organizations to maintain a single source of truth for their user database.
  6. +
  7. STS Auth offers temporary, secure logins using an Identity Provider, simplifying user access and enhancing security.
  8. +
  9. Authentication with AWS IAM Roles allows authentication using AWS IAM roles instead of lakeFS credentials, removing the need to maintain static credentials for lakeFS Enterprise users running on AWS.
  10. +
  11. Auditing provides a detailed action log of events happening within lakeFS, including who performed which action, on which resource - and when.
  12. +
+

+ + + What additional functionality does lakeFS Enterprise provide? + + +

+ + +
    +
  1. lakeFS Mount allows users to virtually mount a remote lakeFS repository onto a local directory. Once mounted, users can access the data as if it resides on their local filesystem, using any tool, library, or framework that reads from a local filesystem.
  2. +
  3. Transactional Mirroring - allows replicating lakeFS repositories into consistent read-only copies in remote locations.
  4. +
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
FeatureOSSEnterprise
Format-agnostic data version control
Cloud-agnostic
Zero Clone copy for isolated environment
Atomic Data Promotion (via merges)
Data stays in one place
Configurable Garbage Collection
Data CI/CD using lakeFS hooks
Integrates with your data stack
Role Based Access Control (RBAC) 
Single Sign On (SSO) 
SCIM Support 
IAM Roles 
Mount Capability 
Audit Logs 
Transactional Mirroring (cross-region) 
Support SLA 
+ +


+You can learn more about the lakeFS Enterprise architecture, or follow the examples in the Quickstart guide.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/sso.html b/v1.46/enterprise/sso.html new file mode 100644 index 000000000..52716caca --- /dev/null +++ b/v1.46/enterprise/sso.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/enterprise/troubleshooting.html b/v1.46/enterprise/troubleshooting.html new file mode 100644 index 000000000..a1f4024d5 --- /dev/null +++ b/v1.46/enterprise/troubleshooting.html @@ -0,0 +1,954 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Troubleshooting lakeFS Enterprise | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Troubleshooting lakeFS Enterprise + + +

+ + + + +

A lakeFS Enterprise deployment has multiple moving parts that must all be deployed and configured correctly. This is especially true during initial setup. To help troubleshoot issues, both lakeFS and fluffy include the flare command.

+

+ + + The flare command + + +

+ +

+ + + Synopsis + + +

+ + +

Both lakefs and fluffy include the flare command

+ +
lakefs flare [flags]
+fluffy flare [flags]
+
+

+ + + Flags + + +

+ + +
        --env-var-filename      the name of the file environment variables will be written to (default: lakefs-env.txt)
+        --include-env-vars      should environment variables be collected by flare (default: true)
+    -o, --output                Output path relative to the current path
+    -p, --package               Package generated artifacts into a .zip file (default: false)
+        --stdout                Output to stdout instead of files (default: false)
+        --zip-filename          The name of the zip file created when the -p option is used (default: lakefs-flare.zip)
+
+

+ + + Example Usage + + +

+ + +
# This will run flare with output set to stdout, which is redirected into a file
+$ ./lakefs flare --stdout > lakefs.flare
+# The same works for fluffy
+$ ./fluffy flare --stdout > fluffy.flare
+
+

+ + + What Information Does the flare Command Collect? + + +

+ +

+ + + Configuration + + +

+ + +

Both lakeFS and fluffy allow configuration to be supplied in multiple ways: configuration file, environment variables, .env files, and command flags. The flare command collects the fully resolved final configuration used by the lakeFS/fluffy process.

+

+ + + Environment Variables + + +

+ + +

When troubleshooting, it’s important to get a view of the environment in which lakeFS/fluffy are running. This is especially true for container-based deployment environments, like Kubernetes, where env vars are used extensively. The flare command collects environment variables with the following prefixes:

+ +
    +
  • LAKEFS_
  • +
  • FLUFFY_
  • +
  • HTTP_
  • +
  • HOSTNAME
  • +
+

+ + + Sanitization and Secret Redaction + + +

+ + +

Both configuration and env var include sensitive secrets. The flare command has a multi-step process for redacting secrets from what it collects. The flare command is able to detect the following types of secrets:

+ +
    +
  • AWS static credentials
  • +
  • Azure storage keys
  • +
  • Basic HTTP auth username/password
  • +
  • JWTs
  • +
  • Bearer auth headers
  • +
  • GitHub tokens
  • +
  • Certificate private keys
  • +
+ +

Aside from the specific secret type listed above, flare also has the ability to detect and redact generic high-entropy strings, which are likely to be secrets.

+ +

Redacted secrets are replaced by a SHA512 hash of the value. This allows comparing them (e.g., between lakeFS and fluffy) without exposing the actual values.

+

+ + + Usage - Collect and Send Flare + + +

+ + +

The following script is intended to be run locally and assumes that lakeFS and fluffy are deployed to a Kubernetes cluster, since this is the recommended setup.
+Running this script requires that kubectl be installed on the machine it is being run from and that kubectl is configured with the correct context and credentials to access the cluster. Aside from running the flare command on both lakeFS and fluffy, this script also fetches the logs from all running pods of lakeFS and fluffy.

+

+ + + Step 1 - Set Script Variables + + +

+ + +

At the top of the script you’ll find the Variables block. It is important to change these values according to how lakeFS is deployed in your cluster.

+ +

NAMESPACE - The K8s namespace where lakeFS and fluffy are deployed
+LAKEFS_DEPLOYMENT - The name of the lakeFS K8s deployment
+FLUFFY_DEPLOYMENT - The name of the fluffy K8s deployment
+LAKEFS_LOGS_OUTPUT_FILE - The name of the local file where lakeFS logs will be saved
+FLUFFY_LOGS_OUTPUT_FILE - The name of the local file where fluffy logs will be saved
+LAKEFS_FLARE_FILE - The name of the local file where the lakeFS flare result will be saved
+FLUFFY_FLARE_FILE - The name of the local file where the fluffy flare result will be saved

+

+ + + Step 2 - Execute the Script + + +

+ + +
#!/bin/bash
+
+RED='\033[0;31m'
+NC='\033[0m'
+
+# Variables
+NAMESPACE=lakefs-prod
+LAKEFS_DEPLOYMENT=lakefs-server
+FLUFFY_DEPLOYMENT=lakefs-fluffy
+LAKEFS_LOGS_OUTPUT_FILE=lakefs.log
+FLUFFY_LOGS_OUTPUT_FILE=fluffy.log
+LAKEFS_FLARE_FILE=lakefs.flare
+FLUFFY_FLARE_FILE=fluffy.flare
+
+# Find kubectl
+KUBECTLCMD=$(which kubectl)
+if [ -z "$KUBECTLCMD" ]
+then
+    echo -e "${RED}Couldn't find kubectl in path${NC}"
+    exit 1
+fi
+
+$KUBECTLCMD get pods -o name -n $NAMESPACE | grep pod/$LAKEFS_DEPLOYMENT | xargs -I {} $KUBECTLCMD logs -n $NAMESPACE --all-containers=true --prefix --ignore-errors --timestamps {} > $LAKEFS_LOGS_OUTPUT_FILE
+$KUBECTLCMD get pods -o name -n $NAMESPACE | grep pod/$FLUFFY_DEPLOYMENT | xargs -I {} $KUBECTLCMD logs -n $NAMESPACE --all-containers=true --prefix --ignore-errors --timestamps {} > $FLUFFY_LOGS_OUTPUT_FILE
+
+$KUBECTLCMD exec deployment/$LAKEFS_DEPLOYMENT -- ./lakefs flare --stdout > $LAKEFS_FLARE_FILE
+$KUBECTLCMD exec deployment/$FLUFFY_DEPLOYMENT -- ./lakefs flare --stdout > $FLUFFY_FLARE_FILE
+
+

+ + + Step 3 - Inspect the Output Files + + +

+ + +

After executing the script you should have four files: lakeFS/fluffy logs and lakeFS/fluffy flare output. Before sharing these files, please review them to make sure all secrets were correctly redacted and that all the collected information is shareable.

+

+ + + Step 4 - Zip Output Files and Attach to Support Ticket + + +

+ + +

Once you’re done inspecting the files, you can zip them into a single file and attach it to a Support Ticket in the support portal.

+

+ + + New Ticket + + +

+ + +

When filing a new ticket, you can attach the zip file using the file upload input at the bottom of the new ticket form.

+ +

new ticket

+

+ + + Existing Ticket + + +

+ + +

To add a file to an existing ticket, click the ticket subject in the support portal. On the ticket details attach a file as a new response. You can also add text to the response, if you’d like to add further details.

+ +

existing ticket

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/enterprise/upgrade.html b/v1.46/enterprise/upgrade.html new file mode 100644 index 000000000..bf21a1f59 --- /dev/null +++ b/v1.46/enterprise/upgrade.html @@ -0,0 +1,715 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Upgrade | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Upgrade + + +

+ + +

For upgrading from lakeFS enterprise to a newer version see lakefs migration.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/experimental/dir_marker.png b/v1.46/experimental/dir_marker.png new file mode 100644 index 000000000..51429cb7c Binary files /dev/null and b/v1.46/experimental/dir_marker.png differ diff --git a/v1.46/experimental/lakectl_local_posix.html b/v1.46/experimental/lakectl_local_posix.html new file mode 100644 index 000000000..c4aacdcea --- /dev/null +++ b/v1.46/experimental/lakectl_local_posix.html @@ -0,0 +1,1046 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakectl local file permissions tracking | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakectl local file permission tracking + + +

+ +

Experimental

+ +

This experimental feature is used to:

+
    +
  • Support mode preservation for files and folders managed via lakectl local for POSIX compliant filesystems only
  • +
  • Support user and group preservation for files and folders managed via lakectl local for POSIX compliant filesystems only
  • +
+ +
+

Note: This feature is currently supported for Unix based Operating Systems only

+
+ +
+

Please make sure to contact the lakeFS team before enabling any experimental features!

+
+

+ + + Configuration + + +

+ + +

By default, this feature is disabled. Enabling file permission tracking is done via the configuration variable +experimental.local.posix_permissions.enabled which can be added to the lakectl.yaml file:

+ +
   credentials:
+     access_key_id: AKIAIOSFDNN7EXAMPLEQ
+     secret_access_key: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+   metastore:
+     glue:
+       catalog_id: ""
+     hive:
+       db_location_uri: file:/user/hive/warehouse/
+       uri: ""
+   server:
+     endpoint_url: http://localhost:8000/api/v1
+   experimental:
+     local:
+       posix_permissions:
+         enabled: true
+
+ +

It is also possible to enable this feature via the corresponding environment variable:
+LAKECTL_EXPERIMENTAL_LOCAL_POSIX_PERMISSIONS_ENABLED

+

+ + + Usage + + +

+ + +

lakectl local will automatically track changes in file permissions and ownership as part of the sync with the remote server. +During the first sync with a remote path, lakectl local will need to update all the objects under this path in order to sync +the permissions with lakeFS. +When first creating the file and directory structure from an existing remote path the default permissions are set:

+
    +
  • 0o0666 & ~umask mode for files
  • +
  • 0o0777 & ~umask mode for directories
  • +
  • Numerical GID and UID of the current user
  • +
+ +
+

Note: Users with stricter umasks will change permissions on push

+
+ +
+

Please note that once a local path is synchronized with a remote path with permissions tracking enabled, you must continue to work with this remote path +with the feature enabled.

+
+

+ + + Changes from default behavior + + +

+ + +

In order to support this feature, some changes needed to be made in lakectl local, and as such the behavior when the feature is enabled is a bit different from the default behavior of lakectl local. +The main differences are:

+
    +
  1. To support tracking of directory permissions and ownership, lakectl local will now write directory markers to the remote server, which will hold this information on the remote object. +These markers are zero sized objects with paths ending in / designed to indicate a directory in S3 and used to save the directory information.
  2. +
  3. Syncing between local and remote paths will take into account changes in file ownership and permissions. In case of change in one of these, the file will show as modified
  4. +
+

+ + + Limitations / Warnings + + +

+ + +
    +
  • All machines on which lakectl local is used with POSIX permissions must share UIDs and GIDs.
  • +
  • Tracked modes will only be accessible via lakectl local and not through the UI, or any lakeFS API calls (e.g., statObject) or S3 gateway operations (e.g., getObject).
  • +
  • Tracked modes are not expected to persist through an upload followed by an overwrite by a client other than lakectl local. After an overwrite, files and directories are expected to have their respective default mode
  • +
  • The POSIX permissions feature and functionality is applicable only to the scope of lakectl local. Using other methods to read/write/diff on paths that were cloned with the feature enabled + will not guarantee consistent behavior compared to using the lakectl local commands. For example: performing lakectl diff might result in a different output than performing lakectl local status
  • +
  • Diff (lakectl local status) takes into account the following attributes: +
      +
    1. Client Mtime
    2. +
    3. File/Dir UID
    4. +
    5. File/Dir GID
    6. +
    7. File/Dir mode
    8. +
    +
  • +
+ +
+

Note: The changes to these attributes are not visualized or reported explicitly; the file will be reported as changed.

+
+ +
    +
  • Merge will fail on a conflict stemming from a change to permissions or ownership. There is no explicit reporting the root cause of the failure (the file will be reported as changed) +In case of failure to read/set/update permissions (due to user insufficient privileges, for example), lakectl local will fail with an informative message. However, there’s no guarantee for atomicity of operations. +In this case, the local directory/remote path might be in an inconsistent state, and it will be up to the user to fix it.
  • +
  • Unprivileged users can use non-lakectl-local commands in conjunction with lakectl local to change owners of files on remote machines in unpredictable ways, +including in some cases to set ownership to users for which they themselves cannot chown files.
  • +
+

+ + + Example + + +

+ + +
    +
  1. Clone remote repository to local path: +
     user@host /tmp/test-perm
    +   % LAKECTL_EXPERIMENTAL_LOCAL_POSIX_PERMISSIONS_ENABLED=true lakectl local clone lakefs://quickstart/dev/     
    + download data/lakes.source.md            ... done! [531B in 0s]
    + download images/quickstart-step-02-bran~ ... done! [3.09KB in 0s]
    + download images/quickstart-step-04-roll~ ... done! [3.22KB in 0s]
    + download images/quickstart-step-03-merg~ ... done! [3.43KB in 0s]
    + download images/quickstart-step-01-quer~ ... done! [3.69KB in 0s]
    + download images/axolotl.png              ... done! [3.82KB in 0s]
    + download images/quickstart-step-00-laun~ ... done! [4.10KB in 0s]
    + download images/quickstart-step-05-acti~ ... done! [4.84KB in 0s]
    + download images/waving-axolotl-transpar~ ... done! [22.92KB in 0s]
    + download README.md                       ... done! [28.99KB in 0s]
    + download images/create-lakefs-branch.png ... done! [56.24KB in 0s]
    + download images/commit-change-02.png     ... done! [64.40KB in 0s]
    + download images/duckdb-editor-05.png     ... done! [65.49KB in 0s]
    + download images/merge02.png              ... done! [70.17KB in 0s]
    + download images/duckdb-editor-03.png     ... done! [75.86KB in 0s]
    + download images/duckdb-editor-06.png     ... done! [87.92KB in 0s]
    + download images/duckdb-editor-04.png     ... done! [90.42KB in 0s]
    + download images/merge01.png              ... done! [92.30KB in 0s]
    + download images/commit-change.png        ... done! [98.80KB in 0s]
    + download images/duckdb-editor-02.png     ... done! [103.16KB in 0s]
    + download images/create-quickstart-repo.~ ... done! [121.72KB in 0s]
    + download images/hooks-05.png             ... done! [139.44KB in 0s]
    + download images/hooks-07.png             ... done! [142.94KB in 0s]
    + download images/hooks-06.png             ... done! [146.62KB in 0s]
    + download images/lakefs-login-screen.png  ... done! [148.67KB in 0s]
    + download images/hooks-02.png             ... done! [151.00KB in 0s]
    + download images/user_config.png          ... done! [166.49KB in 0s]
    + download images/hooks-04.png             ... done! [170.31KB in 0s]
    + download images/hooks-01.png             ... done! [172.13KB in 0s]
    + download images/repo-list.png            ... done! [178.64KB in 0s]
    + download images/hooks-00.png             ... done! [182.45KB in 0s]
    + download images/hooks-03.png             ... done! [182.78KB in 0s]
    + download images/duckdb-main-01.png       ... done! [206.85KB in 0s]
    + download images/duckdb-main-03.png       ... done! [207.50KB in 0s]
    + download images/quickstart-repo.gif      ... done! [213.80KB in 0s]
    + download images/hooks-08.png             ... done! [218.91KB in 0s]
    + download images/repo-contents.png        ... done! [232.40KB in 0s]
    + download images/duckdb-main-02.png       ... done! [239.56KB in 0s]
    + download images/empty-repo-list.png      ... done! [254.93KB in 0s]
    + download lakes.parquet                   ... done! [916.39KB in 1ms]
    +    
    + Successfully cloned lakefs://quickstart/dev/ to /tmp/test-perm.
    +    
    + Clone Summary:
    +    
    + Downloaded: 40
    + Uploaded: 0
    + Removed: 0
    +    
    +
    +
  2. +
  3. The first time we clone objects without permissions, all the files will appear as modified (note the added directory markers): +
     user@host /tmp/test-perm
    +   % LAKECTL_EXPERIMENTAL_LOCAL_POSIX_PERMISSIONS_ENABLED=true lakectl local status                                                                                                                   !10245
    +    
    + diff 'local:///tmp/test-perm' <--> 'lakefs://quickstart/8fa0d994afedbc00e0a44725e8a8d64695e638d4ba683756f5d2411eb7aa8fdf/'...
    + diff 'lakefs://quickstart/8fa0d994afedbc00e0a44725e8a8d64695e638d4ba683756f5d2411eb7aa8fdf/' <--> 'lakefs://quickstart/dev/'...
    +    
    + ╔════════╦══════════╦═══════════════════════════════════════════╗
    + ║ SOURCE ║ CHANGE   ║ PATH                                      ║
    + ╠════════╬══════════╬═══════════════════════════════════════════╣
    + ║ local  ║ modified ║ README.md                                 ║
    + ║ local  ║ added    ║ data/                                     ║
    + ║ local  ║ modified ║ data/lakes.source.md                      ║
    + ║ local  ║ added    ║ images/                                   ║
    + ║ local  ║ modified ║ images/axolotl.png                        ║
    + ║ local  ║ modified ║ images/commit-change-02.png               ║
    + ║ local  ║ modified ║ images/commit-change.png                  ║
    + ║ local  ║ modified ║ images/create-lakefs-branch.png           ║
    + ║ local  ║ modified ║ images/create-quickstart-repo.png         ║
    + ║ local  ║ modified ║ images/duckdb-editor-02.png               ║
    + ║ local  ║ modified ║ images/duckdb-editor-03.png               ║
    + ║ local  ║ modified ║ images/duckdb-editor-04.png               ║
    + ║ local  ║ modified ║ images/duckdb-editor-05.png               ║
    + ║ local  ║ modified ║ images/duckdb-editor-06.png               ║
    + ║ local  ║ modified ║ images/duckdb-main-01.png                 ║
    + ║ local  ║ modified ║ images/duckdb-main-02.png                 ║
    + ║ local  ║ modified ║ images/duckdb-main-03.png                 ║
    + ║ local  ║ modified ║ images/empty-repo-list.png                ║
    + ║ local  ║ modified ║ images/hooks-00.png                       ║
    + ║ local  ║ modified ║ images/hooks-01.png                       ║
    + ║ local  ║ modified ║ images/hooks-02.png                       ║
    + ║ local  ║ modified ║ images/hooks-03.png                       ║
    + ║ local  ║ modified ║ images/hooks-04.png                       ║
    + ║ local  ║ modified ║ images/hooks-05.png                       ║
    + ║ local  ║ modified ║ images/hooks-06.png                       ║
    + ║ local  ║ modified ║ images/hooks-07.png                       ║
    + ║ local  ║ modified ║ images/hooks-08.png                       ║
    + ║ local  ║ modified ║ images/lakefs-login-screen.png            ║
    + ║ local  ║ modified ║ images/merge01.png                        ║
    + ║ local  ║ modified ║ images/merge02.png                        ║
    + ║ local  ║ modified ║ images/quickstart-repo.gif                ║
    + ║ local  ║ modified ║ images/quickstart-step-00-launch.png      ║
    + ║ local  ║ modified ║ images/quickstart-step-01-query.png       ║
    + ║ local  ║ modified ║ images/quickstart-step-02-branch.png      ║
    + ║ local  ║ modified ║ images/quickstart-step-03-merge.png       ║
    + ║ local  ║ modified ║ images/quickstart-step-04-rollback.png    ║
    + ║ local  ║ modified ║ images/quickstart-step-05-actions.png     ║
    + ║ local  ║ modified ║ images/repo-contents.png                  ║
    + ║ local  ║ modified ║ images/repo-list.png                      ║
    + ║ local  ║ modified ║ images/user_config.png                    ║
    + ║ local  ║ modified ║ images/waving-axolotl-transparent-w90.gif ║
    + ║ local  ║ modified ║ lakes.parquet                             ║
    + ╚════════╩══════════╩═══════════════════════════════════════════╝
    +    
    +
    +
  4. +
  5. +

    Let’s commit the changes so that the remote path will be updated with the file permissions:

    + +
     user@host /tmp/test-perm
    +   % LAKECTL_EXPERIMENTAL_LOCAL_POSIX_PERMISSIONS_ENABLED=true lakectl local commit -m "Add file permissions"                                                                                         !10246
    +    
    + Getting branch: dev
    +    
    + diff 'local:///tmp/test-perm' <--> 'lakefs://quickstart/8fa0d994afedbc00e0a44725e8a8d64695e638d4ba683756f5d2411eb7aa8fdf/'...
    + upload data/                             ... done! [0B in 15ms]
    + upload images/                           ... done! [0B in 15ms]
    + upload data/lakes.source.md              ... done! [531B in 11ms]
    + upload images/quickstart-step-02-branch~ ... done! [3.09KB in 11ms]
    + upload images/quickstart-step-04-rollba~ ... done! [3.22KB in 11ms]
    + upload images/quickstart-step-03-merge.~ ... done! [3.43KB in 12ms]
    + upload images/quickstart-step-01-query.~ ... done! [3.69KB in 11ms]
    + upload images/axolotl.png                ... done! [3.82KB in 15ms]
    + upload images/quickstart-step-00-launch~ ... done! [4.10KB in 12ms]
    + upload images/quickstart-step-05-action~ ... done! [4.84KB in 11ms]
    + upload images/waving-axolotl-transparen~ ... done! [22.92KB in 11ms]
    + upload README.md                         ... done! [28.99KB in 15ms]
    + upload images/create-lakefs-branch.png   ... done! [56.24KB in 15ms]
    + upload images/commit-change-02.png       ... done! [64.40KB in 14ms]
    + upload images/duckdb-editor-05.png       ... done! [65.49KB in 14ms]
    + upload images/merge02.png                ... done! [70.17KB in 12ms]
    + upload images/duckdb-editor-03.png       ... done! [75.86KB in 16ms]
    + upload images/duckdb-editor-06.png       ... done! [87.92KB in 13ms]
    + upload images/duckdb-editor-04.png       ... done! [90.42KB in 15ms]
    + upload images/merge01.png                ... done! [92.30KB in 12ms]
    + upload images/commit-change.png          ... done! [98.80KB in 15ms]
    + upload images/duckdb-editor-02.png       ... done! [103.16KB in 12ms]
    + upload images/create-quickstart-repo.png ... done! [121.72KB in 15ms]
    + upload images/hooks-05.png               ... done! [139.44KB in 15ms]
    + upload images/hooks-07.png               ... done! [142.94KB in 11ms]
    + upload images/hooks-06.png               ... done! [146.62KB in 13ms]
    + upload images/lakefs-login-screen.png    ... done! [148.67KB in 12ms]
    + upload images/hooks-02.png               ... done! [151.00KB in 13ms]
    + upload images/user_config.png            ... done! [166.49KB in 11ms]
    + upload images/hooks-04.png               ... done! [170.31KB in 14ms]
    + upload images/hooks-01.png               ... done! [172.13KB in 13ms]
    + upload images/repo-list.png              ... done! [178.64KB in 11ms]
    + upload images/hooks-00.png               ... done! [182.45KB in 13ms]
    + upload images/hooks-03.png               ... done! [182.78KB in 13ms]
    + upload images/duckdb-main-01.png         ... done! [206.85KB in 13ms]
    + upload images/duckdb-main-03.png         ... done! [207.50KB in 15ms]
    + upload images/quickstart-repo.gif        ... done! [213.80KB in 12ms]
    + upload images/hooks-08.png               ... done! [218.91KB in 12ms]
    + upload images/repo-contents.png          ... done! [232.40KB in 11ms]
    + upload images/duckdb-main-02.png         ... done! [239.56KB in 14ms]
    + upload images/empty-repo-list.png        ... done! [254.93KB in 13ms]
    + upload lakes.parquet                     ... done! [916.39KB in 14ms]
    +    
    + Sync Summary:
    +    
    + Downloaded: 0
    + Uploaded: 42
    + Removed: 0
    +    
    + Finished syncing changes. Perform commit on branch...
    + Commit for branch "dev" completed.
    +    
    + ID: 7f1f0984030c43ab388a649fe97c3c1564421118099ccc8c47f74645776e73f2
    + Message: Add file permissions
    + Timestamp: 2024-07-03 16:04:07 +0300 IDT
    + Parents: 8fa0d994afedbc00e0a44725e8a8d64695e638d4ba683756f5d2411eb7aa8fdf
    +    
    +
    +
  6. +
  7. +

    Looking at the remote repository we will notice the directory markers: +dir_marker.png

    +
  8. +
  9. +

    The object metadata will contain the file/dir permissions: +object_stats.png

    +
  10. +
  11. +

    Updating a file’s mode will result in a diff:

    + +
     user@host /tmp/test-perm
    +   % chmod 770 lakes.parquet                                                                                                                                                                          !10247
    +    
    + user@host /tmp/test-perm
    +   % LAKECTL_EXPERIMENTAL_LOCAL_POSIX_PERMISSIONS_ENABLED=true lakectl local status                                                                                                                   !10248
    +    
    + diff 'local:///tmp/test-perm' <--> 'lakefs://quickstart/7f1f0984030c43ab388a649fe97c3c1564421118099ccc8c47f74645776e73f2/'...
    + diff 'lakefs://quickstart/7f1f0984030c43ab388a649fe97c3c1564421118099ccc8c47f74645776e73f2/' <--> 'lakefs://quickstart/dev/'...
    +    
    + ╔════════╦══════════╦═══════════════╗
    + ║ SOURCE ║ CHANGE   ║ PATH          ║
    + ╠════════╬══════════╬═══════════════╣
    + ║ local  ║ modified ║ lakes.parquet ║
    + ╚════════╩══════════╩═══════════════╝
    +    
    +
    +
  12. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/experimental/object_stats.png b/v1.46/experimental/object_stats.png new file mode 100644 index 000000000..b19616414 Binary files /dev/null and b/v1.46/experimental/object_stats.png differ diff --git a/v1.46/faq.html b/v1.46/faq.html new file mode 100644 index 000000000..e59180191 --- /dev/null +++ b/v1.46/faq.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/favicon.ico b/v1.46/favicon.ico new file mode 100644 index 000000000..9dd66cf1b Binary files /dev/null and b/v1.46/favicon.ico differ diff --git a/v1.46/glossary.html b/v1.46/glossary.html new file mode 100644 index 000000000..79940fc4e --- /dev/null +++ b/v1.46/glossary.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks.html b/v1.46/hooks.html new file mode 100644 index 000000000..2ee933063 --- /dev/null +++ b/v1.46/hooks.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks/airflow.html b/v1.46/hooks/airflow.html new file mode 100644 index 000000000..456a28ade --- /dev/null +++ b/v1.46/hooks/airflow.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks/index.html b/v1.46/hooks/index.html new file mode 100644 index 000000000..2ee933063 --- /dev/null +++ b/v1.46/hooks/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks/lua.html b/v1.46/hooks/lua.html new file mode 100644 index 000000000..cbe159558 --- /dev/null +++ b/v1.46/hooks/lua.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks/overview.html b/v1.46/hooks/overview.html new file mode 100644 index 000000000..2ee933063 --- /dev/null +++ b/v1.46/hooks/overview.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/hooks/webhooks.html b/v1.46/hooks/webhooks.html new file mode 100644 index 000000000..3f9cf4e41 --- /dev/null +++ b/v1.46/hooks/webhooks.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/backup-and-restore.html b/v1.46/howto/backup-and-restore.html new file mode 100644 index 000000000..9c8c99444 --- /dev/null +++ b/v1.46/howto/backup-and-restore.html @@ -0,0 +1,917 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Backup and Restore Repository | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Backup and Restore Repository + + +

+ +

This section explains how to backup and restore lakeFS repository for different use-cases:

+
    +
  1. Disaster Recovery: you want to backup the repository regularly so you can restore it in case of any disaster. You’d also need to make sure to backup the repository’s storage namespace to another, preferably geographically separate location.
  2. +
  3. Migrate Repository: you want to migrate a repository from one environment to another lakeFS environment.
  4. +
  5. Clone Repository: you want to clone a repository.
  6. +
+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Commit Changes
  2. +
  3. Backup Repository
  4. +
  5. Restore Repository
  6. +
+ +
+ +

Refer to Python Sample Notebooks to backup, migrate or clone a lakeFS repository

+

+ + + Commit Changes + + +

+ +

Backup process doesn’t backup uncommitted data, so make sure to commit any staged writes before running the backup. But this is an optional process.

+ +

You can manually commit the changes by using lakeFS UI or you can programatically commit any uncommitted changes. This Python example shows how to programatically loop through all branches in lakeFS and commit any uncommitted data but this might take a lot of time if you have many branches in the repo:

+ +
import lakefs
+
+repo = lakefs.Repository("example-repo")
+
+for branch in repo.branches():
+    for diff in repo.branch(branch.id).uncommitted():
+        repo.branch(branch.id).commit(message='Committed changes to backup the repository')
+        break
+
+

+ + + Backup Repository + + +

+ +

+ + + Dump Metadata + + +

+ +

Dump metadata/refs of the repository by using lakeFS API or CLI.

+ +
    +
  • Example code to dump metadata by using lakeFS Python SDK (this process will create _lakefs/refs_manifest.json file in your storage namespace for the repository):
  • +
+ +
lakefs_sdk_client.internal_api.dump_refs("example-repo")
+
+ +
    +
  • Example commands to dump metadata by using lakeFS CLI and upload to S3 storage for the repository:
  • +
+ +
lakectl refs-dump lakefs://example-repo > refs_manifest.json
+
+aws s3 cp refs_manifest.json s3://source-bucket-name/example-repo/_lakefs/refs_manifest.json
+
+ +
    +
  • Example commands to dump metadata by using lakeFS CLI and upload to Azure Blob storage for the repository:
  • +
+ +
lakectl refs-dump lakefs://example-repo > refs_manifest.json
+
+az storage blob upload --file refs_manifest.json --container-name sourceContainer --name example-repo/_lakefs/refs_manifest.json --account-name source-storage-account-name --account-key <source-storage-account-key>
+
+ +

Shutdown lakeFS services immediately after dumping the metadata so nobody can make any changes in the source repository.

+

+ + + Copy Data to Backup Storage Location + + +

+ +

Copy the repository’s storage namespace to another, preferably geographically separate location. Copy command depends on the type of object storage and the tool that you use.

+ +
    +
  • Example S3 command:
  • +
+ +
aws s3 sync s3://source-bucket-name/example-repo s3://target-bucket-name/example-repo
+
+ +
    +
  • Example Azure azcopy command:
  • +
+ +
azcopy copy 'https://source-storage-account-name.blob.core.windows.net/sourceContainer/example-repo/*?source_container_SAS_token' 'https://target-storage-account-name.blob.core.windows.net/targetContainer/example-repo?target_container_SAS_token' --recursive
+
+ +

You can restart lakeFS services after copying the data to backup storage location.

+

+ + + Restore Repository + + +

+ +

+ + + Create a new Bare Repository + + +

+ +

Create a bare lakeFS repository with a new name if you want to clone the repository or use the same repository name if you want to migrate or restore the repository.

+ +
    +
  • Python example to create a bare lakeFS repository using S3 storage:
  • +
+ +
lakefs.Repository("target-example-repo").create(bare=True, storage_namespace="s3://target-bucket-name/example-repo", default_branch="same-default-branch-as-in-source-repo")
+
+ +
    +
  • Python example to create a bare lakeFS repository using Azure storage:
  • +
+ +
lakefs.Repository("target-example-repo").create(bare=True, storage_namespace="https://target-storage-account-name.blob.core.windows.net/targetContainer/example-repo", default_branch="same-default-branch-as-in-source-repo")
+
+ +
    +
  • lakeFS CLI command to create a bare lakeFS repository using S3 storage:
  • +
+ +
lakectl repo create-bare lakefs://target-example-repo s3://target-bucket-name/example-repo --default-branch "same-default-branch-as-in-source-repo"
+
+ +
    +
  • lakeFS CLI command to create a bare lakeFS repository using Azure storage:
  • +
+ +
lakectl repo create-bare lakefs://target-example-repo https://target-storage-account-name.blob.core.windows.net/targetContainer/example-repo --default-branch "same-default-branch-as-in-source-repo"
+
+

+ + + Restore Metadata to new Repository + + +

+ +

Run restore_refs to load back all commits, tags and branches.

+ +
    +
  • Python example to restore metadata to new repository. First download metadata(refs_manifest.json) file created by metadata dump process:
  • +
+ +
aws s3 cp s3://target-bucket-name/example-repo/_lakefs/refs_manifest.json .
+
+ +
azcopy copy 'https://target-storage-account-name.blob.core.windows.net/targetContainer/example-repo/_lakefs/refs_manifest.json?<target_container_SAS_token>' .
+
+ +

Then read refs_manifest.json file and restore metadata to new repository:

+
with open('./refs_manifest.json') as file:
+    refs_manifest_json = json.load(file)
+    print(refs_manifest_json)
+    
+target_lakefs_sdk_client.internal_api.restore_refs(target_repo_name, refs_manifest_json)
+
+ +
    +
  • lakeFS CLI command to restore metadata to new repository using S3 storage:
  • +
+ +
aws s3 cp s3://target-bucket-name/example-repo/_lakefs/refs_manifest.json - | lakectl refs-restore lakefs://target-example-repo --manifest -
+
+ +
    +
  • lakeFS CLI command to restore metadata to new repository using Azure storage:
  • +
+ +
az storage blob download --container-name targetContainer --name example-repo/_lakefs/refs_manifest.json --account-name target-storage-account-name --account-key <target-storage-account-key> | lakectl refs-restore lakefs://target-example-repo --manifest -
+
+ +

Note: If you are running backups regularly, it is highly advised to test the restore process periodically to make sure that you are able to restore the repository in case of disaster.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/catalog_exports.html b/v1.46/howto/catalog_exports.html new file mode 100644 index 000000000..e49cdeef1 --- /dev/null +++ b/v1.46/howto/catalog_exports.html @@ -0,0 +1,938 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Data Catalogs Export | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Data Catalogs Export + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. About Data Catalogs Export
  2. +
  3. How it works
      +
    1. Table Decleration
    2. +
    3. Catalog Exporters
    4. +
    5. Flow
    6. +
    +
  4. +
+ +
+

+ + + About Data Catalogs Export + + +

+ + +

Data Catalog Export is all about integrating query engines (like Spark, AWS Athena, Presto, etc.) with lakeFS.

+ +

Data Catalogs (such as Hive Metastore or AWS Glue) store metadata for services (such as Spark, Trino and Athena). They contain metadata such as the location of the table, information about columns, partitions and much more.

+ +

With Data Catalog Exports, one can leverage the versioning capabilities of lakeFS in external data warehouses and query engines to access tables with branches and commits.

+ +

At the end of this guide, you will be able to query lakeFS data from Athena, Trino and other catalog-dependent tools:

+ +
USE main;
+USE my_branch; -- any branch
+USE v101; -- or tag
+
+SELECT * FROM users 
+INNER JOIN events 
+ON users.id = events.user_id; -- SQL stays the same, branch or tag exist as schema
+
+

+ + + How it works + + +

+ + +

Several well known formats exist today let you export existing tables in lakeFS into a “native” object store representation +which does not require copying the data outside of lakeFS.

+ +

These are metadata representations and can be applied automatically through hooks.

+

+ + + Table Decleration + + +

+ + +

After creating a lakeFS repository, configure tables as table descriptor objects on the repository on the path _lakefs_tables/TABLE.yaml. +Note: the Glue exporter can currently only export tables of type: hive. We expect to add more.

+

+ + + Hive tables + + +

+ + +

Hive metadata server tables are essentially just a set of objects that share a prefix, with no table metadata stored on the object store. You need to configure prefix, partitions, and schema.

+ +
name: animals
+type: hive
+path: path/to/animals/
+partition_columns: ['year']
+schema:
+  type: struct
+  fields:
+    - name: year
+      type: integer
+      nullable: false
+      metadata: {}
+    - name: page
+      type: string
+      nullable: false
+      metadata: {}
+    - name: site
+      type: string
+      nullable: true
+      metadata:
+        comment: a comment about this column
+
+ +

Useful types recognized by Hive include integer, long, short, string, double, float, date, and timestamp.

+

+ + + Catalog Exporters + + +

+ + +

Exporters are code packages accessible through Lua integration. Each exporter is exposed as a Lua function under the package namespace lakefs/catalogexport. Call them from hooks to connect lakeFS tables to various catalogs.

+

+ + + Currently supported exporters + + +

+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ExporterDescriptionNotes
Symlink exporterWrites metadata for the table using Hive’s SymlinkTextInputFormat 
AWS Glue Catalog (+ Athena) exporterCreates a table in Glue using Hive’s format and updates the location to symlink files (reuses Symlink Exporter).See a step-by-step guide on how to integrate with Glue Exporter
Delta Lake table exporterExport Delta Lake tables from lakeFS to an external storage 
Unity Catalog exporterThe Unity Catalog exporter serves the purpose of registering a Delta Lake table in Unity Catalog. It operates in conjunction with the Delta Lake exporter. In this workflow, the Delta Lake exporter is utilized to export a Delta Lake table from lakeFS. Subsequently, the obtained result is passed to the Unity Catalog exporter to facilitate its registration within Unity Catalog.See a step-by-step guide on how to integrate with Unity Catalog Exporter</br>Currently, only AWS S3 storage is supported
+

+ + + Running an Exporter + + +

+ + +

Exporters are meant to run as Lua hooks.

+ +

Configure the actions trigger by using events and branches. Of course, you can add additional custom filtering logic to the Lua script if needed. +The default table name when exported is ${repository_id}_${_lakefs_tables/TABLE.md(name field)}_${ref_name}_${short_commit}.

+ +

Example of an action that will be triggered when a post-commit event happens in the export_table branch.

+ +
name: Glue Table Exporter
+description: export my table to glue  
+on:
+  post-commit:
+    branches: ["export_table"]
+hooks:
+  - id: my_exporter
+    type: lua
+    properties:
+      # exporter script location
+      script_path: "scripts/my_export_script.lua"
+      args:
+        # table descriptor
+        table_source: '_lakefs_tables/my_table.yaml'
+
+ +

Tip: Actions can be extended to customize any desired behavior, for example validating branch names since they are part of the table name:

+ +
# _lakefs_actions/validate_branch_name.yaml
+name: validate-lower-case-branches 
+on:
+  pre-create-branch:
+hooks:
+  - id: check_branch_id
+    type: lua
+    properties:
+      script: |
+        regexp = require("regexp")
+        if not regexp.match("^[a-z0-9\\_\\-]+$", action.branch_id) then
+          error("branches must be lower case, invalid branch ID: " .. action.branch_id)
+        end
+
+

+ + + Flow + + +

+ + +

The following diagram demonstrates what happens when a lakeFS Action triggers runs a lua hook that calls an exporter.

+ +
sequenceDiagram
+    note over Lua Hook: lakeFS Action trigger. <br> Pass Context for the export.
+    Lua Hook->>Exporter: export request
+    note over Table Registry: _lakefs_tables/TABLE.yaml
+    Exporter->>Table Registry: Get table descriptor
+    Table Registry->>Exporter: Parse table structure
+    Exporter->>Object Store: materialize an exported table
+    Exporter->>Catalog: register object store location
+    Query Engine-->Catalog: Query
+    Query Engine-->Object Store: Query
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/copying.html b/v1.46/howto/copying.html new file mode 100644 index 000000000..6d15771d3 --- /dev/null +++ b/v1.46/howto/copying.html @@ -0,0 +1,929 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Copying data to/from lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Copying data to/from lakeFS + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Using DistCp
  2. +
  3. Using Rclone
  4. +
+ +
+

+ + + Using DistCp + + +

+ + +

Apache Hadoop DistCp (distributed copy) is a tool used for large inter/intra-cluster copying. You can easily use it with your lakeFS repositories.

+ +

Note

+ +

In the following examples, we set AWS credentials on the command line for clarity. In production, you should set these properties using one of Hadoop’s standard ways of Authenticating with S3.

+

+ + + Between lakeFS repositories + + +

+ + +

You can use DistCP to copy between two different lakeFS repositories. Replace the access key pair with your lakeFS access key pair:

+ +
hadoop distcp \
+  -Dfs.s3a.path.style.access=true \
+  -Dfs.s3a.access.key="AKIAIOSFODNN7EXAMPLE" \
+  -Dfs.s3a.secret.key="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" \
+  -Dfs.s3a.endpoint="https://lakefs.example.com" \
+  "s3a://example-repo-1/main/example-file.parquet" \
+  "s3a://example-repo-2/main/example-file.parquet"
+
+

+ + + Between S3 buckets and lakeFS + + +

+ + +

To copy data from an S3 bucket to a lakeFS repository, use Hadoop’s per-bucket configuration. +In the following examples, replace the first access key pair with your lakeFS key pair, and the second one with your AWS IAM key pair:

+

+ + + From S3 to lakeFs + + +

+ + +
hadoop distcp \
+  -Dfs.s3a.path.style.access=true \
+  -Dfs.s3a.bucket.example-repo.access.key="AKIAIOSFODNN7EXAMPLE" \
+  -Dfs.s3a.bucket.example-repo.secret.key="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" \
+  -Dfs.s3a.bucket.example-repo.endpoint="https://lakefs.example.com" \
+  -Dfs.s3a.bucket.example-bucket.access.key="AKIAIOSFODNN3EXAMPLE" \
+  -Dfs.s3a.bucket.example-bucket.secret.key="wJalrXUtnFEMI/K3MDENG/bPxRfiCYEXAMPLEKEY" \
+  "s3a://example-bucket/example-file.parquet" \
+  "s3a://example-repo/main/example-file.parquet"
+
+

+ + + From lakeFS to S3 + + +

+ + +
hadoop distcp \
+  -Dfs.s3a.path.style.access=true \
+  -Dfs.s3a.bucket.example-repo.access.key="AKIAIOSFODNN7EXAMPLE" \
+  -Dfs.s3a.bucket.example-repo.secret.key="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY" \
+  -Dfs.s3a.bucket.example-repo.endpoint="https://lakefs.example.com" \
+  -Dfs.s3a.bucket.example-bucket.access.key="AKIAIOSFODNN3EXAMPLE" \
+  -Dfs.s3a.bucket.example-bucket.secret.key="wJalrXUtnFEMI/K3MDENG/bPxRfiCYEXAMPLEKEY" \
+  "s3a://example-repo/main/myfile" \
+  "s3a://example-bucket/myfile"
+
+

+ + + Using Rclone + + +

+ + +

Rclone is a command line program to sync files and directories between cloud providers. +To use it with lakeFS, create an Rclone remote as describe below and then use it as you would any other Rclone remote.

+

+ + + Creating a remote for lakeFS in Rclone + + +

+ + +

To add the remote to Rclone, choose one of the following options:

+

+ + + Option 1: Add an entry in your Rclone configuration file + + +

+ +
    +
  • +

    Find the path to your Rclone configuration file and copy it for the next step.

    + +
    rclone config file
    +# output:
    +# Configuration file is stored at:
    +# /home/myuser/.config/rclone/rclone.conf
    +
    +
  • +
  • +

    If your lakeFS access key is already set in an AWS profile or environment variables, run the following command, replacing the endpoint property with your lakeFS endpoint:

    + +
    cat <<EOT >> /home/myuser/.config/rclone/rclone.conf
    +[lakefs]
    +type = s3
    +provider = Other
    +endpoint = https://lakefs.example.com
    +no_check_bucket = true
    +EOT
    +
    +
  • +
  • +

    Otherwise, also include your lakeFS access key pair in the Rclone configuration file:

    + +
    cat <<EOT >> /home/myuser/.config/rclone/rclone.conf
    +[lakefs]
    +type = s3
    +provider = Other
    +env_auth = false
    +access_key_id = AKIAIOSFODNN7EXAMPLE
    +secret_access_key = wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
    +endpoint = https://lakefs.example.com
    +no_check_bucket = true
    +EOT
    +
    +
  • +
+

+ + + Option 2: Use the Rclone interactive config command + + +

+ + +

Run this command and follow the instructions:

+
rclone config
+
+

Choose AWS S3 as your type of storage, and enter your lakeFS endpoint as your S3 endpoint. +You will have to choose whether you use your environment for authentication (recommended), +or enter the lakeFS access key pair into the Rclone configuration. Select “Edit advanced +config” and accept defaults for all values except no_check_bucket:

+
If set, don't attempt to check the bucket exists or create it.
+
+This can be useful when trying to minimize the number of transactions
+Rclone carries out, if you know the bucket exists already.
+
+This might also be needed if the user you're using doesn't have bucket
+creation permissions. Before v1.52.0, this would have passed silently
+due to a bug.
+
+Enter a boolean value (true or false). Press Enter for the default ("false").
+no_check_bucket> yes
+
+

+ + + Syncing S3 and lakeFS + + +

+ + +
rclone sync mys3remote://mybucket/path/ lakefs:example-repo/main/path
+
+

+ + + Syncing a local directory and lakeFS + + +

+ + +
rclone sync /home/myuser/path/ lakefs:example-repo/main/path
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/aws.html b/v1.46/howto/deploy/aws.html new file mode 100644 index 000000000..37f9cb28f --- /dev/null +++ b/v1.46/howto/deploy/aws.html @@ -0,0 +1,1214 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +AWS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Deploy lakeFS on AWS + + +

+ + +
+

The instructions given here are for a self-managed deployment of lakeFS on AWS.

+ +

For a hosted lakeFS service with guaranteed SLAs, try lakeFS Cloud

+
+ +

When you deploy lakeFS on AWS these are the options available to use:

+ +

+ +

This guide walks you through the options available and how to configure them, finishing with configuring and running lakeFS itself and creating your first repository.

+ + + +

⏰ Expected deployment time: 25 min

+

+ + + Grant lakeFS permissions to DynamoDB + + +

+ + +

By default, lakeFS will create the required DynamoDB table if it does not already exist. You’ll have to give the IAM role used by lakeFS the following permissions:

+ +
{
+    "Version": "2012-10-17",
+    "Statement": [
+        {
+            "Sid": "ListAndDescribe",
+            "Effect": "Allow",
+            "Action": [
+                "dynamodb:List*",
+                "dynamodb:DescribeReservedCapacity*",
+                "dynamodb:DescribeLimits",
+                "dynamodb:DescribeTimeToLive"
+            ],
+            "Resource": "*"
+        },
+        {
+            "Sid": "kvstore",
+            "Effect": "Allow",
+            "Action": [
+                "dynamodb:BatchGet*",
+                "dynamodb:DescribeTable",
+                "dynamodb:Get*",
+                "dynamodb:Query",
+                "dynamodb:Scan",
+                "dynamodb:BatchWrite*",
+                "dynamodb:CreateTable",
+                "dynamodb:Delete*",
+                "dynamodb:Update*",
+                "dynamodb:PutItem"
+            ],
+            "Resource": "arn:aws:dynamodb:*:*:table/kvstore"
+        }
+    ]
+}
+
+ +

💡 You can also use lakeFS with PostgreSQL instead of DynamoDB! See the configuration reference for more information.

+

+ + + Run the lakeFS server + + +

+ + +
+ +
+ +

Connect to your EC2 instance using SSH:

+ +
    +
  1. +

    Create a config.yaml on your EC2 instance, with the following parameters:

    + +
    ---
    +database:
    +  type: "dynamodb"
    +  
    +auth:
    +  encrypt:
    +    # replace this with a randomly-generated string. Make sure to keep it safe!
    +    secret_key: "[ENCRYPTION_SECRET_KEY]"
    +   
    +blockstore:
    +  type: s3
    +
    +
  2. +
  3. Download the binary to run on the EC2 instance.
  4. +
  5. +

    Run the lakefs binary on the EC2 instance:

    + +
    lakefs --config config.yaml run
    +
    +
  6. +
+ +

Note: It’s preferable to run the binary as a service using systemd or your operating system’s facilities.

+

+ + + Advanced: Deploying lakeFS behind an AWS Application Load Balancer + + +

+ + +
    +
  1. Your security groups should allow the load balancer to access the lakeFS server.
  2. +
  3. Create a target group with a listener for port 8000.
  4. +
  5. Setup TLS termination using the domain names you wish to use (e.g., lakefs.example.com and potentially s3.lakefs.example.com, *.s3.lakefs.example.com if using virtual-host addressing).
  6. +
  7. Configure the health-check to use the exposed /_health URL
  8. +
+ +
+
+ +

You can install lakeFS on Kubernetes using a Helm chart.

+ +

To install lakeFS with Helm:

+ +
    +
  1. +

    Copy the Helm values file relevant for S3:

    + +
    secrets:
    +   # replace this with a randomly-generated string
    +   authEncryptSecretKey: [ENCRYPTION_SECRET_KEY]
    +lakefsConfig: |
    +    database:
    +      type: dynamodb
    +    blockstore:
    +      type: s3
    +
    +
  2. +
  3. +

    Fill in the missing values and save the file as conf-values.yaml. For more configuration options, see our Helm chart README.

    + +

    The lakefsConfig parameter is the lakeFS configuration documented here but without sensitive information. +Sensitive information like databaseConnectionString is given through separate parameters, and the chart will inject it into Kubernetes secrets.

    +
  4. +
  5. +

    In the directory where you created conf-values.yaml, run the following commands:

    + +
    # Add the lakeFS repository
    +helm repo add lakefs https://charts.lakefs.io
    +# Deploy lakeFS
    +helm install my-lakefs lakefs/lakefs -f conf-values.yaml
    +
    + +

    my-lakefs is the Helm Release name.

    +
  6. +
+ +

⚠️ Make sure the Kubernetes nodes have access to all buckets/containers with which you intend to use with lakeFS. +If you can’t provide such access, configure lakeFS with an AWS key-pair.

+

+ + + Load balancing + + +

+ + +

To configure a load balancer to direct requests to the lakeFS servers you can use the LoadBalancer Service type or a Kubernetes Ingress. +By default, lakeFS operates on port 8000 and exposes a /_health endpoint that you can use for health checks.

+ +

💡 The NGINX Ingress Controller by default limits the client body size to 1 MiB. +Some clients use bigger chunks to upload objects - for example, multipart upload to lakeFS using the S3 Gateway or +a simple PUT request using the OpenAPI Server. +Checkout Nginx documentation for increasing the limit, or an example of Nginx configuration with MinIO.

+ +
+
+

+ + + Prepare your S3 bucket + + +

+ + +
    +
  1. Take note of the bucket name you want to use with lakeFS
  2. +
  3. +

    Use the following as your bucket policy, filling in the placeholders:

    + +
    + +
    + +
    {
    +   "Id": "lakeFSPolicy",
    +   "Version": "2012-10-17",
    +   "Statement": [
    +      {
    +         "Sid": "lakeFSObjects",
    +         "Action": [
    +            "s3:GetObject",
    +            "s3:PutObject",
    +            "s3:AbortMultipartUpload",
    +            "s3:ListMultipartUploadParts"
    +         ],
    +         "Effect": "Allow",
    +         "Resource": ["arn:aws:s3:::[BUCKET_NAME_AND_PREFIX]/*"],
    +         "Principal": {
    +            "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +         }
    +      },
    +      {
    +         "Sid": "lakeFSBucket",
    +         "Action": [
    +            "s3:ListBucket",
    +            "s3:GetBucketLocation",
    +            "s3:ListBucketMultipartUploads"
    +         ],
    +         "Effect": "Allow",
    +         "Resource": ["arn:aws:s3:::[BUCKET]"],
    +         "Principal": {
    +            "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +         }
    +      }
    +   ]
    +}
    +
    + +
      +
    • Replace [BUCKET_NAME], [ACCOUNT_ID] and [IAM_ROLE] with values relevant to your environment.
    • +
    • [BUCKET_NAME_AND_PREFIX] can be the bucket name. If you want to minimize the bucket policy permissions, use the bucket name together with a prefix (e.g. example-bucket/a/b/c). +This way, lakeFS will be able to create repositories only under this specific path (see: Storage Namespace).
    • +
    • lakeFS will try to assume the role [IAM_ROLE].
    • +
    +
    +
    + +

    To use an S3 Express One Zone directory bucket, use the following policy. Note the lakeFSDirectoryBucket statement which is specifically required for using a directory bucket.

    + +
    {
    +   "Id": "lakeFSPolicy",
    +   "Version": "2012-10-17",
    +   "Statement": [
    +      {
    +         "Sid": "lakeFSObjects",
    +         "Action": [
    +            "s3:GetObject",
    +            "s3:PutObject",
    +            "s3:AbortMultipartUpload",
    +            "s3:ListMultipartUploadParts"
    +         ],
    +         "Effect": "Allow",
    +         "Resource": ["arn:aws:s3:::[BUCKET_NAME_AND_PREFIX]/*"],
    +         "Principal": {
    +            "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +         }
    +      },
    +      {
    +         "Sid": "lakeFSBucket",
    +         "Action": [
    +            "s3:ListBucket",
    +            "s3:GetBucketLocation",
    +            "s3:ListBucketMultipartUploads"
    +         ],
    +         "Effect": "Allow",
    +         "Resource": ["arn:aws:s3:::[BUCKET]"],
    +         "Principal": {
    +            "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +         }
    +      },
    +      {
    +         "Sid": "lakeFSDirectoryBucket",
    +         "Action": [
    +            "s3express:CreateSession"
    +         ],
    +         "Effect": "Allow",
    +         "Resource": "arn:aws:s3express:[REGION]:[ACCOUNT_ID]:bucket/[BUCKET_NAME]"
    +      }
    +   ]
    +}
    +
    + +
      +
    • Replace [BUCKET_NAME], [ACCOUNT_ID] and [IAM_ROLE] with values relevant to your environment.
    • +
    • [BUCKET_NAME_AND_PREFIX] can be the bucket name. If you want to minimize the bucket policy permissions, use the bucket name together with a prefix (e.g. example-bucket/a/b/c). +This way, lakeFS will be able to create repositories only under this specific path (see: Storage Namespace).
    • +
    • lakeFS will try to assume the role [IAM_ROLE].
    • +
    +
    +
    +

    If required lakeFS can operate without accessing the data itself, this permission section is useful if you are using presigned URLs mode or the lakeFS Hadoop FileSystem Spark integration. +Since this FileSystem performs many operations directly on the storage, lakeFS requires less permissive permissions, resulting in increased security.

    + +

    lakeFS always requires permissions to access the _lakefs prefix under your storage namespace, in which metadata +is stored (learn more).

    + +

    By setting this policy without presign mode you’ll be able to perform only metadata operations through lakeFS, meaning that you’ll not be able +to use lakeFS to upload or download objects. Specifically you won’t be able to:

    +
      +
    • Upload objects using the lakeFS GUI (Works with presign mode)
    • +
    • Upload objects through Spark using the S3 gateway
    • +
    • Run lakectl fs commands (unless using presign mode with --pre-sign flag)
    • +
    • Use Actions and Hooks
    • +
    + +
    {
    +  "Id": "[POLICY_ID]",
    +  "Version": "2012-10-17",
    +  "Statement": [
    +{
    +  "Sid": "lakeFSObjects",
    +  "Action": [
    +     "s3:GetObject",
    +     "s3:PutObject"
    +  ],
    +  "Effect": "Allow",
    +  "Resource": [
    +     "arn:aws:s3:::[STORAGE_NAMESPACE]/_lakefs/*"
    +  ],
    +  "Principal": {
    +    "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +  }
    +},
    + {
    +    "Sid": "lakeFSBucket",
    +    "Action": [
    +       "s3:ListBucket",
    +       "s3:GetBucketLocation"
    +    ],
    +    "Effect": "Allow",
    +    "Resource": ["arn:aws:s3:::[BUCKET]"],
    +    "Principal": {
    +       "AWS": ["arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"]
    +    }
    + }
    +  ]
    +}
    +
    + +

    We can use presigned URLs mode without allowing access to the data from the lakeFS server directly. +We can achieve this by using condition keys such as aws:referer, aws:SourceVpc and aws:SourceIp.

    + +

    For example, assume the following scenario:

    +
      +
    • lakeFS is deployed outside the company (i.e lakeFS cloud or other VPC not vpc-123)
    • +
    • We don’t want lakeFS to be able to access the data, so we use presign URL, we still need lakeFS role to be able to sign the URL.
    • +
    • We want to allow access from the internal company VPC: vpc-123.
    • +
    + +
          {
    +         "Sid": "allowLakeFSRoleFromCompanyOnly",
    +         "Effect": "Allow",
    +         "Principal": {
    +             "AWS": "arn:aws:iam::[ACCOUNT_ID]:role/[IAM_ROLE]"
    +         },
    +         "Action": [
    +             "s3:GetObject",
    +             "s3:PutObject",
    +         ],
    +         "Resource": [
    +            "arn:aws:s3:::[BUCKET]/*",
    +         ],
    +         "Condition": {
    +             "StringEquals": {
    +                 "aws:SourceVpc": "vpc-123"
    +             }
    +         }
    +     }
    +
    + +
    +
    +
  4. +
+

+ + + S3 Storage Tier Classes + + +

+ + +

lakeFS currently supports the following S3 Storage Classes:

+ +
    +
  1. S3 Standard - The default AWS S3 storage tier. Fully supported.
  2. +
  3. S3 Express One-Zone - Fully supported.
  4. +
  5. S3 Glacier Instant Retrival - Supported with limitations: currently, pre-signed URLs are not supported when using Instant Retrival. The outstanding feature request could be tracked here.
  6. +
+ +

Other storage classes are currently unsupported - either because they have not been tested with lakeFS or because they cannot be supported.

+ +

If you need lakeFS to support a storage tier that isn’t currently on the supported list, please open an issue on GitHub.

+

+ + + Alternative: use an AWS user + + +

+ + +

lakeFS can authenticate with your AWS account using an AWS user, using an access key and secret. To allow this, change the policy’s Principal accordingly:

+
 "Principal": {
+   "AWS": ["arn:aws:iam::<ACCOUNT_ID>:user/<IAM_USER>"]
+ }
+
+

+ + + Create the admin user + + +

+ + +

When you first open the lakeFS UI, you will be asked to create an initial admin user.

+ +
    +
  1. Open http://<lakefs-host>/ in your browser. If you haven’t set up a load balancer, this will likely be http://<instance ip address>:8000/
  2. +
  3. +

    On first use, you’ll be redirected to the setup page:

    + +

    Create user

    +
  4. +
  5. +

    Follow the steps to create an initial administrator user. Save the credentials you’ve received somewhere safe, you won’t be able to see them again!

    + +

    Setup Done

    +
  6. +
  7. Follow the link and go to the login screen. Use the credentials from the previous step to log in.
  8. +
+

+ + + Create your first repository + + +

+ + +
    +
  1. Use the credentials from the previous step to log in
  2. +
  3. +

    Click Create Repository and choose Blank Repository.

    + +

    Create Repo

    +
  4. +
  5. Under Storage Namespace, enter a path to your desired location on the object store. This is where data written to this repository will be stored.
  6. +
  7. Click Create Repository
  8. +
  9. +

    You should now have a configured repository, ready to use!

    + +

    Repo Created

    +
  10. +
+ +

Congratulations! Your environment is now ready 🤩

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/azure.html b/v1.46/howto/deploy/azure.html new file mode 100644 index 000000000..4dc1745d3 --- /dev/null +++ b/v1.46/howto/deploy/azure.html @@ -0,0 +1,1123 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Azure | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Deploy lakeFS on Azure + + +

+ + +
+

The instructions given here are for a self-managed deployment of lakeFS on Azure.

+ +

For a hosted lakeFS service with guaranteed SLAs, try lakeFS Cloud

+
+ +

When you deploy lakeFS on Azure these are the options available to use:

+ +

+ +

This guide walks you through the options available and how to configure them, finishing with configuring and running lakeFS itself and creating your first repository.

+ + + +

⏰ Expected deployment time: 25 min

+

+ + + 1. Object Storage + + +

+ + +

lakeFS supports the following Azure Storage types:

+ +
    +
  1. Azure Blob Storage
  2. +
  3. Azure Data Lake Storage Gen2 (HNS)
  4. +
+ +

Data Lake Storage Gen1 is not supported.

+

+ + + 2. Authentication Method + + +

+ + +

lakeFS supports two ways to authenticate with Azure.

+ +
+ + +
+ +

lakeFS uses environment variables to determine credentials to use for authentication. The following authentication methods are supported:

+ +
    +
  1. Managed Service Identity (MSI)
  2. +
  3. Service Principal RBAC
  4. +
  5. Azure CLI
  6. +
+ +

For deployments inside the Azure ecosystem it is recommended to use a managed identity.

+ +

More information on authentication methods and environment variables can be found here

+

+ + + How to Create Service Principal for Resource Group + + +

+ + +

It is recommended to create a resource group that consists of all the resources lakeFS should have access to.

+ +

Using a resource group will allow dynamic removal/addition of services from the group, effectively providing/preventing access for lakeFS to these resources without requiring any changes in configuration in lakeFS or providing lakeFS with any additional credentials.

+ +

The minimal role required for the service principal is “Storage Blob Data Contributor”

+ +

The following Azure CLI command creates a service principal for a resource group called “lakeFS” with permission to access (read/write/delete) +Blob Storage resources in the resource group and with an expiry of 5 years

+ +
az ad sp create-for-rbac \
+  --role "Storage Blob Data Contributor" \
+  --scopes /subscriptions/XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX/resourceGroups/lakeFS --years 5
+    
+Creating 'Storage Blob Data Contributor' role assignment under scope '/subscriptions/947382ea-681a-4541-99ab-b718960c6289/resourceGroups/lakeFS'
+The output includes credentials that you must protect. Be sure that you do not include these credentials in your code or check the credentials into your source control. For more information, see https://aka.ms/azadsp-cli
+{
+  "appId": "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX",
+  "displayName": "azure-cli-2023-01-30-06-18-30",
+  "password": "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX",
+  "tenant": "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX"
+}
+
+ +

The command output should be used to populate the following environment variables:

+ +
AZURE_CLIENT_ID      =  $appId
+AZURE_TENANT_ID      =  $tenant
+AZURE_CLIENT_SECRET  =  $password
+
+ +

Note: Service Principal credentials have an expiry date and lakeFS will lose access to resources unless credentials are renewed on time.

+ +

Note: It is possible to provide both account based credentials and environment variables to lakeFS. In that case - lakeFS will use +the account credentials for any access to data located in the given account, and will try to use the identity credentials for any data located outside the given account.

+ +
+ +
+ +

Storage account credentials can be set directly in the lakeFS configuration using the following parameters:

+ +
    +
  • blockstore.azure.storage_account
  • +
  • blockstore.azure.storage_access_key
  • +
+

+ + + Limitations + + +

+ + +

Please note that using this authentication method limits lakeFS to the scope of the given storage account.

+ +

Specifically, the following operations will not work:

+ +
    +
  1. Import of data from different storage accounts
  2. +
  3. Copy/Read/Write of data that was imported from a different storage account
  4. +
  5. Create pre-signed URL for data that was imported from a different storage account
  6. +
+ +
+
+

+ + + 3. K/V Store + + +

+ + +

lakeFS stores metadata in a database for its versioning engine. This is done via a Key-Value interface that can be implemented on any DB engine and lakeFS comes with several built-in driver implementations (You can read more about it here). The database used doesn’t have to be a dedicated K/V database.

+ +
+ +
+ +

CosmosDB is a managed database service provided by Azure. lakeFS supports CosmosDB For NoSQL as a database backend.

+ +
    +
  1. Follow the official Azure documentation +on how to create a CosmosDB account for NoSQL and connect to it.
  2. +
  3. Once your CosmosDB account is set up, you can create a Database for +lakeFS. For lakeFS ACID guarantees, make sure to select the Bounded +staleness consistency, +for single region deployments.
  4. +
  5. Create a new container in the database and select type +partitionKey as the Partition key (case sensitive).
  6. +
  7. Pass the endpoint, database name and container name to lakeFS as +described in the configuration guide. +You can either pass the CosmosDB’s account read-write key to lakeFS, or +use a managed identity to authenticate to CosmosDB, as described +earlier.
  8. +
+ +

A note on CosmosDB capacity modes: lakeFS usage of CosmosDB is still in its +early days and has not been battle tested. Both capacity modes, Provisioned +and Serverless, has been tested for some workloads and passed with flying +colors. The Provisioned mode was configured with 400-4000 RU/s.

+ +
+
+ +

Below we show you how to create a database on Azure Database, but you can use any PostgreSQL database as long as it’s accessible by your lakeFS installation.

+ +

If you already have a database, take note of the connection string and skip to the next step

+ +
    +
  1. Follow the official Azure documentation on how to create a PostgreSQL instance and connect to it. +Make sure that you’re using PostgreSQL version >= 11.
  2. +
  3. Once your Azure Database for PostgreSQL server is set up and the server is in the Available state, take note of the endpoint and username. +Azure postgres Connection String
  4. +
  5. Make sure your Access control roles allow you to connect to the database instance.
  6. +
+ +
+
+

+ + + 4. Run the lakeFS server + + +

+ + +

Now that you’ve chosen and configured object storage, a K/V store, and authentication—you’re ready to configure and run lakeFS. There are three different ways you can run lakeFS:

+ +
+ +
+ +

Connect to your VM instance using SSH:

+ +
    +
  1. +

    Create a config.yaml on your VM, with the following parameters:

    + +
    ---
    +database:
    +  type: "postgres"
    +  postgres:
    +    connection_string: "[DATABASE_CONNECTION_STRING]"
    +  
    +auth:
    +  encrypt:
    +    # replace this with a randomly-generated string. Make sure to keep it safe!
    +    secret_key: "[ENCRYPTION_SECRET_KEY]"
    +   
    +blockstore:
    +  type: azure
    +  azure:
    +
    +
  2. +
  3. Download the binary to run on the VM.
  4. +
  5. +

    Run the lakefs binary:

    + +
    lakefs --config config.yaml run
    +
    +
  6. +
+ +

Note: It’s preferable to run the binary as a service using systemd or your operating system’s facilities.

+ +
+
+ +

To support container-based environments, you can configure lakeFS using environment variables. Here is a docker run +command to demonstrate starting lakeFS using Docker:

+ +
docker run \
+  --name lakefs \
+  -p 8000:8000 \
+  -e LAKEFS_DATABASE_TYPE="postgres" \
+  -e LAKEFS_DATABASE_POSTGRES_CONNECTION_STRING="[DATABASE_CONNECTION_STRING]" \
+  -e LAKEFS_AUTH_ENCRYPT_SECRET_KEY="[ENCRYPTION_SECRET_KEY]" \
+  -e LAKEFS_BLOCKSTORE_TYPE="azure" \
+  -e LAKEFS_BLOCKSTORE_AZURE_STORAGE_ACCOUNT="[YOUR_STORAGE_ACCOUNT]" \
+  -e LAKEFS_BLOCKSTORE_AZURE_STORAGE_ACCESS_KEY="[YOUR_ACCESS_KEY]" \
+  treeverse/lakefs:latest run
+
+ +

See the reference for a complete list of environment variables.

+ +
+
+ +

You can install lakeFS on Kubernetes using a Helm chart.

+ +

To install lakeFS with Helm:

+ +
    +
  1. +

    Copy the Helm values file relevant for Azure Blob:

    + +
    secrets:
    +    # replace this with the connection string of the database you created in a previous step:
    +    databaseConnectionString: [DATABASE_CONNECTION_STRING]
    +    # replace this with a randomly-generated string
    +    authEncryptSecretKey: [ENCRYPTION_SECRET_KEY]
    +lakefsConfig: |
    +    blockstore:
    +      type: azure
    +      azure:
    +    #  If you chose to authenticate via access key, unmark the following rows and insert the values from the previous step 
    +    #  storage_account: [your storage account]
    +    #  storage_access_key: [your access key]
    +
    +
  2. +
  3. +

    Fill in the missing values and save the file as conf-values.yaml. For more configuration options, see our Helm chart README.

    + +

    The lakefsConfig parameter is the lakeFS configuration documented here but without sensitive information. +Sensitive information like databaseConnectionString is given through separate parameters, and the chart will inject it into Kubernetes secrets.

    +
  4. +
  5. +

    In the directory where you created conf-values.yaml, run the following commands:

    + +
    # Add the lakeFS repository
    +helm repo add lakefs https://charts.lakefs.io
    +# Deploy lakeFS
    +helm install my-lakefs lakefs/lakefs -f conf-values.yaml
    +
    + +

    my-lakefs is the Helm Release name.

    +
  6. +
+

+ + + Load balancing + + +

+ + +

To configure a load balancer to direct requests to the lakeFS servers you can use the LoadBalancer Service type or a Kubernetes Ingress. +By default, lakeFS operates on port 8000 and exposes a /_health endpoint that you can use for health checks.

+ +

💡 The NGINX Ingress Controller by default limits the client body size to 1 MiB. +Some clients use bigger chunks to upload objects - for example, multipart upload to lakeFS using the S3-compatible Gateway or +a simple PUT request using the OpenAPI Server. +Check out Nginx documentation for increasing the limit, or an example of Nginx configuration with MinIO.

+ +
+
+

+ + + Create the admin user + + +

+ + +

When you first open the lakeFS UI, you will be asked to create an initial admin user.

+ +
    +
  1. Open http://<lakefs-host>/ in your browser. If you haven’t set up a load balancer, this will likely be http://<instance ip address>:8000/
  2. +
  3. +

    On first use, you’ll be redirected to the setup page:

    + +

    Create user

    +
  4. +
  5. +

    Follow the steps to create an initial administrator user. Save the credentials you’ve received somewhere safe, you won’t be able to see them again!

    + +

    Setup Done

    +
  6. +
  7. Follow the link and go to the login screen. Use the credentials from the previous step to log in.
  8. +
+

+ + + Create your first repository + + +

+ + +
    +
  1. Use the credentials from the previous step to log in
  2. +
  3. +

    Click Create Repository and choose Blank Repository.

    + +

    Create Repo

    +
  4. +
  5. Under Storage Namespace, enter a path to your desired location on the object store. This is where data written to this repository will be stored.
  6. +
  7. Click Create Repository
  8. +
  9. +

    You should now have a configured repository, ready to use!

    + +

    Repo Created

    +
  10. +
+ +

Congratulations! Your environment is now ready 🤩

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/gcp.html b/v1.46/howto/deploy/gcp.html new file mode 100644 index 000000000..0901a4836 --- /dev/null +++ b/v1.46/howto/deploy/gcp.html @@ -0,0 +1,952 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +GCP | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Deploy lakeFS on GCP + + +

+ + +
+

The instructions given here are for a self-managed deployment of lakeFS on GCP.

+ +

For a hosted lakeFS service with guaranteed SLAs, please contact us for details of lakeFS Cloud on GCP.

+
+ +

When you deploy lakeFS on GCP these are the options available to use:

+ +

+ + + +

⏰ Expected deployment time: 25 min

+

+ + + Create a Database + + +

+ + +

lakeFS requires a PostgreSQL database to synchronize actions on your repositories. +We will show you how to create a database on Google Cloud SQL, but you can use any PostgreSQL database as long as it’s accessible by your lakeFS installation.

+ +

If you already have a database, take note of the connection string and skip to the next step

+ +
    +
  1. Follow the official Google documentation on how to create a PostgreSQL instance. +Make sure you’re using PostgreSQL version >= 11.
  2. +
  3. On the Users tab in the console, create a user. The lakeFS installation will use it to connect to your database.
  4. +
  5. Choose the method by which lakeFS will connect to your database. Google recommends using +the SQL Auth Proxy.
  6. +
+

+ + + Run the lakeFS Server + + +

+ + +
+ +
+ +
    +
  1. +

    Save the following configuration file as config.yaml:

    + +
    ---
    +database:
    +  type: "postgres"
    +  postgres:
    +    connection_string: "[DATABASE_CONNECTION_STRING]"
    +auth:
    +  encrypt:
    +    # replace this with a randomly-generated string:
    +    secret_key: "[ENCRYPTION_SECRET_KEY]"
    +blockstore:
    +  type: gs
    +   # Uncomment the following lines to give lakeFS access to your buckets using a service account:
    +   # gs:
    +   #   credentials_json: [YOUR SERVICE ACCOUNT JSON STRING]
    +
    +
  2. +
  3. Download the binary to run on the GCE instance.
  4. +
  5. Run the lakefs binary on the GCE machine: +
    lakefs --config config.yaml run
    +
    +

    Note: it is preferable to run the binary as a service using systemd or your operating system’s facilities.

    +
  6. +
+ +
+
+ +

To support container-based environments like Google Cloud Run, lakeFS can be configured using environment variables. Here is a docker run +command to demonstrate starting lakeFS using Docker:

+ +
docker run \
+  --name lakefs \
+  -p 8000:8000 \
+  -e LAKEFS_DATABASE_TYPE="postgres" \
+  -e LAKEFS_DATABASE_POSTGRES_CONNECTION_STRING="[DATABASE_CONNECTION_STRING]" \
+  -e LAKEFS_AUTH_ENCRYPT_SECRET_KEY="[ENCRYPTION_SECRET_KEY]" \
+  -e LAKEFS_BLOCKSTORE_TYPE="gs" \
+  treeverse/lakefs:latest run
+
+ +

See the reference for a complete list of environment variables.

+ +
+
+ +

You can install lakeFS on Kubernetes using a Helm chart.

+ +

To install lakeFS with Helm:

+ +
    +
  1. +

    Copy the Helm values file relevant for Google Storage:

    + +
    secrets:
    +    # replace DATABASE_CONNECTION_STRING with the connection string of the database you created in a previous step.
    +    # e.g.: postgres://postgres:myPassword@localhost/postgres:5432
    +    databaseConnectionString: [DATABASE_CONNECTION_STRING]
    +    # replace this with a randomly-generated string
    +    authEncryptSecretKey: [ENCRYPTION_SECRET_KEY]
    +lakefsConfig: |
    +    blockstore:
    +      type: gs
    +      # Uncomment the following lines to give lakeFS access to your buckets using a service account:
    +      # gs:
    +      #   credentials_json: [YOUR SERVICE ACCOUNT JSON STRING]
    +
    +
  2. +
  3. +

    Fill in the missing values and save the file as conf-values.yaml. For more configuration options, see our Helm chart README.

    + +

    The lakefsConfig parameter is the lakeFS configuration documented here but without sensitive information. +Sensitive information like databaseConnectionString is given through separate parameters, and the chart will inject it into Kubernetes secrets.

    +
  4. +
  5. +

    In the directory where you created conf-values.yaml, run the following commands:

    + +
    # Add the lakeFS repository
    +helm repo add lakefs https://charts.lakefs.io
    +# Deploy lakeFS
    +helm install my-lakefs lakefs/lakefs -f conf-values.yaml
    +
    + +

    my-lakefs is the Helm Release name.

    +
  6. +
+

+ + + Load balancing + + +

+ + +

To configure a load balancer to direct requests to the lakeFS servers you can use the LoadBalancer Service type or a Kubernetes Ingress. +By default, lakeFS operates on port 8000 and exposes a /_health endpoint that you can use for health checks.

+ +

💡 The NGINX Ingress Controller by default limits the client body size to 1 MiB. +Some clients use bigger chunks to upload objects - for example, multipart upload to lakeFS using the S3-compatible Gateway or +a simple PUT request using the OpenAPI Server. +Checkout Nginx documentation for increasing the limit, or an example of Nginx configuration with MinIO.

+ +
+
+

+ + + Create the admin user + + +

+ + +

When you first open the lakeFS UI, you will be asked to create an initial admin user.

+ +
    +
  1. Open http://<lakefs-host>/ in your browser. If you haven’t set up a load balancer, this will likely be http://<instance ip address>:8000/
  2. +
  3. +

    On first use, you’ll be redirected to the setup page:

    + +

    Create user

    +
  4. +
  5. +

    Follow the steps to create an initial administrator user. Save the credentials you’ve received somewhere safe, you won’t be able to see them again!

    + +

    Setup Done

    +
  6. +
  7. Follow the link and go to the login screen. Use the credentials from the previous step to log in.
  8. +
+

+ + + Create your first repository + + +

+ + +
    +
  1. Use the credentials from the previous step to log in
  2. +
  3. +

    Click Create Repository and choose Blank Repository.

    + +

    Create Repo

    +
  4. +
  5. Under Storage Namespace, enter a path to your desired location on the object store. This is where data written to this repository will be stored.
  6. +
  7. Click Create Repository
  8. +
  9. +

    You should now have a configured repository, ready to use!

    + +

    Repo Created

    +
  10. +
+ +

Congratulations! Your environment is now ready 🤩

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/index.html b/v1.46/howto/deploy/index.html new file mode 100644 index 000000000..8e52afcff --- /dev/null +++ b/v1.46/howto/deploy/index.html @@ -0,0 +1,744 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Install lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Deploy and Setup lakeFS + + +

+ + +
+

The instructions given here are for a self-managed deployment of lakeFS.

+ +

For a hosted lakeFS service with guaranteed SLAs, try lakeFS Cloud

+
+ +

This section will guide you through deploying lakeFS on top of an object store. You will require a database, and can optionally configure authentication using providers specific to your deployment platform.

+ +

Which options are available depends on your deployment platform. For example, the object store available on Azure differs from that on AWS.

+ +

+

+ + + Deployment and Setup Details + + +

+ + +

lakeFS releases include binaries for common operating systems, a containerized option or a Helm chart.

+ +

Check out our guides below for full deployment details:

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/onprem.html b/v1.46/howto/deploy/onprem.html new file mode 100644 index 000000000..380966eaa --- /dev/null +++ b/v1.46/howto/deploy/onprem.html @@ -0,0 +1,1056 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +On-Premises | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + On-Premises Deployment + + +

+ + +
+

The instructions given here are for a self-managed deployment of lakeFS.

+ +

For a hosted lakeFS service with guaranteed SLAs, try lakeFS Cloud

+
+ + + +

⏰ Expected deployment time: 25 min

+

+ + + Prerequisites + + +

+ + +

To use lakeFS on-premises, you can either use the local blockstore adapter or have access to an S3-compatible object store such as MinIO.

+ +

For more information on how to set up MinIO, see the official deployment guide

+

+ + + Setting up a database + + +

+ + +

lakeFS requires a PostgreSQL database to synchronize actions on your repositories. +This section assumes that you already have a PostgreSQL >= 11.0 database accessible.

+

+ + + Setting up a lakeFS Server + + +

+ + +
+ +
+ +

Connect to your host using SSH:

+ +
    +
  1. +

    Create a config.yaml on your VM, with the following parameters:

    + +
    ---
    +database:
    +  type: "postgres"
    +  postgres:
    +    connection_string: "[DATABASE_CONNECTION_STRING]"
    +  
    +auth:
    +  encrypt:
    +    # replace this with a randomly-generated string. Make sure to keep it safe!
    +    secret_key: "[ENCRYPTION_SECRET_KEY]"
    +   
    +blockstore:
    +  type: s3
    +  s3:
    +     force_path_style: true
    +     endpoint: http://<minio_endpoint>
    +     discover_bucket_region: false
    +     credentials:
    +        access_key_id: <minio_access_key>
    +        secret_access_key: <minio_secret_key>
    +
    + +

    ⚠️ Notice that the lakeFS Blockstore type is set to s3 - This configuration works with S3-compatible storage engines such as MinIO.

    +
  2. +
  3. +

    Download the binary to the server.

    +
  4. +
  5. +

    Run the lakefs binary:

    + +
    lakefs --config config.yaml run
    +
    +
  6. +
+ +

Note: It’s preferable to run the binary as a service using systemd or your operating system’s facilities.

+ +
+
+ +

To support container-based environments, you can configure lakeFS using environment variables. Here is a docker run +command to demonstrate starting lakeFS using Docker:

+ +
docker run \
+  --name lakefs \
+  -p 8000:8000 \
+  -e LAKEFS_DATABASE_TYPE="postgres" \
+  -e LAKEFS_DATABASE_POSTGRES_CONNECTION_STRING="[DATABASE_CONNECTION_STRING]" \
+  -e LAKEFS_AUTH_ENCRYPT_SECRET_KEY="[ENCRYPTION_SECRET_KEY]" \
+  -e LAKEFS_BLOCKSTORE_TYPE="s3" \
+  -e LAKEFS_BLOCKSTORE_S3_FORCE_PATH_STYLE="true" \
+  -e LAKEFS_BLOCKSTORE_S3_ENDPOINT="http://<minio_endpoint>" \
+  -e LAKEFS_BLOCKSTORE_S3_DISCOVER_BUCKET_REGION="false" \
+  -e LAKEFS_BLOCKSTORE_S3_CREDENTIALS_ACCESS_KEY_ID="<minio_access_key>" \
+  -e LAKEFS_BLOCKSTORE_S3_CREDENTIALS_SECRET_ACCESS_KEY="<minio_secret_key>" \
+  treeverse/lakefs:latest run
+
+ +

⚠️ Notice that the lakeFS Blockstore type is set to s3 - This configuration works with S3-compatible storage engines such as MinIO.

+ +

See the reference for a complete list of environment variables.

+ +
+
+ +

You can install lakeFS on Kubernetes using a Helm chart.

+ +

To install lakeFS with Helm:

+ +
    +
  1. +

    Copy the Helm values file relevant for S3-Compatible storage (MinIO in this example):

    + +
    secrets:
    +    # replace this with the connection string of the database you created in a previous step:
    +    databaseConnectionString: [DATABASE_CONNECTION_STRING]
    +    # replace this with a randomly-generated string
    +    authEncryptSecretKey: [ENCRYPTION_SECRET_KEY]
    +lakefsConfig: |
    +    blockstore:
    +      type: s3
    +      s3:
    +        force_path_style: true
    +        endpoint: http://<minio_endpoint>
    +        discover_bucket_region: false
    +        credentials:
    +          access_key_id: <minio_access_key>
    +          secret_access_key: <minio_secret_key>
    +
    + +

    ⚠️ Notice that the lakeFS Blockstore type is set to s3 - This configuration works with S3-compatible storage engines such as MinIO.

    +
  2. +
  3. +

    Fill in the missing values and save the file as conf-values.yaml. For more configuration options, see our Helm chart README.

    + +

    The lakefsConfig parameter is the lakeFS configuration documented here but without sensitive information. +Sensitive information like databaseConnectionString is given through separate parameters, and the chart will inject it into Kubernetes secrets.

    +
  4. +
  5. +

    In the directory where you created conf-values.yaml, run the following commands:

    + +
    # Add the lakeFS repository
    +helm repo add lakefs https://charts.lakefs.io
    +# Deploy lakeFS
    +helm install my-lakefs lakefs/lakefs -f conf-values.yaml
    +
    + +

    my-lakefs is the Helm Release name.

    +

    + + + Load balancing + + +

    + + +

    To configure a load balancer to direct requests to the lakeFS servers you can use the LoadBalancer Service type or a Kubernetes Ingress. +By default, lakeFS operates on port 8000 and exposes a /_health endpoint that you can use for health checks.

    + +

    💡 The NGINX Ingress Controller by default limits the client body size to 1 MiB. +Some clients use bigger chunks to upload objects - for example, multipart upload to lakeFS using the S3-compatible Gateway or +a simple PUT request using the OpenAPI Server. +Checkout Nginx documentation for increasing the limit, or an example of Nginx configuration with MinIO.

    +
  6. +
+ +
+
+

+ + + Secure connection + + +

+ + +

Using a load balancer or cluster manager for TLS/SSL termination is recommended. It helps speed the decryption process and reduces the processing burden from lakeFS.

+ +

In case lakeFS needs to listen and serve with HTTPS, for example for development purposes, update its config yaml with the following section:

+ +
tls:
+  enabled: true
+  cert_file: server.crt   # provide path to your certificate file
+  key_file: server.key    # provide path to your server private key
+
+

+ + + Local Blockstore + + +

+ + +

You can configure a block adapter to a POSIX compatible storage location shared by all lakeFS instances. +Using the shared storage location, both data and metadata will be stored there.

+ +

Using the local blockstore import and allowing lakeFS access to a specific prefix, it is possible to import files from a shared location. +Import is not enabled by default, as it doesn’t assume the local path is shared and there is a security concern about accessing a path outside the specified in the blockstore configuration. +Enabling is done by blockstore.local.import_enabled and blockstore.local.allowed_external_prefixes as described in the configuration reference.

+

+ + + Sample configuration using local blockstore + + +

+ + +
database:
+  type: "postgres"
+  postgres:
+    connection_string: "[DATABASE_CONNECTION_STRING]"
+  
+auth:
+  encrypt:
+    # replace this with a randomly-generated string. Make sure to keep it safe!
+    secret_key: "[ENCRYPTION_SECRET_KEY]"
+
+blockstore:
+  type: local
+  local:
+    path: /shared/location/lakefs_data    # location where data and metadata kept by lakeFS
+    import_enabled: true                  # required to be true to enable import files
+                                          # from `allowed_external_prefixes` locations
+    allowed_external_prefixes:
+      - /shared/location/files_to_import  # location with files we can import into lakeFS, require access from lakeFS
+
+

+ + + Limitations + + +

+ + +
    +
  • Using a local adapter on a shared location is relativly new and not battle-tested yet
  • +
  • lakeFS doesn’t control the way a shared location is managed across machines
  • +
  • When using lakectl or the lakeFS UI, you can currently import only directories. If you need to import a single file, use the HTTP API or API Clients with type=object in the request body and destination=<full-path-to-file>.
  • +
  • Garbage collector (for committed and uncommitted) and lakeFS Hadoop FileSystem currently unsupported
  • +
+

+ + + Create the admin user + + +

+ + +

When you first open the lakeFS UI, you will be asked to create an initial admin user.

+ +
    +
  1. Open http://<lakefs-host>/ in your browser. If you haven’t set up a load balancer, this will likely be http://<instance ip address>:8000/
  2. +
  3. +

    On first use, you’ll be redirected to the setup page:

    + +

    Create user

    +
  4. +
  5. +

    Follow the steps to create an initial administrator user. Save the credentials you’ve received somewhere safe, you won’t be able to see them again!

    + +

    Setup Done

    +
  6. +
  7. Follow the link and go to the login screen. Use the credentials from the previous step to log in.
  8. +
+

+ + + Create your first repository + + +

+ + +
    +
  1. Use the credentials from the previous step to log in
  2. +
  3. +

    Click Create Repository and choose Blank Repository.

    + +

    Create Repo

    +
  4. +
  5. Under Storage Namespace, enter a path to your desired location on the object store. This is where data written to this repository will be stored.
  6. +
  7. Click Create Repository
  8. +
  9. +

    You should now have a configured repository, ready to use!

    + +

    Repo Created

    +
  10. +
+ +

Congratulations! Your environment is now ready 🤩

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/deploy/upgrade.html b/v1.46/howto/deploy/upgrade.html new file mode 100644 index 000000000..81cf66ac5 --- /dev/null +++ b/v1.46/howto/deploy/upgrade.html @@ -0,0 +1,888 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Upgrade lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Upgrading lakeFS + + +

+ + +

Note: For a fully managed lakeFS service with guaranteed SLAs, try lakeFS Cloud

+ +

Upgrading lakeFS from a previous version usually just requires re-deploying with the latest image (or downloading the latest version if you’re using the binary). +If you’re upgrading, check whether the release requires a migration.

+

+ + + When DB migrations are required + + +

+ +

+ + + lakeFS 0.103.0 or greater + + +

+ + +

Version 0.103.0 added support for rolling KV upgrade. This means that users who already migrated to the KV ref-store (versions 0.80.0 and above) no longer have to pass through specific versions for migration. +This includes ACL migration which was introduced in lakeFS version 0.98.0. +Running lakefs migrate up on the latest lakeFS version will perform all the necessary migrations up to that point.

+

+ + + lakeFS 0.80.0 or greater (KV Migration) + + +

+ + +

Starting with version 0.80.2, lakeFS has transitioned from using a PostgreSQL based database implementation to a Key-Value datastore interface supporting +multiple database implementations. More information can be found here.
+Users upgrading from a previous version of lakeFS must pass through the KV migration version (0.80.2) before upgrading to newer versions of lakeFS.

+ +
+

IMPORTANT: Pre Migrate Requirements

+
    +
  • Users using OS environment variables for database configuration must define the connection_string explicitly or as environment variable before proceeding with the migration.
  • +
  • Database storage free capacity of at least twice the amount of the currently used capacity
  • +
  • It is strongly recommended to perform these additional steps: +
      +
    • Commit all uncommitted data on branches
    • +
    • Create a snapshot of your database
    • +
    +
  • +
  • By default, old database tables are not being deleted by the migration process, and should be removed manually after a successful migration. +To enable table drop as part of the migration, set the database.drop_tables configuration param to true
  • +
+
+

+ + + Migration Steps + + +

+ +

For each lakeFS instance currently running with the database

+
    +
  1. Modify the database section under lakeFS configuration yaml: +
      +
    1. Add type field with "postgres" as value
    2. +
    3. Copy the current configuration parameters to a new section called postgres
    4. +
    + +
    ---
    +database:
    + type: "postgres"
    + connection_string: "postgres://localhost:5432/postgres?sslmode=disable"
    + max_open_connections: 20
    +   
    + postgres:
    +   connection_string: "postgres://localhost:5432/postgres?sslmode=disable"
    +   max_open_connections: 20
    +
    +
  2. +
  3. Stop all lakeFS instances
  4. +
  5. +

    Using the lakefs binary for the new version (0.80.2), run the following:

    + +
    lakefs migrate up
    +
    +
  6. +
  7. +

    lakeFS will run the migration process, which in the end should display the following message with no errors:

    + +
    time="2022-08-10T14:46:25Z" level=info msg="KV Migration took 717.629563ms" func="pkg/logging.(*logrusEntryWrapper).Infof" file="build/pkg/logging/logger.go:246" TempDir=/tmp/kv_migrate_2913402680
    +
    +
  8. +
  9. +

    It is now possible to remove the old database configuration. The updated configuration should look as such:

    + +
    ---
    +database:
    + type: "postgres"
    +   
    + postgres:
    +   connection_string: "postgres://localhost:5432/postgres?sslmode=disable"
    +   max_open_connections: 20
    +
    +
  10. +
  11. Deploy (or run) the new version of lakeFS.
  12. +
+

+ + + lakeFS 0.30.0 or greater + + +

+ + +

In case migration is required, you first need to stop the running lakeFS service. +Using the lakefs binary for the new version, run the following:

+ +
lakefs migrate up
+
+ +

Deploy (or run) the new version of lakeFS.

+ +

Note that an older version of lakeFS cannot run on a migrated database.

+

+ + + Prior to lakeFS 0.30.0 + + +

+ + +

Note: with lakeFS < 0.30.0, you should first upgrade to 0.30.0 following this guide. Then, proceed to upgrade to the newest version.

+ +

Starting version 0.30.0, lakeFS handles your committed metadata in a new way, which is more robust and has better performance. +To move your existing data, you will need to run the following upgrade commands.

+ +

Verify lakeFS version == 0.30.0 (can skip if using Docker)

+ +
lakefs --version
+
+ +

Migrate data from the previous format:

+ +
lakefs migrate db
+
+ +

Or migrate using Docker image:

+ +
docker run --rm -it -e LAKEFS_DATABASE_CONNECTION_STRING=<database connection string> treeverse/lakefs:rocks-migrate migrate db
+
+ +

Once migrated, it is possible to now use more recent lakeFS versions. Please refer to their release notes for more information on ugrading and usage).

+ +

If you want to start over, discarding your existing data, you need to explicitly state this in your lakeFS configuration file. +To do so, add the following to your configuration (relevant only for 0.30.0):

+ +
cataloger:
+  type: rocks
+
+

+ + + Data Migration for Version v0.50.0 + + +

+ + +

If you are using a version before 0.50.0, you must first perform the previous upgrade to that version. {: note: .note-warning }

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/export.html b/v1.46/howto/export.html new file mode 100644 index 000000000..a0c1308ca --- /dev/null +++ b/v1.46/howto/export.html @@ -0,0 +1,1022 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Export Data | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Exporting Data + + +

+ +

The export operation copies all data from a given lakeFS commit to +a designated object store location.

+ +

For instance, the contents lakefs://example/main might be exported on +s3://company-bucket/example/latest. Clients entirely unaware of lakeFS could use that +base URL to access latest files on main. Clients aware of lakeFS can continue to use +the lakeFS S3 endpoint to access repository files on s3://example/main, as well as +other versions and uncommitted versions.

+ +

Possible use-cases:

+
    +
  1. External consumers of data don’t have access to your lakeFS installation.
  2. +
  3. Some data pipelines in the organization are not fully migrated to lakeFS.
  4. +
  5. You want to experiment with lakeFS as a side-by-side installation first.
  6. +
  7. Create copies of your data lake in other regions (taking into account read pricing).
  8. +
+ + +

+ + + Exporting Data With Spark + + +

+ +

+ + + Using spark-submit + + +

+ +

You can use the export main in three different modes:

+ +
    +
  1. +

    Export all the objects from branch example-branch on example-repo repository to S3 location s3://example-bucket/prefix/:

    + +
    .... example-repo s3://example-bucket/prefix/ --branch=example-branch
    +
    +
  2. +
  3. +

    Export all the objects from a commit c805e49bafb841a0875f49cd555b397340bbd9b8 on example-repo repository to S3 location s3://example-bucket/prefix/:

    + +
    .... example-repo s3://example-bucket/prefix/ --commit_id=c805e49bafb841a0875f49cd555b397340bbd9b8
    +
    +
  4. +
  5. +

    Export only the diff between branch example-branch and commit c805e49bafb841a0875f49cd555b397340bbd9b8 +on example-repo repository to S3 location s3://example-bucket/prefix/:

    + +
    .... example-repo s3://example-bucket/prefix/ --branch=example-branch --prev_commit_id=c805e49bafb841a0875f49cd555b397340bbd9b8
    +
    +
  6. +
+ +

The complete spark-submit command would look as follows:

+ +
spark-submit --conf spark.hadoop.lakefs.api.url=https://<LAKEFS_ENDPOINT>/api/v1 \
+  --conf spark.hadoop.lakefs.api.access_key=<LAKEFS_ACCESS_KEY_ID> \
+  --conf spark.hadoop.lakefs.api.secret_key=<LAKEFS_SECRET_ACCESS_KEY> \
+  --packages io.lakefs:lakefs-spark-client_2.12:0.14.1 \
+  --class io.treeverse.clients.Main export-app example-repo s3://example-bucket/prefix \
+  --branch=example-branch
+
+ +

The command assumes that the Spark cluster has permissions to write to s3://example-bucket/prefix. +Otherwise, add spark.hadoop.fs.s3a.access.key and spark.hadoop.fs.s3a.secret.key with the proper credentials.

+

+ + + Networking + + +

+ + +

Spark export communicates with the lakeFS server. Very large repositories +may require increasing a read timeout. If you run into timeout errors +during communication from the Spark job to lakeFS consider increasing these +timeouts:

+ +
    +
  • Add -c spark.hadoop.lakefs.api.read.timeout_seconds=TIMEOUT_IN_SECONDS +(default 10) to allow lakeFS more time to respond to requests.
  • +
  • Add -c +spark.hadoop.lakefs.api.connection.timeout_seconds=TIMEOUT_IN_SECONDS +(default 10) to wait longer for lakeFS to accept connections.
  • +
+

+ + + Using custom code (Notebook/Spark) + + +

+ + +

Set up lakeFS Spark metadata client with the endpoint and credentials as instructed in the previous page.

+ +

The client exposes the Exporter object with three export options:

+ +
  1. +Export *all* the objects at the HEAD of a given branch. Does not include +files that were added to that branch but were not committed. + +
    + +
    +
    exportAllFromBranch(branch: String)
    +
    +
    +
    +
  2. +
  3. Export ALL objects from a commit: + +
    + +
    +
    exportAllFromCommit(commitID: String)
    +
    +
    +
    +
  4. +
  5. Export just the diff between a commit and the HEAD of a branch. + + This is an ideal option for continuous exports of a branch since it will change only the files + that have been changed since the previous commit. + +
    + +
    +
    exportFrom(branch: String, prevCommitID: String)
    +
    +
    +
    +
  6. +
+

+ + + Success/Failure Indications + + +

+ + +

When the Spark export operation ends, an additional status file will be added to the root +object storage destination. +If all files were exported successfully, the file path will be of the form: EXPORT_<commitID>_<ISO-8601-time-UTC>_SUCCESS. +For failures: the form will beEXPORT_<commitID>_<ISO-8601-time-UTC>_FAILURE, and the file will include a log of the failed files operations.

+

+ + + Export Rounds (Spark success files) + + +

+ +

Some files should be exported before others, e.g., a Spark _SUCCESS file exported before other files under +the same prefix might send the wrong indication.

+ +

The export operation may contain several rounds within the same export. +A failing round will stop the export of all the files of the next rounds.

+ +

By default, lakeFS will use the SparkFilter and have 2 rounds for each export. +The first round will export any non-Spark _SUCCESS files. Second round will export all Spark’s _SUCCESS files. +You may override the default behavior by passing a custom filter to the Exporter.

+

+ + + Example + + +

+ + +
  1. First configure the `Exporter` instance: + +
    + +
    +
    import io.treeverse.clients.{ApiClient, Exporter}
    +import org.apache.spark.sql.SparkSession
    +
    +val endpoint = "http://<LAKEFS_ENDPOINT>/api/v1"
    +val accessKey = "<LAKEFS_ACCESS_KEY_ID>"
    +val secretKey = "<LAKEFS_SECRET_ACCESS_KEY>"
    +
    +val repo = "example-repo"
    +
    +val spark = SparkSession.builder().appName("I can export").master("local").getOrCreate()
    +val sc = spark.sparkContext
    +sc.hadoopConfiguration.set("lakefs.api.url", endpoint)
    +sc.hadoopConfiguration.set("lakefs.api.access_key", accessKey)
    +sc.hadoopConfiguration.set("lakefs.api.secret_key", secretKey)
    +
    +// Add any required spark context configuration for s3
    +val rootLocation = "s3://company-bucket/example/latest"
    +
    +val apiClient = new ApiClient(endpoint, accessKey, secretKey)
    +val exporter = new Exporter(spark, apiClient, repo, rootLocation)
    +
    +
    +
  2. +
  3. Now you can export all objects from `main` branch to `s3://company-bucket/example/latest`: + +
    + +
    +
    val branch = "main"
    +exporter.exportAllFromBranch(branch)
    +
    +
    +
  4. +
  5. Assuming a previous successful export on commit `f3c450d8cd0e84ac67e7bc1c5dcde9bef82d8ba7`, +you can alternatively export just the difference between `main` branch and the commit: + +
    + +
    +
    val branch = "main"
    +val commit = "f3c450d8cd0e84ac67e7bc1c5dcde9bef82d8ba7"
    +exporter.exportFrom(branch, commit)
    +
    +
    +
+

+ + + Exporting Data with Docker + + +

+ + +

This option is recommended if you don’t have Spark at your tool-set. +It doesn’t support distribution across machines, therefore may have a lower performance. +Using this method, you can export data from lakeFS to S3 using the export options (in a similar way to the Spark export):

+ +
    +
  1. +

    Export all objects from a branch example-branch on example-repo repository to S3 location s3://destination-bucket/prefix/:

    + +
    .... example-repo s3://destination-bucket/prefix/ --branch="example-branch"
    +
    +
  2. +
  3. +

    Export all objects from a commit c805e49bafb841a0875f49cd555b397340bbd9b8 on example-repo repository to S3 location s3://destination-bucket/prefix/:

    + +
    .... example-repo s3://destination-bucket/prefix/ --commit_id=c805e49bafb841a0875f49cd555b397340bbd9b8
    +
    +
  4. +
  5. +

    Export only the diff between branch example-branch and commit c805e49bafb841a0875f49cd555b397340bbd9b8 +on example-repo repository to S3 location s3://destination-bucket/prefix/:

    + +
    .... example-repo s3://destination-bucket/prefix/ --branch="example-branch" --prev_commit_id=c805e49bafb841a0875f49cd555b397340bbd9b8
    +
    +
  6. +
+ +

You will need to add the relevant environment variables. +The complete docker run command would look like:

+ +
docker run \
+    -e LAKEFS_ACCESS_KEY_ID=XXX -e LAKEFS_SECRET_ACCESS_KEY=YYY \
+   -e LAKEFS_ENDPOINT=https://<LAKEFS_ENDPOINT>/ \
+   -e AWS_ACCESS_KEY_ID=XXX -e AWS_SECRET_ACCESS_KEY=YYY \
+   treeverse/lakefs-rclone-export:latest \
+      example-repo \
+      s3://destination-bucket/prefix/ \
+      --branch="example-branch"
+
+ +

Note: This feature uses rclone, +and specifically rclone sync. This can change the destination path, therefore the s3 destination location must be designated to lakeFS export.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/garbage-collection-index.html b/v1.46/howto/garbage-collection-index.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection-index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/garbage-collection.html b/v1.46/howto/garbage-collection.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/garbage-collection/committed.html b/v1.46/howto/garbage-collection/committed.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection/committed.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/garbage-collection/gc.html b/v1.46/howto/garbage-collection/gc.html new file mode 100644 index 000000000..0277223b2 --- /dev/null +++ b/v1.46/howto/garbage-collection/gc.html @@ -0,0 +1,1005 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Garbage Collection | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Garbage Collection + + +

+ + +

lakeFS Cloud users enjoy a managed garbage collection service, and do not need to run this Spark program.

+ +

By default, lakeFS keeps all your objects forever. This allows you to travel back in time to previous versions of your data. +However, sometimes you may want to remove the objects from the underlying storage completely. +Reasons for this include cost-reduction and privacy policies.

+ +

The garbage collection (GC) job is a Spark program that removes the following from the underlying storage:

+
    +
  1. Committed objects that have been deleted (or replaced) in lakeFS, and are considered expired according to rules you define.
  2. +
  3. Uncommitted objects that are no longer accessible +
      +
    • For example, objects deleted before ever being committed.
    • +
    +
  4. +
+ + +

+ + + Garbage collection rules + + +

+ + +

These rules only apply to objects that have been committed at some point. +Without retention rules, only inaccessible uncommitted objects will be removed by the job.

+ +

Garbage collection rules determine for how long an object is kept in the storage after it is deleted (or replaced) in lakeFS. +For every branch, the GC job retains deleted objects for the number of days defined for the branch. +In the absence of a branch-specific rule, the default rule for the repository is used. +If an object is present in more than one branch ancestry, it is removed only after the retention period has ended for +all relevant branches.

+ +

Example GC rules for a repository:

+
{
+  "default_retention_days": 14,
+  "branches": [
+    {"branch_id": "main", "retention_days": 21},
+    {"branch_id": "dev", "retention_days": 7}
+  ]
+}
+
+ +

In the above example, objects will be retained for 14 days after deletion by default. +However, if present in the branch main, objects will be retained for 21 days. +Objects present only in the dev branch will be retained for 7 days after they are deleted.

+

+ + + How to configure garbage collection rules + + +

+ + +

To define retention rules, either use the lakectl command, the lakeFS web UI, or API:

+ +
+ +
+ +

Create a JSON file with your GC rules:

+ +
cat <<EOT >> example_repo_gc_rules.json
+{
+  "default_retention_days": 14,
+  "branches": [
+    {"branch_id": "main", "retention_days": 21},
+    {"branch_id": "dev", "retention_days": 7}
+  ]
+}
+EOT
+
+ +

Set the GC rules using lakectl:

+
lakectl gc set-config lakefs://example-repo -f example_repo_gc_rules.json 
+
+ +
+
+

From the lakeFS web UI:

+ +
    +
  1. Navigate to the main page of your repository.
  2. +
  3. Go to Settings -> Garbage Collection.
  4. +
  5. Click Edit policy and paste your GC rule into the text box as a JSON.
  6. +
  7. Save your changes.
  8. +
+ +

GC Rules From UI

+
+
+

+ + + How to run the garbage collection job + + +

+ + +

To run the job, use the following spark-submit command (or using your preferred method of running Spark programs).

+ +
+ +
+
spark-submit --class io.treeverse.gc.GarbageCollection \
+  --packages org.apache.hadoop:hadoop-aws:2.7.7 \
+  -c spark.hadoop.lakefs.api.url=https://lakefs.example.com:8000/api/v1  \
+  -c spark.hadoop.lakefs.api.access_key=<LAKEFS_ACCESS_KEY> \
+  -c spark.hadoop.lakefs.api.secret_key=<LAKEFS_SECRET_KEY> \
+  -c spark.hadoop.fs.s3a.access.key=<S3_ACCESS_KEY> \
+  -c spark.hadoop.fs.s3a.secret.key=<S3_SECRET_KEY> \
+  http://treeverse-clients-us-east.s3-website-us-east-1.amazonaws.com/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar \
+  example-repo us-east-1
+
+
+
+ +

If you want to access your storage using the account key:

+ +
spark-submit --class io.treeverse.gc.GarbageCollection \
+  --packages org.apache.hadoop:hadoop-aws:3.2.1 \
+  -c spark.hadoop.lakefs.api.url=https://lakefs.example.com:8000/api/v1  \
+  -c spark.hadoop.lakefs.api.access_key=<LAKEFS_ACCESS_KEY> \
+  -c spark.hadoop.lakefs.api.secret_key=<LAKEFS_SECRET_KEY> \
+  -c spark.hadoop.fs.azure.account.key.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=<AZURE_STORAGE_ACCESS_KEY> \
+  http://treeverse-clients-us-east.s3-website-us-east-1.amazonaws.com/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar \
+  example-repo
+
+ +

Or, if you want to access your storage using an Azure service principal:

+ +
spark-submit --class io.treeverse.gc.GarbageCollection \
+  --packages org.apache.hadoop:hadoop-aws:3.2.1 \
+  -c spark.hadoop.lakefs.api.url=https://lakefs.example.com:8000/api/v1  \
+  -c spark.hadoop.lakefs.api.access_key=<LAKEFS_ACCESS_KEY> \
+  -c spark.hadoop.lakefs.api.secret_key=<LAKEFS_SECRET_KEY> \
+  -c spark.hadoop.fs.azure.account.auth.type.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=OAuth \
+  -c spark.hadoop.fs.azure.account.oauth.provider.type.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=org.apache.hadoop.fs.azurebfs.oauth2.ClientCredsTokenProvider \
+  -c spark.hadoop.fs.azure.account.oauth2.client.id.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=<application-id> \
+  -c spark.hadoop.fs.azure.account.oauth2.client.secret.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=<service-credential-key> \
+  -c spark.hadoop.fs.azure.account.oauth2.client.endpoint.<AZURE_STORAGE_ACCOUNT>.dfs.core.windows.net=https://login.microsoftonline.com/<directory-id>/oauth2/token \
+  http://treeverse-clients-us-east.s3-website-us-east-1.amazonaws.com/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar \
+  example-repo
+
+ +

Notes:

+
    +
  • On Azure, GC was tested only on Spark 3.3.0, but may work with other Spark and Hadoop versions.
  • +
  • In case you don’t have hadoop-azure package as part of your environment, you should add the package to your spark-submit with --packages org.apache.hadoop:hadoop-azure:3.2.1
  • +
  • For GC to work on Azure blob, soft delete should be disabled.
  • +
+
+ +
+

⚠️ At the moment, only the “mark” phase of the Garbage Collection is supported for GCP. +That is, this program will output a list of expired objects, and you will have to delete them manually. +We have concrete plans to extend this support to actually delete the objects.

+ +
spark-submit --class  io.treeverse.gc.GarbageCollection \
+  --jars https://storage.googleapis.com/hadoop-lib/gcs/gcs-connector-hadoop3-latest.jar \
+  -c spark.hadoop.lakefs.api.url=https://lakefs.example.com:8000/api/v1  \
+  -c spark.hadoop.lakefs.api.access_key=<LAKEFS_ACCESS_KEY> \
+  -c spark.hadoop.lakefs.api.secret_key=<LAKEFS_SECRET_KEY> \
+  -c spark.hadoop.google.cloud.auth.service.account.enable=true \
+  -c spark.hadoop.google.cloud.auth.service.account.json.keyfile=<PATH_TO_JSON_KEYFILE> \
+  -c spark.hadoop.fs.gs.project.id=<GCP_PROJECT_ID> \
+  -c spark.hadoop.fs.gs.impl=com.google.cloud.hadoop.fs.gcs.GoogleHadoopFileSystem \
+  -c spark.hadoop.fs.AbstractFileSystem.gs.impl=com.google.cloud.hadoop.fs.gcs.GoogleHadoopFS \
+  -c spark.hadoop.lakefs.gc.do_sweep=false  \
+  http://treeverse-clients-us-east.s3-website-us-east-1.amazonaws.com/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar \
+  example-repo
+
+ +

This program will not delete anything. +Instead, it will find all the objects that are safe to delete and save a list containing all their keys, in Parquet format. +The list will then be found under the path:

+
gs://<STORAGE_NAMESPACE>/_lakefs/retention/gc/unified/<RUN_ID>/deleted/
+
+ +

Note that this is a path in your Google Storage bucket, and not in your lakeFS repository. +It is now safe to remove the objects that appear in this list directly from the storage.

+ +
+
+ +

You will find the list of objects removed by the job in the storage +namespace of the repository. It is saved in Parquet format under _lakefs/retention/gc/unified/<RUN_ID>/deleted/.

+

+ + + Mark and Sweep stages + + +

+ + +

You can break the job into two stages:

+
    +
  • Mark: find objects to remove, without actually removing them.
  • +
  • Sweep: remove the objects.
  • +
+

+ + + Mark-only mode + + +

+ + +

To make GC run the mark stage only, add the following to your spark-submit command:

+
spark.hadoop.lakefs.gc.do_sweep=false
+
+ +

In mark-only mode, GC will write the keys of the expired objects under: <REPOSITORY_STORAGE_NAMESPACE>/_lakefs/retention/gc/unified/<MARK_ID>/. +MARK_ID is generated by the job. You can find it in the driver’s output:

+ +
Report for mark_id=gmc6523jatlleurvdm30 path=s3a://example-bucket/_lakefs/retention/gc/unified/gmc6523jatlleurvdm30
+
+

+ + + Sweep-only mode + + +

+ + +

To make GC run the sweep stage only, add the following properties to your spark-submit command:

+
spark.hadoop.lakefs.gc.do_mark=false
+spark.hadoop.lakefs.gc.mark_id=<MARK_ID> # Replace <MARK_ID> with the identifier you obtained from a previous mark-only run
+
+

+ + + Garbage collection notes + + +

+ + +
    +
  1. +

    In order for an object to be removed, it must not exist on the HEAD of any branch. +You should remove stale branches to prevent them from retaining old objects. +For example, consider a branch that has been merged to main and has become stale. +An object which is later deleted from main will always be present in the stale branch, preventing it from being removed.

    +
  2. +
  3. +

    lakeFS will never delete objects outside your repository’s storage namespace. +In particular, objects that were imported using lakectl import or the UI import wizard will not be affected by GC jobs.

    +
  4. +
  5. +

    In cases where deleted objects are brought back to life while a GC job is running (for example, by reverting a commit), +the objects may or may not be deleted.

    +
  6. +
  7. +

    Garbage collection does not remove any commits: you will still be able to use commits containing removed objects, +but trying to read these objects from lakeFS will result in a 410 Gone HTTP status.

    +
  8. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/garbage-collection/index.html b/v1.46/howto/garbage-collection/index.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/garbage-collection/internals.html b/v1.46/howto/garbage-collection/internals.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection/internals.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/garbage-collection/managed-gc.html b/v1.46/howto/garbage-collection/managed-gc.html new file mode 100644 index 000000000..7c5eb9c04 --- /dev/null +++ b/v1.46/howto/garbage-collection/managed-gc.html @@ -0,0 +1,755 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Managed Garbage Collection | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Managed Garbage Collection + + +

+ +

lakeFS Cloud

+ +
+

Managed GC is only available for lakeFS Cloud. If you are using the self-managed lakeFS, garbage collection is available to run manually.

+
+

+ + + Benefits of using managed GC + + +

+ +
    +
  • The quick and safe way to delete your unnecessary objects
  • +
  • No operational overhead
  • +
  • SLA for when your objects are deleted
  • +
  • Support from the Treeverse team
  • +
+

+ + + How it works + + +

+ +

Similarly to the self-managed lakeFS, managed GC uses garbage collection rules to determine which objects to delete. +However, it uses our super-fast and efficient engine to detect stale objects and branches (depends on your configuration) and prioritize them for deletion.

+

+ + + Setting up + + +

+ +

Enable managed GC through the lakeFS Cloud onboarding setup wizard. +This will create additional cloud resources for us to use and have access to delete those objects.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/garbage-collection/standalone-gc.html b/v1.46/howto/garbage-collection/standalone-gc.html new file mode 100644 index 000000000..ee4c03283 --- /dev/null +++ b/v1.46/howto/garbage-collection/standalone-gc.html @@ -0,0 +1,1299 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Standalone Garbage Collection | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Standalone Garbage Collection + + +

+ +

lakeFS Enterprise

+ +

experimental

+ +
+

Standalone GC is only available for lakeFS Enterprise.

+
+ +
+

Standalone GC is experimental and offers limited capabilities compared to the Spark-backed GC. For large scale environments, we recommend using the Spark-backed solution.

+
+ + +

+ + + What is Standalone GC? + + +

+ + +

Standalone GC is a simplified version of the Spark-backed GC that runs without any external dependencies, delivered as a standalone +docker image. It supports S3 and self-managed S3 compatible storages such as MinIO.

+

+ + + Limitations + + +

+ + +
    +
  1. No horizontal scalability: Only a single instance of lakefs-sgc can operate on a given repository at a time.
  2. +
  3. Mark phase only: Standalone GC supports only the mark phase, identifying objects for deletion but not executing +the sweep stage to delete them. It functions similarly to the GC’s mark-only mode.
  4. +
  5. Only supports AWS S3 and S3-compatible object storages. However, supporting Azure blob and GCS are in our roadmap.
  6. +
+

+ + + Installation + + +

+ +

+ + + Step 1: Obtain Dockerhub token + + +

+ +

+ + + lakeFS Enterprise customers + + +

+ + +

Contact your account manager to verify that Standalone GC is included in your license. Then use your dockerhub token for +the externallakefs user.

+

+ + + New to lakeFS Enterprise + + +

+ + +

Please contact us to get trial access to Standalone GC.

+

+ + + Step 2: Login to Dockerhub with this token + + +

+ + +
docker login -u <token>
+
+

+ + + Step 3: Download the docker image + + +

+ + +

Download the image from the lakefs-sgc repository:

+
docker pull treeverse/lakefs-sgc:<tag>
+
+

+ + + Setup + + +

+ +

+ + + Permissions + + +

+ + +

To run lakefs-sgc, you need both AWS (or S3-compatible) storage and lakeFS user permissions as outlined below:

+

+ + + Storage permissions + + +

+ + +

The minimum required permissions for AWS or S3-compatible storage are:

+
{
+  "Version": "2012-10-17",
+  "Statement": [
+    {
+      "Effect": "Allow",
+      "Action": [
+        "s3:PutObject",
+        "s3:GetObject"
+      ],
+      "Resource": [
+        "arn:aws:s3:::some-bucket/some/prefix/*"
+      ]
+    },
+    {
+      "Effect": "Allow",
+      "Action": [
+        "s3:ListBucket"
+      ],
+      "Resource": [
+        "arn:aws:s3:::some-bucket"
+      ]
+    },
+    {
+      "Effect": "Allow",
+      "Action": [
+        "s3:ListAllMyBuckets"
+      ],
+      "Resource": [
+        "arn:aws:s3:::*"
+      ]
+    }
+  ]
+}
+
+

In this example, the repository storage namespace is s3://some-bucket/some/prefix.

+

+ + + lakeFS permissions + + +

+ + +

The minimum required permissions for lakeFS are:

+
{
+  "statement": [
+    {
+      "action": [
+        "retention:PrepareGarbageCollectionCommits",
+        "retention:PrepareGarbageCollectionUncommitted",
+        "fs:ReadConfig",
+        "fs:ReadRepository",
+        "fs:ListObjects",
+        "fs:ReadConfig"
+      ],
+      "effect": "allow",
+      "resource": "arn:lakefs:fs:::repository/<repository>"
+    }
+  ]
+}
+
+

+ + + Credentials + + +

+ + +

Standalone GC supports S3 and S3-compatible storage backends and relies on AWS credentials for authentication. To set up +credentials on the lakefs-sgc docker container, follow AWS guidelines, such as those outlined in this guide. +For details on how to pass credentials to lakefs-sgc, refer to the instructions in How to Run Standalone GC.

+

+ + + Using S3-compatible clients + + +

+ + +

lakefs-sgc leverages AWS credentials to work seamlessly with S3-compatible storage solutions, such as MinIO. +Follow the steps below to set up and use lakefs-sgc with an S3-compatible client:

+ +
    +
  1. Add a profile to your ~/.aws/config file: +
    [profile minio]
    +region = us-east-1
    +endpoint_url = <MinIO URL>
    +s3 =
    +    signature_version = s3v4
    +
    +
  2. +
  3. Add an access and secret keys to your ~/.aws/credentials file: +
    [minio]
    +aws_access_key_id     = <MinIO access key>
    +aws_secret_access_key = <MinIO secret key>
    +
    +
  4. +
  5. Run the lakefs-sgc docker image and pass it the minio profile - see example below.
  6. +
+

+ + + Configuration + + +

+ + +

The following configuration keys are available:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
KeyDescriptionDefault valuePossible values
logging.formatLogs output format“text”“text”,”json”
logging.levelLogs level“info”“error”,”warn”,info”,”debug”,”trace”
logging.outputWhere to output the logs to”-“”-“ (stdout), “=” (stderr), or any string for file path
cache_dirDirectory to use for caching data during run~/.lakefs-sgc/datastring
aws.max_page_sizeMax number of items per page when listing objects in AWS1000number
aws.s3.addressing_path_styleWhether or not to use path-style when reading objects from AWStrueboolean
lakefs.endpoint_urlThe URL to the lakeFS installation - should end with /api/v1NOT SETURL
lakefs.access_key_idAccess key to the lakeFS installationNOT SETstring
lakefs.secret_access_keySecret access key to the lakeFS installationNOT SETstring
+ +

These keys can be provided in the following ways:

+
    +
  1. Config file: Create a YAML file with the keys, each . is a new nesting level.
    +For example, logging.level will be: +
    logging:
    +  level: <value> # info,debug...
    +
    +

    Then, pass it to the program using the --config path/to/config.yaml argument.

    +
  2. +
  3. Environment variables: by setting LAKEFS_SGC_<KEY>, with uppercase letters and .s converted to _s.
    +For example logging.level will be: +
    export LAKEFS_SGC_LOGGING_LEVEL=info
    +
    +
  4. +
+ +

Example (minimalistic) config file:

+
logging:
+  level: debug
+lakefs:
+  endpoint_url: https://your.url/api/v1
+  access_key_id: <lakeFS access key>
+  secret_access_key: <lakeFS secret key>
+
+

+ + + How to Run Standalone GC? + + +

+ +

+ + + Command line reference + + +

+ +

+ + + Flags + + +

+ +
    +
  • -c, --config: config file to use (default is $HOME/.lakefs-sgc.yaml)
  • +
+

+ + + Commands + + +

+ +

run

+ +

Usage:
+lakefs-sgc run <repository>

+ +

Flags:

+
    +
  • --cache-dir: directory to cache read files (default is $HOME/.lakefs-sgc/data/)
  • +
  • --parallelism: number of parallel downloads for metadata files (default 10)
  • +
  • --presign: use pre-signed URLs when downloading/uploading data (recommended) (default true)
  • +
+ +

To run standalone GC, choose the method you prefer to pass AWS credentials and invoke the commands below.

+

+ + + Directly passing in credentials parsed from ~/.aws/credentials + + +

+ + +
docker run \
+-e AWS_REGION=<region> \
+-e AWS_SESSION_TOKEN="$(grep 'aws_session_token' ~/.aws/credentials | awk -F' = ' '{print $2}')" \
+-e AWS_ACCESS_KEY_ID="$(grep 'aws_access_key_id' ~/.aws/credentials | awk -F' = ' '{print $2}')" \
+-e AWS_SECRET_ACCESS_KEY="$(grep 'aws_secret_access_key' ~/.aws/credentials | awk -F' = ' '{print $2}')" \
+-e LAKEFS_SGC_LAKEFS_ENDPOINT_URL=<lakefs endpoint URL> \
+-e LAKEFS_SGC_LAKEFS_ACCESS_KEY_ID=<lakefs accesss key> \
+-e LAKEFS_SGC_LAKEFS_SECRET_ACCESS_KEY=<lakefs secret key> \
+-e LAKEFS_SGC_LOGGING_LEVEL=debug \
+treeverse/lakefs-sgc:<tag> run <repository>
+
+

+ + + Mounting the ~/.aws directory + + +

+ + +

When working with S3-compatible clients, it’s often more convenient to mount the ~/.aws directory and pass in the desired profile.

+ +

First, change the permissions for ~/.aws/* to allow the docker container to read this directory:

+
chmod 644 ~/.aws/*
+
+ +

Then, run the docker image and mount ~/.aws to the lakefs-sgc home directory on the docker container:

+
docker run \
+--network=host \
+-v ~/.aws:/home/lakefs-sgc/.aws \
+-e AWS_REGION=us-east-1 \
+-e AWS_PROFILE=<profile> \
+-e LAKEFS_SGC_LAKEFS_ENDPOINT_URL=<lakefs endpoint URL> \
+-e LAKEFS_SGC_LAKEFS_ACCESS_KEY_ID=<lakefs accesss key> \
+-e LAKEFS_SGC_LAKEFS_SECRET_ACCESS_KEY=<lakefs secret key> \
+-e LAKEFS_SGC_LOGGING_LEVEL=debug \
+treeverse/lakefs-sgc:<tag> run <repository>
+
+

+ + + Get the List of Objects Marked for Deletion + + +

+ + +

lakefs-sgc will write its reports to <REPOSITORY_STORAGE_NAMESPACE>/_lakefs/retention/gc/reports/<RUN_ID>/.
+RUN_ID is generated during runtime by the Standalone GC. You can find it in the logs:

+
"Marking objects for deletion" ... run_id=gcoca17haabs73f2gtq0
+
+ +

In this prefix, you’ll find 2 objects:

+
    +
  • deleted.csv - Containing all marked objects in a CSV containing one address column. Example: +
     address
    + "data/gcnobu7n2efc74lfa5ug/csfnri7n2efc74lfa69g,_e7P9j-1ahTXtofw7tWwJUIhTfL0rEs_dvBrClzc_QE"
    + "data/gcnobu7n2efc74lfa5ug/csfnri7n2efc74lfa78g,mKZnS-5YbLzmK0pKsGGimdxxBlt8QZzCyw1QeQrFvFE"
    + ...
    +
    +
  • +
  • summary.json - A small json summarizing the GC run. Example: +
     {
    +     "run_id": "gcoca17haabs73f2gtq0",
    +     "success": true,
    +     "first_slice": "gcss5tpsrurs73cqi6e0",
    +     "start_time": "2024-10-27T13:19:26.890099059Z",
    +     "cutoff_time": "2024-10-27T07:19:26.890099059Z",
    +     "num_deleted_objects": 33000
    + }
    +
    +
  • +
+

+ + + Delete marked objects + + +

+ + +

We recommend starting by backing up the marked objects to a different bucket before deleting them. After ensuring the +backup is complete, you can proceed to delete the objects directly from the backup location.

+ +

Use the following script to backup marked objects to another bucket:

+
# Update these variables with your actual values
+storage_ns=<storage namespace (s3://...)>
+output_bucket=<output bucket (s3://...)>
+run_id=<GC run id>
+
+# Download the CSV file
+aws s3 cp "$storage_ns/_lakefs/retention/gc/reports/$run_id/deleted.csv" "./run_id-$run_id.csv"
+
+# Move all addresses to the output bucket under the "run_id=$run_id" prefix
+cat run_id-$run_id.csv | tail -n +2 | xargs -I {} aws s3 mv "$storage_ns/{}" "$output_bucket/run_id=$run_id/"
+
+ +

To delete the marked objects, use the following script:

+
# Update these variables with your actual values
+output_bucket=<output bucket (s3://...)>
+run_id=<GC run id>
+
+aws s3 rm $output_bucket/run_id=$run_id --recursive
+
+ +
+

Tip: Remember to periodically delete the backups to actually reduce storage costs.

+
+

+ + + Lab tests + + +

+ + +

Standalone GC was tested on the lakeFS setup below.

+

+ + + Repository spec + + +

+ + +
    +
  • 100k objects
  • +
  • 250 commits
  • +
  • 100 branches
  • +
+

+ + + Machine spec + + +

+ + +
    +
  • 4GiB RAM
  • +
  • 8 CPUs
  • +
+

+ + + Testing results + + +

+ + +
    +
  • Time: < 5m
  • +
  • Disk space: 123MB
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/garbage-collection/uncommitted.html b/v1.46/howto/garbage-collection/uncommitted.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/howto/garbage-collection/uncommitted.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/hooks/airflow.html b/v1.46/howto/hooks/airflow.html new file mode 100644 index 000000000..bfa2e3479 --- /dev/null +++ b/v1.46/howto/hooks/airflow.html @@ -0,0 +1,845 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Airflow Hooks | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Airflow Hooks + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Action file Airflow hook properties
  2. +
  3. Hook Record in configuration field
  4. +
+ +
+ +

Airflow Hook triggers a DAG run in an Airflow installation using Airflow’s REST API. +The hook run succeeds if the DAG was triggered, and fails otherwise.

+

+ + + Action file Airflow hook properties + + +

+ + +

See the Action configuration for overall configuration schema and details.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PropertyDescriptionData TypeExampleRequiredEnvironment Variables Supported
urlThe URL of the Airflow instanceStringhttp://localhost:8080yesno
dag_idThe DAG to triggerStringexample_dagyesno
usernameThe name of the Airflow user performing the requestStringadminyesno
passwordThe password of the Airflow user performing the requestStringadminyesyes
dag_confDAG run configuration that will be passed as isJSON nono
wait_for_dagWait for DAG run to complete and reflect state (default: false)Boolean nono
timeoutTime to wait for the DAG run to complete (default: 1m)String (golang’s Duration representation) nono
+ +

Example:

+
...
+hooks:
+  - id: trigger_my_dag
+    type: airflow
+    description: Trigger an example_dag
+    properties:
+       url: "http://localhost:8000"
+       dag_id: "example_dag"
+       username: "admin"
+       password: "{{ ENV.AIRFLOW_SECRET }}"
+       dag_conf:
+          some: "additional_conf"
+...
+
+

+ + + Hook Record in configuration field + + +

+ + +

lakeFS will add an entry to the Airflow request configuration property (conf) with the event that triggered the action.

+ +

The key of the record will be lakeFS_event and the value will match the one described here

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/hooks/index.html b/v1.46/howto/hooks/index.html new file mode 100644 index 000000000..5e60b4ae5 --- /dev/null +++ b/v1.46/howto/hooks/index.html @@ -0,0 +1,1059 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Actions and Hooks | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Actions and Hooks in lakeFS + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Overview
  2. +
  3. Configuration
  4. +
  5. Supported Events
  6. +
  7. Runs API & CLI
  8. +
  9. Result Files
  10. +
+ +
+ +

Like other version control systems, lakeFS allows you to configure Actions to trigger when predefined events occur. There are numerous uses for Actions, including:

+ +
    +
  1. Format Validator: +A webhook that checks new files to ensure they are of a set of allowed data formats.
  2. +
  3. Schema Validator: +A webhook that reads new Parquet and ORC files to ensure they don’t contain a block list of column names (or name prefixes). +This is useful for avoiding accidental PII exposure.
  4. +
  5. Integration with external systems: +Post-merge and post-commit hooks could be used to export metadata about the change to another system. A common example is exporting symlink.txt files that allow e.g. AWS Athena to read data from lakeFS.
  6. +
  7. Notifying downstream consumers: +Running a post-merge hook to trigger an Airflow DAG or to send a Webhook to an API, notifying it of the change that happened
  8. +
+ +

For step-by-step examples of hooks in action check out the lakeFS Quickstart and the lakeFS samples repository.

+ + +

+ + + Overview + + +

+ + +

An action defines one or more hooks to execute. lakeFS supports three types of hook:

+ +
    +
  1. Lua - uses an embedded Lua VM
  2. +
  3. Webhook - makes a REST call to an external URL
  4. +
  5. Airflow - triggers a DAG in Airflow
  6. +
+ +

“Before” hooks must run successfully before their action. If the hook fails, it aborts the action. Lua hooks and Webhooks are synchronous, and lakeFS waits for them to run to completion. Airflow hooks are asynchronous: lakeFS stops waiting as soon as Airflow accepts triggering the DAG.

+

+ + + Configuration + + +

+ + +

There are two parts to configuration an Action:

+ +
    +
  1. Create an Action file and upload it to the lakeFS repository
  2. +
  3. Configure the hook(s) that you specified in the Action file. How these are configured will depend on the type of hook.
  4. +
+

+ + + Action files + + +

+ + +

An Action is a list of Hooks with the same trigger configuration, i.e. an event will trigger all Hooks under an Action or none at all.

+ +

The Hooks under an Action are ordered and so is their execution.

+ +

Before each hook execution the if boolean expression is evaluated. The expression can use the functions success() and failure(), which return true if the hook’s actions succeeded or failed, respectively.

+ +

By default, when if is empty or omitted, the step will run only if no error occurred (the same as success()).

+

+ + + Action File schema + + +

+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PropertyDescriptionData TypeRequiredDefault Value
name Identifes the Action fileStringnoAction filename
on List of events that will trigger the hooksListyes 
on<event>.branches Glob pattern list of branches that triggers the hooksListnoNot applicable to Tag events. If empty, Action runs on all branches
hooks List of hooks to be executedListyes 
hook.id ID of the hook, must be unique within the action.Stringyes 
hook.type Type of the hook (types)Stringyes 
hook.description Description for the hookStringno 
hook.if Expression that will be evaluated before execute the hookStringnoNo value is the same as evaluate success()
hook.properties Hook’s specific configuration, see Lua, WebHook, and Airflow for detailsDictionarytrue 
+

+ + + Example Action File + + +

+ + +
name: Good files check
+description: set of checks to verify that branch is good
+on:
+  pre-commit:
+  pre-merge:
+    branches:
+      - main
+hooks:
+  - id: no_temp
+    type: webhook
+    description: checking no temporary files found
+    properties:
+      url: "https://example.com/webhook?notmp=true?t=1za2PbkZK1bd4prMuTDr6BeEQwWYcX2R"
+  - id: no_freeze
+    type: webhook
+    description: check production is not in dev freeze
+    properties:
+      url: "https://example.com/webhook?nofreeze=true?t=1za2PbkZK1bd4prMuTDr6BeEQwWYcX2R"
+  - id: alert
+    type: webhook
+    if: failure()
+    description: notify alert system when check failed
+    properties:
+       url: "https://example.com/alert"
+       query_params:
+          title: good files webhook failed
+  - id: notification
+    type: webhook
+    if: true
+    description: notify that will always run - no matter if one of the previous steps failed
+    properties:
+       url: "https://example.com/notification"
+       query_params:
+          title: good files completed
+
+ +

Note: lakeFS will validate action files only when an Event has occurred.
+Use lakectl actions validate <path> to validate your action files locally.

+

+ + + Uploading Action files + + +

+ + +

Action files should be uploaded with the prefix _lakefs_actions/ to the lakeFS repository. +When an actionable event (see Supported Events above) takes place, lakeFS will read all files with prefix _lakefs_actions/ +in the repository branch where the action occurred. +A failure to parse an Action file will result with a failing Run.

+ +

For example, lakeFS will search and execute all the matching Action files with the prefix lakefs://example-repo/feature-1/_lakefs_actions/ on:

+
    +
  1. Commit to feature-1 branch on example-repo repository.
  2. +
  3. Merge to main branch from feature-1 branch on repo1 repository.
  4. +
+

+ + + Supported Events + + +

+ + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
EventDescription
pre-commitRuns when the commit occurs, before the commit is finalized
post-commitRuns after the commit is finalized
pre-mergeRuns on the source branch when the merge occurs, before the merge is finalized
post-mergeRuns on the merge result, after the merge is finalized
pre-create-branchRuns on the source branch prior to creating a new branch
post-create-branchRuns on the new branch after the branch was created
pre-delete-branchRuns prior to deleting a branch
post-delete-branchRuns after the branch was deleted
pre-create-tagRuns prior to creating a new tag
post-create-tagRuns after the tag was created
pre-delete-tagRuns prior to deleting a tag
post-delete-tagRuns after the tag was deleted
+ +

lakeFS Actions are handled per repository and cannot be shared between repositories. +A failure of any Hook under any Action of a pre-* event will result in aborting the lakeFS operation that is taking place. +Hook failures under any Action of a post-* event will not revert the operation.

+ +

Hooks are managed by Action files that are written to a prefix in the lakeFS repository. +This allows configuration-as-code inside lakeFS, where Action files are declarative and written in YAML.

+

+ + + Runs API & CLI + + +

+ + +

A Run is an instantiation of the repository’s Action files when the triggering event occurs. +For example, if your repository contains a pre-commit hook, every commit would generate a Run for that specific commit.

+ +

lakeFS will fetch, parse and filter the repository Action files and start to execute the Hooks under each Action. +All executed Hooks (each with hook_run_id) exist in the context of that Run (run_id).

+ +

The lakeFS API and lakectl expose the results of executions per repository, branch, commit, and specific Action. +The endpoint also allows to download the execution log of any executed Hook under each Run for observability.

+

+ + + Result Files + + +

+ + +

The metadata section of lakeFS repository with each Run contains two types of files:

+
    +
  1. _lakefs/actions/log/<runID>/<hookRunID>.log - Execution log of the specific Hook run.
  2. +
  3. _lakefs/actions/log/<runID>/run.manifest - Manifest with all Hooks execution for the run with their results and additional metadata.
  4. +
+ +

Note: Metadata section of a lakeFS repository is where lakeFS keeps its metadata, like commits and metaranges. +Metadata files stored in the metadata section aren’t accessible like user stored files.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/hooks/lua.html b/v1.46/howto/hooks/lua.html new file mode 100644 index 000000000..e5cf89422 --- /dev/null +++ b/v1.46/howto/hooks/lua.html @@ -0,0 +1,2458 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Lua Hooks | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Lua Hooks + + +

+ + +

lakeFS supports running hooks without relying on external components using an embedded Lua VM

+ +

Using Lua hooks, it is possible to pass a Lua script to be executed directly by the lakeFS server when an action occurs.

+ +

The Lua runtime embedded in lakeFS is limited for security reasons. It provides a narrow set of APIs and functions that by default do not allow:

+ +
    +
  1. Accessing any of the running lakeFS server’s environment
  2. +
  3. Accessing the local filesystem available the lakeFS process
  4. +
+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Action File Lua Hook Properties
  2. +
  3. Example Lua Hooks
  4. +
  5. Lua Library reference
  6. +
+ +
+

+ + + Action File Lua Hook Properties + + +

+ + +

See the Action configuration for overall configuration schema and details.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PropertyDescriptionData TypeRequiredDefault Value
argsOne or more arguments to pass to the hookDictionaryfalse 
scriptAn inline Lua scriptStringeither this or script_path must be specified 
script_pathThe path in lakeFS to a Lua scriptStringeither this or script must be specified 
+

+ + + Example Lua Hooks + + +

+ + +

For more examples and configuration samples, check out the examples/hooks/ directory in the lakeFS repository. You’ll also find step-by-step examples of hooks in action in the lakeFS samples repository.

+

+ + + Display information about an event + + +

+ + +

This example will print out a JSON representation of the event that occurred:

+ +
name: dump_all
+on:
+  post-commit:
+  post-merge:
+  post-create-tag:
+  post-create-branch:
+hooks:
+  - id: dump_event
+    type: lua
+    properties:
+      script: |
+        json = require("encoding/json")
+        print(json.marshal(action))
+
+

+ + + Ensure that a commit includes a mandatory metadata field + + +

+ + +

A more useful example: ensure every commit contains a required metadata field:

+ +
name: pre commit metadata field check
+on:
+pre-commit:
+    branches:
+    - main
+    - dev
+hooks:
+  - id: ensure_commit_metadata
+    type: lua
+    properties:
+      args:
+        notebook_url: {"pattern": "my-jupyter.example.com/.*"}
+        spark_version:  {}
+      script_path: lua_hooks/ensure_metadata_field.lua
+
+ +

Lua code at lakefs://repo/main/lua_hooks/ensure_metadata_field.lua:

+ +
regexp = require("regexp")
+for k, props in pairs(args) do
+  current_value = action.commit.metadata[k]
+  if current_value == nil then
+    error("missing mandatory metadata field: " .. k)
+  end
+  if props.pattern and not regexp.match(props.pattern, current_value) then
+    error("current value for commit metadata field " .. k .. " does not match pattern: " .. props.pattern .. " - got: " .. current_value)
+  end
+end
+
+ +

For more examples and configuration samples, check out the examples/hooks/ directory in the lakeFS repository.

+

+ + + Lua Library reference + + +

+ + +

The Lua runtime embedded in lakeFS is limited for security reasons. The provided APIs are shown below.

+

+ + + array(table) + + +

+ + +

Helper function to mark a table object as an array for the runtime by setting _is_array: true metatable field.

+

+ + + aws + + +

+ +

+ + + aws/s3_client + + +

+ +

S3 client library.

+ +
local aws = require("aws")
+-- pass valid AWS credentials
+local client = aws.s3_client("ACCESS_KEY_ID", "SECRET_ACCESS_KEY", "REGION")
+
+

+ + + aws/s3_client.get_object(bucket, key) + + +

+ + +

Returns the body (as a Lua string) of the requested object and a boolean value that is true if the requested object exists

+

+ + + aws/s3_client.put_object(bucket, key, value) + + +

+ + +

Sets the object at the given bucket and key to the value of the supplied value string

+

+ + + aws/s3_client.delete_object(bucket [, key]) + + +

+ + +

Deletes the object at the given key

+

+ + + aws/s3_client.list_objects(bucket [, prefix, continuation_token, delimiter]) + + +

+ + +

Returns a table of results containing the following structure:

+ +
    +
  • is_truncated: (boolean) whether there are more results to paginate through using the continuation token
  • +
  • next_continuation_token: (string) to pass in the next request to get the next page of results
  • +
  • results (table of tables) information about the objects (and prefixes if a delimiter is used)
  • +
+ +

a result could in one of the following structures

+ +
{
+   ["key"] = "a/common/prefix/",
+   ["type"] = "prefix"
+}
+
+ +

or:

+ +
{
+   ["key"] = "path/to/object",
+   ["type"] = "object",
+   ["etag"] = "etagString",
+   ["size"] = 1024,
+   ["last_modified"] = "2023-12-31T23:10:00Z"
+}
+
+

+ + + aws/s3_client.delete_recursive(bucket, prefix) + + +

+ + +

Deletes all objects under the given prefix

+

+ + + aws/glue + + +

+ + +

Glue client library.

+ +
local aws = require("aws")
+-- pass valid AWS credentials
+local glue = aws.glue_client("ACCESS_KEY_ID", "SECRET_ACCESS_KEY", "REGION")
+
+

+ + + aws/glue.create_database(database, options) + + +

+ + +

Create a new Database in Glue Catalog.

+ +

Parameters:

+ +
    +
  • database(string): Glue Database name.
  • +
  • options(table) (optional): +
      +
    • error_on_already_exists(boolean): Whether the call fail with an error if a DB with this name already exists
    • +
    • create_db_input(Table): a Table that is passed “as is” to AWS and is parallel to the AWS SDK CreateDatabaseInput
    • +
    +
  • +
+ +

Example:

+ +
local opts = {
+    error_on_already_exists = false,
+    create_db_input = {DatabaseInput = {Description = "Created via LakeFS Action"}, Tags = {Owner = "Joe"}}
+}
+glue.create_database(db, opts)
+
+

+ + + aws/glue.delete_database(database, catalog_id) + + +

+ + +

Delete an existing Database in Glue Catalog.

+ +

Parameters:

+ +
    +
  • database(string): Glue Database name.
  • +
  • catalog_id(string) (optional): Glue Catalog ID
  • +
+ +

Example:

+ +
glue.delete_database(db, "461129977393")
+
+

+ + + aws/glue.get_table(database, table [, catalog_id) + + +

+ + +

Describe a table from the Glue Catalog.

+ +

Example:

+ +
local table, exists = glue.get_table(db, table_name)
+if exists then
+  print(json.marshal(table))
+
+

+ + + aws/glue.create_table(database, table_input, [, catalog_id]) + + +

+ + +

Create a new table in Glue Catalog. +The table_input argument is a JSON that is passed “as is” to AWS and is parallel to the AWS SDK TableInput

+ +

Example:

+ +
local json = require("encoding/json")
+local input = {
+    Name = "my-table",
+    PartitionKeys = array(partitions),
+    -- etc...
+}
+local json_input = json.marshal(input)
+glue.create_table("my-db", table_input)
+
+

+ + + aws/glue.update_table(database, table_input, [, catalog_id, version_id, skip_archive]) + + +

+ + +

Update an existing Table in Glue Catalog. +The table_input is the same as the argument in glue.create_table function.

+

+ + + aws/glue.delete_table(database, table_input, [, catalog_id]) + + +

+ + +

Delete an existing Table in Glue Catalog.

+

+ + + azure + + +

+ +

+ + + azure/blob_client + + +

+ +

Azure blob client library.

+ +
local azure = require("azure")
+-- pass valid Azure credentials
+local client = azure.blob_client("AZURE_STORAGE_ACCOUNT", "AZURE_ACCESS_KEY")
+
+

+ + + azure/blob_client.get_object(path_uri) + + +

+ + +

Returns the body (as a Lua string) of the requested object and a boolean value that is true if the requested object exists
+path_uri - A valid Azure blob storage uri in the form of https://myaccount.blob.core.windows.net/mycontainer/myblob

+

+ + + azure/blob_client.put_object(path_uri, value) + + +

+ + +

Sets the object at the given bucket and key to the value of the supplied value string
+path_uri - A valid Azure blob storage uri in the form of https://myaccount.blob.core.windows.net/mycontainer/myblob

+

+ + + azure/blob_client.delete_object(path_uri) + + +

+ + +

Deletes the object at the given key
+path_uri - A valid Azure blob storage uri in the form of https://myaccount.blob.core.windows.net/mycontainer/myblob

+

+ + + azure/abfss_transform_path(path) + + +

+ + +

Transform an HTTPS Azure URL to a ABFSS scheme. Used by the delta_exporter function to support Azure Unity catalog use cases
+path - A valid Azure blob storage URL in the form of https://myaccount.blob.core.windows.net/mycontainer/myblob

+

+ + + crypto + + +

+ +

+ + + crypto/aes/encryptCBC(key, plaintext) + + +

+ + +

Returns a ciphertext for the aes encrypted text

+

+ + + crypto/aes/decryptCBC(key, ciphertext) + + +

+ + +

Returns the decrypted (plaintext) string for the encrypted ciphertext

+

+ + + crypto/hmac/sign_sha256(message, key) + + +

+ + +

Returns a SHA256 hmac signature for the given message with the supplied key (using the SHA256 hashing algorithm)

+

+ + + crypto/hmac/sign_sha1(message, key) + + +

+ + +

Returns a SHA1 hmac signature for the given message with the supplied key (using the SHA1 hashing algorithm)

+

+ + + crypto/md5/digest(data) + + +

+ + +

Returns the MD5 digest (string) of the given data

+

+ + + crypto/sha256/digest(data) + + +

+ + +

Returns the SHA256 digest (string) of the given data

+

+ + + databricks/client(databricks_host, databricks_service_principal_token) + + +

+ + +

Returns a table representing a Databricks client with the register_external_table and create_or_get_schema methods.

+

+ + + databricks/client.create_schema(schema_name, catalog_name, get_if_exists) + + +

+ + +

Creates a schema, or retrieves it if exists, in the configured Databricks host’s Unity catalog. +If a schema doesn’t exist, a new schema with the given schema_name will be created under the given catalog_name. +Returns the created/fetched schema name.

+ +

Parameters:

+ +
    +
  • schema_name(string): The required schema name
  • +
  • catalog_name(string): The catalog name under which the schema will be created (or from which it will be fetched)
  • +
  • get_if_exists(boolean): In case of failure due to an existing schema with the given schema_name in the given +catalog_name, return the schema.
  • +
+ +

Example:

+ +
local databricks = require("databricks")
+local client = databricks.client("https://my-host.cloud.databricks.com", "my-service-principal-token")
+local schema_name = client.create_schema("main", "mycatalog", true)
+
+

+ + + databricks/client.register_external_table(table_name, physical_path, warehouse_id, catalog_name, schema_name, metadata) + + +

+ + +

Registers an external table under the provided warehouse ID, catalog name, and schema name. +In order for this method call to succeed, an external location should be configured in the catalog, with the +physical_path’s root storage URI (for example: s3://mybucket). +Returns the table’s creation status.

+ +

Parameters:

+ +
    +
  • table_name(string): Table name.
  • +
  • physical_path(string): A location to which the external table will refer, e.g. s3://mybucket/the/path/to/mytable.
  • +
  • warehouse_id(string): The SQL warehouse ID used in Databricks to run the CREATE TABLE query (fetched from the SQL warehouse +Connection Details, or by running databricks warehouses get, choosing your SQL warehouse and fetching its ID).
  • +
  • catalog_name(string): The name of the catalog under which a schema will be created (or fetched from).
  • +
  • schema_name(string): The name of the schema under which the table will be created.
  • +
  • metadata(table): A table of metadata to be added to the table’s registration. The metadata table should be of the form: +{key1 = "value1", key2 = "value2", ...}.
  • +
+ +

Example:

+ +
local databricks = require("databricks")
+local client = databricks.client("https://my-host.cloud.databricks.com", "my-service-principal-token")
+local status = client.register_external_table("mytable", "s3://mybucket/the/path/to/mytable", "examwarehouseple", "my-catalog-name", "myschema")
+
+ + +

+ + + encoding/base64/encode(data) + + +

+ + +

Encodes the given data to a base64 string

+

+ + + encoding/base64/decode(data) + + +

+ + +

Decodes the given base64 encoded data and return it as a string

+

+ + + encoding/base64/url_encode(data) + + +

+ + +

Encodes the given data to an unpadded alternate base64 encoding defined in RFC 4648.

+

+ + + encoding/base64/url_decode(data) + + +

+ + +

Decodes the given unpadded alternate base64 encoding defined in RFC 4648 and return it as a string

+

+ + + encoding/hex/encode(value) + + +

+ + +

Encode the given value string to hexadecimal values (string)

+

+ + + encoding/hex/decode(value) + + +

+ + +

Decode the given hexadecimal string back to the string it represents (UTF-8)

+

+ + + encoding/json/marshal(table) + + +

+ + +

Encodes the given table into a JSON string

+

+ + + encoding/json/unmarshal(string) + + +

+ + +

Decodes the given string into the equivalent Lua structure

+

+ + + encoding/yaml/marshal(table) + + +

+ + +

Encodes the given table into a YAML string

+

+ + + encoding/yaml/unmarshal(string) + + +

+ + +

Decodes the given YAML encoded string into the equivalent Lua structure

+

+ + + encoding/parquet/get_schema(payload) + + +

+ + +

Read the payload (string) as the contents of a Parquet file and return its schema in the following table structure:

+ +
{
+  { ["name"] = "column_a", ["type"] = "INT32" },
+  { ["name"] = "column_b", ["type"] = "BYTE_ARRAY" }
+}
+
+

+ + + formats + + +

+ +

+ + + formats/delta_client(key, secret, region) + + +

+ + +

Creates a new Delta Lake client used to interact with the lakeFS server.

+
    +
  • key: lakeFS access key id
  • +
  • secret: lakeFS secret access key
  • +
  • region: The region in which your lakeFS server is configured at.
  • +
+

+ + + formats/delta_client.get_table(repository_id, reference_id, prefix) + + +

+ + +

Returns a representation of a Delta Lake table under the given repository, reference, and prefix. +The format of the response is two tables:

+
    +
  1. the first is a table of the format {number, {string}} where number is a version in the Delta Log, and the mapped {string} +array contains JSON strings of the different Delta Lake log operations listed in the mapped version entry. e.g.: +
    {
    +  0 = {
    + "{\"commitInfo\":...}",
    + "{\"add\": ...}",
    + "{\"remove\": ...}"
    +  },
    +  1 = {
    + "{\"commitInfo\":...}",
    + "{\"add\": ...}",
    + "{\"remove\": ...}"
    +  }
    +}
    +
    +
  2. +
  3. the second is a table of the metadata of the current table snapshot. The metadata table can be used to initialize the Delta Lake table in an external Catalog.
    +It consists of the following fields: +
      +
    • id: The table’s ID
    • +
    • name: The table’s name
    • +
    • description: The table’s description
    • +
    • schema_string: The table’s schema string
    • +
    • partition_columns: The table’s partition columns
    • +
    • configuration: The table’s configuration
    • +
    • created_time: The table’s creation time
    • +
    +
  4. +
+

+ + + gcloud + + +

+ +

+ + + gcloud/gs_client(gcs_credentials_json_string) + + +

+ + +

Create a new Google Cloud Storage client using a string that contains a valid credentials.json file content.

+

+ + + gcloud/gs.write_fuse_symlink(source, destination, mount_info) + + +

+ + +

Will create a gcsfuse symlink +from the source (typically a lakeFS physical address for an object) to a given destination.

+ +

mount_info is a Lua table with "from" and "to" keys - since symlinks don’t work for gs://... URIs, they need to point +to the mounted location instead. from will be removed from the beginning of source, and destination will be added instead.

+ +

Example:

+ +
source = "gs://bucket/lakefs/data/abc/def"
+destination = "gs://bucket/exported/path/to/object"
+mount_info = {
+    ["from"] = "gs://bucket",
+    ["to"] = "/home/user/gcs-mount"
+}
+gs.write_fuse_symlink(source, destination, mount_info)
+-- Symlink: "/home/user/gcs-mount/exported/path/to/object" -> "/home/user/gcs-mount/lakefs/data/abc/def"
+
+

+ + + hook + + +

+ + +

A set of utilities to aide in writing user friendly hooks.

+

+ + + hook/fail(message) + + +

+ + +

Will abort the current hook’s execution with the given message. This is similar to using error(), but is typically used to separate +generic runtime errors (an API call that returned an unexpected response) and explict failure of the calling hook.

+ +

When called, errors will appear without a stacktrace, and the error message will be directly the one given as message.

+ +
> hook = require("hook")
+> hook.fail("this hook shall not pass because of: " .. reason)
+
+

+ + + lakefs + + +

+ + +

The Lua Hook library allows calling back to the lakeFS API using the identity of the user that triggered the action. +For example, if user A tries to commit and triggers a pre-commit hook - any call made inside that hook to the lakeFS +API, will automatically use user A’s identity for authorization and auditing purposes.

+

+ + + lakefs/create_tag(repository_id, reference_id, tag_id) + + +

+ + +

Create a new tag for the given reference

+

+ + + lakefs/diff_refs(repository_id, lef_reference_id, right_reference_id [, after, prefix, delimiter, amount]) + + +

+ + +

Returns an object-wise diff between left_reference_id and right_reference_id.

+

+ + + lakefs/list_objects(repository_id, reference_id [, after, prefix, delimiter, amount]) + + +

+ + +

List objects in the specified repository and reference (branch, tag, commit ID, etc.). +If delimiter is empty, will default to a recursive listing. Otherwise, common prefixes up to delimiter will be shown as a single entry.

+

+ + + lakefs/get_object(repository_id, reference_id, path) + + +

+ + +

Returns 2 values:

+ +
    +
  1. The HTTP status code returned by the lakeFS API
  2. +
  3. The content of the specified object as a lua string
  4. +
+

+ + + lakefs/diff_branch(repository_id, branch_id [, after, amount, prefix, delimiter]) + + +

+ + +

Returns an object-wise diff of uncommitted changes on branch_id.

+

+ + + lakefs/stat_object(repository_id, ref_id, path) + + +

+ + +

Returns a stat object for the given path under the given reference and repository.

+

+ + + lakefs/catalogexport/glue_exporter.get_full_table_name(descriptor, action_info) + + +

+ + +

Generate glue table name.

+ +

Parameters:

+ +
    +
  • descriptor(Table): Object from (e.g. _lakefs_tables/my_table.yaml).
  • +
  • action_info(Table): The global action object.
  • +
+

+ + + lakefs/catalogexport/delta_exporter + + +

+ + +

A package used to export Delta Lake tables from lakeFS to an external cloud storage.

+

+ + + lakefs/catalogexport/delta_exporter.export_delta_log(action, table_def_names, write_object, delta_client, table_descriptors_path, path_transformer) + + +

+ + +

The function used to export Delta Lake tables. +The return value is a table with mapping of table names to external table location (from which it is possible to query the data) and latest Delta table version’s metadata.
+The response is of the form: +{<table_name> = {path = "s3://mybucket/mypath/mytable", metadata = {id = "table_id", name = "table_name", ...}}}.

+ +

Parameters:

+ +
    +
  • action: The global action object
  • +
  • table_def_names: Delta tables name list (e.g. {"table1", "table2"})
  • +
  • write_object: A writer function with function(bucket, key, data) signature, used to write the exported Delta Log (e.g. aws/s3_client.put_object or azure/blob_client.put_object)
  • +
  • delta_client: A Delta Lake client that implements get_table: function(repo, ref, prefix)
  • +
  • table_descriptors_path: The path under which the table descriptors of the provided table_def_names reside
  • +
  • path_transformer: (Optional) A function(path) used for transforming the path of the saved delta logs path fields as well as the saved table physical path (used to support Azure Unity catalog use cases)
  • +
+ +

Delta export example for AWS S3:

+ +
---
+name: delta_exporter
+on:
+  post-commit: null
+hooks:
+  - id: delta_export
+    type: lua
+    properties:
+      script: |
+        local aws = require("aws")
+        local formats = require("formats")
+        local delta_exporter = require("lakefs/catalogexport/delta_exporter")
+        local json = require("encoding/json")
+
+        local table_descriptors_path = "_lakefs_tables"
+        local sc = aws.s3_client(args.aws.access_key_id, args.aws.secret_access_key, args.aws.region)
+        local delta_client = formats.delta_client(args.lakefs.access_key_id, args.lakefs.secret_access_key, args.aws.region)
+        local delta_table_details = delta_export.export_delta_log(action, args.table_defs, sc.put_object, delta_client, table_descriptors_path)
+        
+        for t, details in pairs(delta_table_details) do
+          print("Delta Lake exported table \"" .. t .. "\"'s location: " .. details["path"] .. "\n")
+          print("Delta Lake exported table \"" .. t .. "\"'s metadata:\n")
+          for k, v in pairs(details["metadata"]) do
+            if type(v) == "table" then
+              print("\t" .. k .. " = " .. json.marshal(v) .. "\n")
+            else
+              print("\t" .. k .. " = " .. v .. "\n")
+            end
+          end
+        end
+      args:
+        aws:
+          access_key_id: <AWS_ACCESS_KEY_ID>
+          secret_access_key: <AWS_SECRET_ACCESS_KEY>
+          region: us-east-1
+        lakefs:
+          access_key_id: <LAKEFS_ACCESS_KEY_ID> 
+          secret_access_key: <LAKEFS_SECRET_ACCESS_KEY>
+        table_defs:
+          - mytable
+
+ +

For the table descriptor under the _lakefs_tables/mytable.yaml:

+
---
+name: myTableActualName
+type: delta
+path: a/path/to/my/delta/table
+
+ +

Delta export example for Azure Blob Storage:

+ +
name: Delta Exporter
+on:
+  post-commit:
+    branches: ["*"]
+hooks:
+  - id: delta_exporter
+    type: lua
+    properties:
+      script: |
+        local azure = require("azure")
+        local formats = require("formats")
+        local delta_exporter = require("lakefs/catalogexport/delta_exporter")
+
+        local table_descriptors_path = "_lakefs_tables"
+        local sc = azure.blob_client(args.azure.storage_account, args.azure.access_key)
+        local function write_object(_, key, buf)
+          return sc.put_object(key,buf)
+        end
+        local delta_client = formats.delta_client(args.lakefs.access_key_id, args.lakefs.secret_access_key)
+        local delta_table_details = delta_export.export_delta_log(action, args.table_defs, sc.put_object, delta_client, table_descriptors_path)
+        
+        for t, details in pairs(delta_table_details) do
+          print("Delta Lake exported table \"" .. t .. "\"'s location: " .. details["path"] .. "\n")
+          print("Delta Lake exported table \"" .. t .. "\"'s metadata:\n")
+          for k, v in pairs(details["metadata"]) do
+            if type(v) == "table" then
+              print("\t" .. k .. " = " .. json.marshal(v) .. "\n")
+            else
+              print("\t" .. k .. " = " .. v .. "\n")
+            end
+          end
+        end
+      args:
+        azure:
+          storage_account: ""
+          access_key: ""
+        lakefs: # provide credentials of a user that has access to the script and Delta Table
+          access_key_id: ""
+          secret_access_key: ""
+        table_defs:
+          - mytable
+
+
+

+ + + lakefs/catalogexport/table_extractor + + +

+ + +

Utility package to parse _lakefs_tables/ descriptors.

+

+ + + lakefs/catalogexport/table_extractor.list_table_descriptor_entries(client, repo_id, commit_id) + + +

+ + +

List all YAML files under _lakefs_tables/* and return a list of type [{physical_address, path}], ignores hidden files. +The client is lakefs client.

+

+ + + lakefs/catalogexport/table_extractor.get_table_descriptor(client, repo_id, commit_id, logical_path) + + +

+ + +

Read a table descriptor and parse YAML object. Will set partition_columns to {} if no partitions are defined. +The client is lakefs client.

+

+ + + lakefs/catalogexport/hive.extract_partition_pager(client, repo_id, commit_id, base_path, partition_cols, page_size) + + +

+ + +

Hive format partition iterator each result set is a collection of files under the same partition in lakeFS.

+ +

Example:

+ +
local lakefs = require("lakefs")
+local pager = hive.extract_partition_pager(lakefs, repo_id, commit_id, prefix, partitions, 10)
+for part_key, entries in pager do
+    print("partition: " .. part_key)
+    for _, entry in ipairs(entries) do
+        print("path: " .. entry.path .. " physical: " .. entry.physical_address)
+    end
+end
+
+ + + +

Writes metadata for a table using Hive’s SymlinkTextInputFormat. +Currently only S3 is supported.

+ +

The default export paths per commit:

+ +
${storageNamespace}
+_lakefs/
+    exported/
+        ${ref}/
+            ${commitId}/
+                ${tableName}/
+                    p1=v1/symlink.txt
+                    p1=v2/symlink.txt
+                    p1=v3/symlink.txt
+                    ...
+
+ + + +

Export Symlink files that represent a table to S3 location.

+ +

Parameters:

+ +
    +
  • s3_client: Configured client.
  • +
  • table_src_path(string): Path to the table spec YAML file in _lakefs_tables (e.g. _lakefs_tables/my_table.yaml).
  • +
  • action_info(table): The global action object.
  • +
  • options(table): +
      +
    • debug(boolean): Print extra info.
    • +
    • export_base_uri(string): Override the prefix in S3 e.g. s3://other-bucket/path/.
    • +
    • writer(function(bucket, key, data)): If passed then will not use s3 client, helpful for debug.
    • +
    +
  • +
+ +

Example:

+ +
local exporter = require("lakefs/catalogexport/symlink_exporter")
+local aws = require("aws")
+-- args are user inputs from a lakeFS action.
+local s3 = aws.s3_client(args.aws.aws_access_key_id, args.aws.aws_secret_access_key, args.aws.aws_region)
+exporter.export_s3(s3, args.table_descriptor_path, action, {debug=true})
+
+

+ + + lakefs/catalogexport/glue_exporter + + +

+ + +

A Package for automating the export process from lakeFS stored tables into Glue catalog.

+

+ + + lakefs/catalogexport/glue_exporter.export_glue(glue, db, table_src_path, create_table_input, action_info, options) + + +

+ + +

Represent lakeFS table in Glue Catalog. +This function will create a table in Glue based on configuration. +It assumes that there is a symlink location that is already created and only configures it by default for the same commit.

+ +

Parameters:

+ +
    +
  • glue: AWS glue client
  • +
  • db(string): glue database name
  • +
  • table_src_path(string): path to table spec (e.g. _lakefs_tables/my_table.yaml)
  • +
  • create_table_input(table): Input equal mapping to table_input in AWS, the same as we use for glue.create_table. +should contain inputs describing the data format (e.g. InputFormat, OutputFormat, SerdeInfo) since the exporter is agnostic to this. +by default this function will configure table location and schema.
  • +
  • action_info(table): the global action object.
  • +
  • options(table): +
      +
    • table_name(string): Override default glue table name
    • +
    • debug(boolean
    • +
    • export_base_uri(string): Override the default prefix in S3 for symlink location e.g. s3://other-bucket/path/
    • +
    • create_db_input(table): if this is specified, then it indicates we want to create a new database for the table export. The parameter expects a table that is converted to JSON and passed “as is” to AWS and is parallel to the AWS SDK CreateDatabaseInput
    • +
    +
  • +
+ +

When creating a glue table, the final table input will consist of the create_table_input input parameter and lakeFS computed defaults that will override it:

+ +
    +
  • Name Gable table name get_full_table_name(descriptor, action_info).
  • +
  • PartitionKeys Partition columns usually deduced from _lakefs_tables/${table_src_path}.
  • +
  • TableType = “EXTERNAL_TABLE”
  • +
  • StorageDescriptor: Columns usually deduced from _lakefs_tables/${table_src_path}.
  • +
  • StorageDescriptor.Location = symlink_location
  • +
+ +

Example:

+ +
local aws = require("aws")
+local exporter = require("lakefs/catalogexport/glue_exporter")
+local glue = aws.glue_client(args.aws_access_key_id, args.aws_secret_access_key, args.aws_region)
+-- table_input can be passed as a simple Key-Value object in YAML as an argument from an action, this is inline example:
+local table_input = {
+  StorageDescriptor: 
+    InputFormat: "org.apache.hadoop.hive.ql.io.SymlinkTextInputFormat"
+    OutputFormat: "org.apache.hadoop.hive.ql.io.IgnoreKeyTextOutputFormat"
+    SerdeInfo:
+      SerializationLibrary: "org.apache.hadoop.hive.ql.io.parquet.serde.ParquetHiveSerDe"
+  Parameters: 
+    classification: "parquet"
+    EXTERNAL: "TRUE"
+    "parquet.compression": "SNAPPY"
+}
+exporter.export_glue(glue, "my-db", "_lakefs_tables/animals.yaml", table_input, action, {debug=true, create_db_input = {DatabaseInput = {Description="DB exported from LakeFS"}, Tags = {Owner = "Joe"}}})
+
+

+ + + lakefs/catalogexport/glue_exporter.get_full_table_name(descriptor, action_info) + + +

+ + +

Generate glue table name.

+ +

Parameters:

+ +
    +
  • descriptor(Table): Object from (e.g. _lakefs_tables/my_table.yaml).
  • +
  • action_info(Table): The global action object.
  • +
+

+ + + lakefs/catalogexport/unity_exporter + + +

+ + +

A package used to register exported Delta Lake tables to Databricks’ Unity catalog.

+

+ + + lakefs/catalogexport/unity_exporter.register_tables(action, table_descriptors_path, delta_table_details, databricks_client, warehouse_id) + + +

+ + +

The function used to register exported Delta Lake tables in Databricks’ Unity Catalog. +The registration will use the following paths to register the table: +<catalog>.<branch name>.<table_name> where the branch name will be used as the schema name. +The return value is a table with mapping of table names to registration request status.

+ +

Note: (Azure users) Databricks catalog external locations is supported only for ADLS Gen2 storage accounts.
+When exporting Delta tables using the lakefs/catalogexport/delta_exporter.export_delta_log function, the path_transformer must be
+used to convert the paths scheme to abfss. The built-in azure lua library provides this functionality with transformPathToAbfss.

+ +

Parameters:

+ +
    +
  • action(table): The global action table
  • +
  • table_descriptors_path(string): The path under which the table descriptors of the provided table_paths reside.
  • +
  • delta_table_details(table): Table names to physical paths mapping and table metadata (e.g. {table1 = {path = "s3://mybucket/mytable1", metadata = {id = "table_1_id", name = "table1", ...}}, table2 = {path = "s3://mybucket/mytable2", metadata = {id = "table_2_id", name = "table2", ...}}}.)
  • +
  • databricks_client(table): A Databricks client that implements create_or_get_schema: function(id, catalog_name) and register_external_table: function(table_name, physical_path, warehouse_id, catalog_name, schema_name)
  • +
  • warehouse_id(string): Databricks warehouse ID.
  • +
+ +

Example: +The following registers an exported Delta Lake table to Unity Catalog.

+ +
local databricks = require("databricks")
+local unity_export = require("lakefs/catalogexport/unity_exporter")
+
+local delta_table_locations = {
+  ["table1"] = "s3://mybucket/mytable1",
+}
+-- Register the exported table in Unity Catalog:
+local action_details = {
+  repository_id = "my-repo",
+  commit_id = "commit_id",
+  branch_id = "main",
+}
+local databricks_client = databricks.client("<DATABRICKS_HOST>", "<DATABRICKS_TOKEN>")
+local registration_statuses = unity_export.register_tables(action_details, "_lakefs_tables", delta_table_locations, databricks_client, "<WAREHOUSE_ID>")
+
+for t, status in pairs(registration_statuses) do
+  print("Unity catalog registration for table \"" .. t .. "\" completed with status: " .. status .. "\n")
+end
+
+ +

For the table descriptor under the _lakefs_tables/delta-table-descriptor.yaml:

+
---
+name: my_table_name
+type: delta
+path: path/to/delta/table/data
+catalog: my-catalog
+
+ +

For detailed step-by-step guide on how to use unity_exporter.register_tables as a part of a lakeFS action refer to +the Unity Catalog docs.

+

+ + + path/parse(path_string) + + +

+ + +

Returns a table for the given path string with the following structure:

+ +
> require("path")
+> path.parse("a/b/c.csv")
+{
+    ["parent"] = "a/b/"
+    ["base_name"] = "c.csv"
+} 
+
+

+ + + path/join(*path_parts) + + +

+ + +

Receives a variable number of strings and returns a joined string that represents a path:

+ +
> path = require("path")
+> path.join("/", "path/", "to", "a", "file.data")
+path/o/a/file.data
+
+

+ + + path/is_hidden(path_string [, seperator, prefix]) + + +

+ + +

returns a boolean - true if the given path string is hidden (meaning it starts with prefix) - or if any of its parents start with prefix.

+ +
> require("path")
+> path.is_hidden("a/b/c") -- false
+> path.is_hidden("a/b/_c") -- true
+> path.is_hidden("a/_b/c") -- true
+> path.is_hidden("a/b/_c/") -- true
+
+

+ + + path/default_separator() + + +

+ + +

Returns a constant string (/)

+

+ + + regexp/match(pattern, s) + + +

+ + +

Returns true if the string s matches pattern. +This is a thin wrapper over Go’s regexp.MatchString.

+

+ + + regexp/quote_meta(s) + + +

+ + +

Escapes any meta-characters in string s and returns a new string

+

+ + + regexp/compile(pattern) + + +

+ + +

Returns a regexp match object for the given pattern

+

+ + + regexp/compiled_pattern.find_all(s, n) + + +

+ + +

Returns a table list of all matches for the pattern, (up to n matches, unless n == -1 in which case all possible matches will be returned)

+

+ + + regexp/compiled_pattern.find_all_submatch(s, n) + + +

+ + +

Returns a table list of all sub-matches for the pattern, (up to n matches, unless n == -1 in which case all possible matches will be returned). +Submatches are matches of parenthesized subexpressions (also known as capturing groups) within the regular expression, +numbered from left to right in order of opening parenthesis. +Submatch 0 is the match of the entire expression, submatch 1 is the match of the first parenthesized subexpression, and so on

+

+ + + regexp/compiled_pattern.find(s) + + +

+ + +

Returns a string representing the left-most match for the given pattern in string s

+

+ + + regexp/compiled_pattern.find_submatch(s) + + +

+ + +

find_submatch returns a table of strings holding the text of the leftmost match of the regular expression in s and the matches, if any, of its submatches

+

+ + + strings/split(s, sep) + + +

+ + +

returns a table of strings, the result of splitting s with sep.

+

+ + + strings/trim(s) + + +

+ + +

Returns a string with all leading and trailing white space removed, as defined by Unicode

+

+ + + strings/replace(s, old, new, n) + + +

+ + +

Returns a copy of the string s with the first n non-overlapping instances of old replaced by new. +If old is empty, it matches at the beginning of the string and after each UTF-8 sequence, yielding up to k+1 replacements for a k-rune string.

+ +

If n < 0, there is no limit on the number of replacements

+

+ + + strings/has_prefix(s, prefix) + + +

+ + +

Returns true if s begins with prefix

+

+ + + strings/has_suffix(s, suffix) + + +

+ + +

Returns true if s ends with suffix

+

+ + + strings/contains(s, substr) + + +

+ + +

Returns true if substr is contained anywhere in s

+

+ + + time/now() + + +

+ + +

Returns a float64 representing the amount of nanoseconds since the unix epoch (01/01/1970 00:00:00).

+

+ + + time/format(epoch_nano, layout, zone) + + +

+ + +

Returns a string representation of the given epoch_nano timestamp for the given Timezone (e.g. "UTC", "America/Los_Angeles", …) +The layout parameter should follow Go’s time layout format.

+

+ + + time/format_iso(epoch_nano, zone) + + +

+ + +

Returns a string representation of the given epoch_nano timestamp for the given Timezone (e.g. "UTC", "America/Los_Angeles", …) +The returned string will be in ISO8601 format.

+

+ + + time/sleep(duration_ns) + + +

+ + +

Sleep for duration_ns nanoseconds

+

+ + + time/since(epoch_nano) + + +

+ + +

Returns the amount of nanoseconds elapsed since epoch_nano

+

+ + + time/add(epoch_time, duration_table) + + +

+ + +

Returns a new timestamp (in nanoseconds passed since 01/01/1970 00:00:00) for the given duration. +The duration should be a table with the following structure:

+ +
> require("time")
+> time.add(time.now(), {
+    ["hour"] = 1,
+    ["minute"] = 20,
+    ["second"] = 50
+})
+
+

You may omit any of the fields from the table, resulting in a default value of 0 for omitted fields

+

+ + + time/parse(layout, value) + + +

+ + +

Returns a float64 representing the amount of nanoseconds since the unix epoch (01/01/1970 00:00:00). +This timestamp will represent date value parsed using the layout format.

+ +

The layout parameter should follow Go’s time layout format

+

+ + + time/parse_iso(value) + + +

+ + +

Returns a float64 representing the amount of nanoseconds since the unix epoch (01/01/1970 00:00:00 for value. +The value string should be in ISO8601 format

+

+ + + uuid/new() + + +

+ + +

Returns a new 128-bit RFC 4122 UUID in string representation.

+

+ + + net/url + + +

+ + +

Provides a parse function parse a URL string into parts, returns a table with the URL’s host, path, scheme, query and fragment.

+ +
> local url = require("net/url")
+> url.parse("https://example.com/path?p1=a#section")
+{
+    ["host"] = "example.com"
+    ["path"] = "/path"
+    ["scheme"] = "https"
+    ["query"] = "p1=a"
+    ["fragment"] = "section"
+}
+
+

+ + + net/http (optional) + + +

+ + +

Provides a request function that performs an HTTP request. +For security reasons, this package is not available by default as it enables http requests to be sent out from the lakeFS instance network. The feature should be enabled under actions.lua.net_http_enabled configuration. +Request will time out after 30 seconds.

+ +
http.request(url [, body])
+http.request{
+  url = string,
+  [method = string,]
+  [headers = header-table,]
+  [body = string,]
+}
+
+ +

Returns a code (number), body (string), headers (table) and status (string).

+ +
    +
  • code - status code number
  • +
  • body - string with the response body
  • +
  • headers - table with the response request headers (key/value or table of values)
  • +
  • status - status code text
  • +
+ +

The first form of the call will perform GET requests or POST requests if the body parameter is passed.

+ +

The second form accepts a table and allows you to customize the request method and headers.

+ +

Example of a GET request

+ +
local http = require("net/http")
+local code, body = http.request("https://example.com")
+if code == 200 then
+    print(body)
+else
+    print("Failed to get example.com - status code: " .. code)
+end
+
+
+ +

Example of a POST request

+ +
local http = require("net/http")
+local code, body = http.request{
+    url="https://httpbin.org/post",
+    method="POST",
+    body="custname=tester",
+    headers={["Content-Type"]="application/x-www-form-urlencoded"},
+}
+if code == 200 then
+    print(body)
+else
+    print("Failed to post data - status code: " .. code)
+end
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/hooks/webhooks.html b/v1.46/howto/hooks/webhooks.html new file mode 100644 index 000000000..cbc88a231 --- /dev/null +++ b/v1.46/howto/hooks/webhooks.html @@ -0,0 +1,935 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Webhooks | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Webhooks + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Action File Webhook properties
  2. +
  3. Request body schema
  4. +
+ +
+ +

A Webhook is a Hook type that sends an HTTP POST request to the configured URL. +Any non 2XX response by the responding endpoint will fail the Hook, cancel the execution of the following Hooks +under the same Action. For pre-* hooks, the triggering operation will also be aborted.

+ +

Warning: You should not use pre-* webhooks for long-running tasks, since they block the performed operation. +Moreover, the branch is locked during the execution of pre-* hooks, so the webhook server cannot perform any write operations on the branch (like uploading or commits).

+

+ + + Action File Webhook properties + + +

+ + +

See the Action configuration for overall configuration schema and details.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PropertyDescriptionData TypeRequiredDefault ValueEnv Vars Support
urlThe URL address of the requestStringtrue no
timeoutTime to wait for response before failing the hookString (golang’s Duration representation)false1 minuteno
query_paramsList of query params that will be added to the requestDictionary(String:String or String:List(String)false yes
headersHeaders to add to the requestDictionary(String:String)false yes
+ +

Secrets & Environment Variables
+lakeFS Actions supports secrets by using environment variables. +The format {{ ENV.SOME_ENV_VAR }} will be replaced with the value of $SOME_ENV_VAR +during the execution of the action. If that environment variable doesn’t exist in the lakeFS server environment, the action run will fail.

+ +

All environment variables need to begin with “LAKEFSACTIONS_”. Otherwise, they will be blocked. +Additionally, the actions.env.enabled configuration parameter can be set to false to block access to all environment variables.

+ +

Example:

+ +
...
+hooks:
+  - id: prevent_user_columns
+    type: webhook
+    description: Ensure no user_* columns under public/
+    properties:
+      url: "http://<host:port>/webhooks/schema"
+      timeout: 1m30s
+      query_params:
+        disallow: ["user_", "private_"]
+        prefix: public/
+      headers:
+        secret_header: "{{ ENV.MY_SECRET }}"
+...
+
+

+ + + Request body schema + + +

+ +

Upon execution, a webhook will send a request containing a JSON object with the following fields:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
FieldDescriptionType
event_typeType of the event that triggered the Actionstring
event_timeTime of the event that triggered the Action (RFC3339 formatted)string
action_nameContaining Hook Action’s Namestring
hook_idID of the Hookstring
repository_idID of the Repositorystring
branch_id1ID of the Branchstring
source_refReference to the source on which the event was triggeredstring
commit_message2The message for the commit (or merge) that is taking placestring
committer2Name of the committerstring
commit_metadata2The metadata for the commit that is taking placestring
commit_id[^2,^4]The ID of the commit that is being createdstring
tag_id3The ID of the created/deleted tagstring
+ +

Example:

+
{
+  "event_type": "pre-merge",
+  "event_time": "2021-02-28T14:03:31Z",
+  "action_name": "test action",
+  "hook_id": "prevent_user_columns",
+  "repository_id": "repo1",
+  "branch_id": "feature-1",
+  "source_ref": "feature-1",
+  "commit_message": "merge commit message",
+  "commit_id": "5891b5b522d5df086d0ff0b110fbd9d21bb4fc7163af34d08286a2e846f6be03",
+  "committer": "committer",
+  "commit_metadata": {
+    "key": "value"
+  }
+}
+
+
+
    +
  1. +

    N\A for Tag events 

    +
  2. +
  3. +

    N\A for Tag and Create/Delete Branch events  2 3

    +
  4. +
  5. +

    Applicable only for Tag events 

    +
  6. +
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/import.html b/v1.46/howto/import.html new file mode 100644 index 000000000..0390186fc --- /dev/null +++ b/v1.46/howto/import.html @@ -0,0 +1,923 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Import Data | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

This section describes how to import existing data into a lakeFS repository, without copying it. +If you are interested in copying data into lakeFS, see Copying data to/from lakeFS.

+

+ + + Importing data into lakeFS + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Prerequisites
  2. +
  3. Using the lakeFS UI
  4. +
  5. Using the CLI: lakectl import
  6. +
  7. Examples
  8. +
+ +
+ + +

+ + + Prerequisites + + +

+ + +
    +
  • Importing is permitted for users in the Supers (open-source) group or the SuperUsers (Cloud/Enterprise) group. + To learn how lakeFS Cloud and lakeFS Enterprise users can fine-tune import permissions, see Fine-grained permissions below.
  • +
  • The lakeFS server must have permissions to list the objects in the source bucket.
  • +
  • The source bucket must be on the same cloud provider and in the same region as your repository.
  • +
+

+ + + Using the lakeFS UI + + +

+ + +
    +
  1. In your repository’s main page, click the Import button to open the import dialog.
  2. +
  3. Under Import from, fill in the location on your object store you would like to import from.
  4. +
  5. Fill in the import destination in lakeFS. This should be a path under the current branch.
  6. +
  7. Add a commit message, and optionally commit metadata.
  8. +
  9. Press Import.
  10. +
+ +

Once the import is complete, a new commit containing the imported objects will be created in the destination branch.

+ +

lakeFS UI import dialog

+

+ + + Using the CLI: lakectl import + + +

+ +

The lakectl import command acts the same as the UI import wizard. It commits the changes to the selected branch.

+ +
+ +
+
lakectl import \
+  --from s3://bucket/optional/prefix/ \
+  --to lakefs://my-repo/my-branch/optional/path/
+
+
+
+
lakectl import \
+   --from https://storageAccountName.blob.core.windows.net/container/optional/prefix/ \
+   --to lakefs://my-repo/my-branch/optional/path/
+
+
+
+
lakectl import \
+   --from gs://bucket/optional/prefix/ \
+   --to lakefs://my-repo/my-branch/optional/path/
+
+
+
+

+ + + Notes + + +

+ + +
    +
  1. Any previously existing objects under the destination prefix will be deleted.
  2. +
  3. The import duration depends on the amount of imported objects, but will roughly be a few thousand objects per second.
  4. +
  5. For security reasons, if you are using lakeFS on top of your local disk (blockstore.type=local), you need to enable the import feature explicitly. +To do so, set the blockstore.local.import_enabled to true and specify the allowed import paths in blockstore.local.allowed_external_prefixes (see configuration reference). +When using lakectl or the lakeFS UI, you can currently import only directories locally. If you need to import a single file, use the HTTP API or API Clients with type=object in the request body and destination=<full-path-to-file>.
  6. +
  7. Making changes to data in the original bucket will not be reflected in lakeFS, and may cause inconsistencies.
  8. +
+

+ + + Examples + + +

+ +

To explore practical examples and real-world use cases of importing data into lakeFS, +we recommend checking out our comprehensive blog post on the subject.

+

+ + + Fine-grained permissions + + +

+ +

lakeFS Cloud

+

lakeFS Enterprise

+ +

With RBAC support, The lakeFS user running the import command should have the following permissions in lakeFS: +fs:WriteObject, fs:CreateMetaRange, fs:CreateCommit, fs:ImportFromStorage and fs:ImportCancel.

+ +

As mentioned above, all of these permissions are available by default to the Supers (open-source) group or the SuperUsers (Cloud/Enterprise).

+

+ + + Provider-specific permissions + + +

+ + +

In addition, the following for provider-specific permissions may be required:

+ +
+ +
+

+ + + AWS S3: Importing from public buckets + + +

+ + +

lakeFS needs access to the imported location to first list the files to import and later read the files upon users request.

+ +

There are some use cases where the user would like to import from a destination which isn’t owned by the account running lakeFS. +For example, importing public datasets to experiment with lakeFS and Spark.

+ +

lakeFS will require additional permissions to read from public buckets. For example, for S3 public buckets, +the following policy needs to be attached to the lakeFS S3 service-account to allow access to public buckets, while blocking access to other owned buckets:

+ +
   {
+     "Version": "2012-10-17",
+     "Statement": [
+       {
+         "Sid": "PubliclyAccessibleBuckets",
+         "Effect": "Allow",
+         "Action": [
+            "s3:GetBucketVersioning",
+            "s3:ListBucket",
+            "s3:GetBucketLocation",
+            "s3:ListBucketMultipartUploads",
+            "s3:ListBucketVersions",
+            "s3:GetObject",
+            "s3:GetObjectVersion",
+            "s3:AbortMultipartUpload",
+            "s3:ListMultipartUploadParts"
+         ],
+         "Resource": ["*"],
+         "Condition": {
+           "StringNotEquals": {
+             "s3:ResourceAccount": "<YourAccountID>"
+           }
+         }
+       }
+     ]
+   }
+
+ +
+
+ +

Note: The use of the adls hint for ADLS Gen2 storage accounts is deprecated, please use the original source url for import.

+ +

See Azure deployment on limitations when using account credentials.

+ +
+
+

No specific prerequisites

+
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/index.html b/v1.46/howto/index.html new file mode 100644 index 000000000..01bac5778 --- /dev/null +++ b/v1.46/howto/index.html @@ -0,0 +1,833 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +How-To | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS - How To + + +

+ + +

lakeFS Docs

+

+ + + Installation and upgrades + + +

+ + + +

+ + + Getting data in and out of lakeFS + + +

+ + + +

+ + + Actions and Hooks in lakeFS + + +

+ + +
    +
  • Use Actions and Hooks as part of your workflow to validate data, enforce constraints, and do more when events occur.
  • +
+

+ + + Branch Protection + + +

+ + +
    +
  • Branch Protection prevents commits directly to a branch. This is a good way to enforce good practice and make sure that changes to important branches are only done by a merge.
  • +
+

+ + + Pull Requests + + +

+ + + +

+ + + lakeFS Sizing Guide + + +

+ + + +

+ + + Garbage Collection + + +

+ + +
    +
  • lakeFS will keep all of your objects forever, unless you tell it otherwise. Use Garbage Collection (GC) to remove objects from the underlying storage. + If you want GC to happen automagically then you can use Managed Garbage Collection which is available as part of lakeFS Cloud.
  • +
+ + + +
    +
  • Private Link enables lakeFS Cloud to interact with your infrastructure using private networking.
  • +
+

+ + + Unity Delta Sharing + + +

+ + +
    +
  • lakeFS Unity Delta Sharing provides a read-only experience from Unity Catalog for lakeFS customers.
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/local-checkouts.html b/v1.46/howto/local-checkouts.html new file mode 100644 index 000000000..d2eae02ad --- /dev/null +++ b/v1.46/howto/local-checkouts.html @@ -0,0 +1,1276 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Work with lakeFS data locally | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Working with lakeFS Data Locally + + +

+ + +

lakeFS is a scalable data version control system designed to scale to billions of objects. The larger the data, the less +feasible it becomes to consume it from a single machine. lakeFS addresses this challenge by enabling efficient management +of large-scale data stored remotely.

+ +

In addition to its capability to manage large datasets, lakeFS offers the flexibility +to work with versioned data by exposing it as a local filesystem directory.

+ +

This page explains lakeFS Mount and lakectl local: two common ways of exposing lakeFS data locally, with different performance characteristics.

+ + +

+ + + Use cases + + +

+ +

+ + + Local development of ML models + + +

+ + +

The development of machine learning models is a dynamic and iterative process, including experimentation with various data versions, +transformations, algorithms, and hyperparameters. To optimize this iterative workflow, experiments must be conducted with speed, +ease of tracking, and reproducibility in mind. Localizing the model data during development enhances the development process. +It accelerates the development process by enabling interactive and offline development and reducing data access latency.

+ +

The local availability of data is required to seamlessly integrate data version control systems and source control systems +like Git. This integration is vital for achieving model reproducibility, allowing for a more efficient and collaborative +model development environment.

+

+ + + Data Locality for Optimized GPU Utilization + + +

+ + +

Training Deep Learning models requires expensive GPUs. In the context of running such programs, the goal is to optimize +GPU usage and prevent them from sitting idle. Many deep learning tasks involve accessing images, and in some cases, the +same images are accessed multiple times. Localizing the data can eliminate redundant round trip times to access remote +storage, resulting in cost savings.

+ + +

+ + + lakeFS Mount: Efficiently expose lakeFS Data as a local directory + + +

+ + +

⚠️ lakeFS Mount is currently in preview. There is no installation required, please contact us to get access.

+

+ + + Prerequisites: + + +

+ + +
    +
  • A working lakeFS Server running either lakeFS Enterprise or lakeFS Cloud
  • +
  • You’ve installed the lakectl command line utility: this is the official lakeFS command line interface, on top of which lakeFS Mount is built.
  • +
  • lakectl is configured properly to access your lakeFS server as detailed in the configuration instructions
  • +
+

+ + + Mounting a lakeFS reference as a local directory + + +

+ + +

lakeFS Mount works by exposing a virtual mountpoint on the host computer.

+ +

This “acts” as a local directory, allowing applications to read write and interact with data as it is all local to the machine, while lakeFS Mount optimizes this behind the scenes by lazily fetching data as requested, caching accessed objects and efficiently managing metadata to ensure best in class performance. Read more about how lakeFS Mount optimizes performance

+ +

Mounting a reference is a single command:

+ +
everest mount lakefs://example-repo/example-branch/path/to/data/ ./my_local_dir
+
+ +

Once executed, the my_local_dir directory should appear to have the contents of the remote path we provided. We can verify this:

+ +
ls -l ./my_local_dir/
+
+ +

Which should return the listing of the mounted path.

+ +

lakeFS Mount allows quite a bit of tuning to ensure optimal performance. Read more about how lakeFS Mount works and how to configure it.

+

+ + + Reading from a mount + + +

+ + +

Reading from a lakeFS Mount requires no special tools, integrations or SDKs! Simply point your code to the directory and read from it as if it was in fact local:

+ +
#!/usr/bin/env python
+import glob
+
+for image_path in glob.glob('./my_local_dir/*.png'):
+    with open(image_path, 'rb') as f:
+        process(f)
+
+
+

+ + + Unmounting + + +

+ + +

When done, simply run:

+ +
everest umount ./my_local_dir
+
+ +

This will unmount the lakeFS Mount, cleaning up background tasks

+

+ + + lakectl local: Sync lakeFS data with a local directory + + +

+ + +

The local command of lakeFS’ CLI lakectl enables working with lakeFS data locally by copying the data onto the host machine. +It allows syncing local directories with remote lakeFS locations, +and to seamlessly integrate lakeFS with Git.

+ +

Here are the available lakectl local commands:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
CommandWhat it doesNotes 
initConnects between a local directory and a lakeFS remote URI to enable data syncTo undo a directory init, delete the .lakefs_ref.yaml file created in the initialized directory 
cloneClones lakeFS data from a path into an empty local directory and initializes the directoryA directory can only track a single lakeFS remote location. i.e., you cannot clone data into an already initialized directory 
listLists directories that are synced with lakeFSIt is recommended to follow any init or clone command with a list command to verify its success 
statusShows remote and local changes to the directory and the remote location it tracks  
commitCommits changes from local directory to the lakeFS branch it tracksUncommitted changes to directories connected to lakeFS remote locations will not reflect in lakeFS until after doing lakectl local commit. 
pullFetches latest changes from a lakeFS remote location into a connected local directory  
checkoutSyncs a local directory with the state of a lakeFS ref  
+ +

Note: The data size you work with locally should be reasonable for smooth operation on a local machine which is typically no larger than 15 GB.

+

+ + + Example: Using lakectl local in tandem with Git + + +

+ + +

We are going to develop an ML model that predicts whether an image is an Alpaca or not. Our goal is to improve the input +for the model. The code for the model is versioned by Git while the model dataset is versioned by lakeFS. We will +be using lakectl local to tie code versions to data versions to achieve model reproducibility.

+

+ + + Setup + + +

+ + +

To get start with, we have initialized a Git repo called is_alpaca that includes the model code: +Git repo

+ +

We also created a lakeFS repository and uploaded the is_alpaca train dataset +by Kaggel into it: +lakeFS repo

+

+ + + Create an Isolated Environment for Experiments + + +

+ + +

Our goal is to improve the model predictions. To meet our goal, we will experiment with editing the training dataset. +We will run our experiments in isolation to not change anything until after we are certain the data is improved and ready.

+ +

Let’s create a new lakeFS branch called experiment-1. Our is_alpaca dataset is accessible on that branch, +and we will interact with the data from that branch only. +experiment-1-branch

+ +

On the code side, we will create a Git branch also called experiment-1 to not pollute our main branch with a dataset +which is under tuning.

+

+ + + Clone lakeFS Data into a Local Git Repository + + +

+ + +

Inspecting the train.py script, we can see that it expects an input on the input directory.

+
#!/usr/bin/env python
+import tensorflow as tf
+
+input_location = './input'
+model_location = './models/is_alpaca.h5'
+
+def get_ds(subset):
+    return tf.keras.utils.image_dataset_from_directory(
+        input_location, validation_split=0.2, subset=subset,
+        seed=123, image_size=(244, 244), batch_size=32)
+
+train_ds = get_ds("training")
+val_ds = get_ds("validation")
+
+model = tf.keras.Sequential([
+    tf.keras.layers.Rescaling(1./255),
+    tf.keras.layers.Conv2D(32, 3, activation='relu'),
+    tf.keras.layers.MaxPooling2D(),
+    tf.keras.layers.Conv2D(32, 3, activation='relu'),
+    tf.keras.layers.MaxPooling2D(),
+    tf.keras.layers.Conv2D(32, 3, activation='relu'),
+    tf.keras.layers.MaxPooling2D(),
+    tf.keras.layers.Flatten(),
+    tf.keras.layers.Dense(128, activation='relu'),
+    tf.keras.layers.Dense(2)])
+
+# Fit and save
+loss_fn = tf.keras.losses.SparseCategoricalCrossentropy(from_logits=True)
+model.compile(optimizer='adam', loss=loss_fn, metrics=['accuracy'])
+model.fit(train_ds, validation_data=val_ds, epochs=3)
+model.save(model_location)
+
+ +

This means that to be able to locally develop our model and experiment with it we need to have the is_alpaca dataset managed +by lakeFS available locally on that path. To do that, we will use the lakectl local clone +command from our local Git repository root:

+
lakectl local clone lakefs://is-alpaca/experiment-1/dataset/train/ input
+
+

This command will do a diff between out local input directory (that did not exist until now) and the provided lakeFS path +and identify that there are files to be downloaded from lakeFS.

+
Successfully cloned lakefs://is-alpaca/experiment-1/dataset/train/ to ~/ml_models/is_alpaca/input
+
+Clone Summary:
+
+Downloaded: 250
+Uploaded: 0
+Removed: 0
+
+ +

Running lakectl local list from our Git repository root will show that the +input directory is now in sync with a lakeFS prefix (Remote URI), and what lakeFS version of the data (Synced Commit) +the is it tracking:

+
 is_alpaca % lakectl local list                 
++-----------+------------------------------------------------+------------------------------------------------------------------+
+| DIRECTORY | REMOTE URI                                     | SYNCED COMMIT                                                    |
++-----------+------------------------------------------------+------------------------------------------------------------------+
+| input     | lakefs://is-alpaca/experiment-1/dataset/train/ | 589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3 |
++-----------+------------------------------------------------+------------------------------------------------------------------+
+
+

+ + + Tie Code Version and Data Version + + +

+ + +

Now let’s tell Git to stage the dataset we’ve added and inspect our Git branch status:

+
is_alpaca % git add input/
+is_alpaca % git status 
+On branch experiment-1
+Changes to be committed:
+  (use "git restore --staged <file>..." to unstage)
+	new file:   input/.lakefs_ref.yaml
+
+Changes not staged for commit:
+  (use "git add <file>..." to update what will be committed)
+  (use "git restore <file>..." to discard changes in working directory)
+	modified:   .gitignore
+
+ +

We can see that the .gitignore file changed, and that the files we cloned from lakeFS into the input directory are not +tracked by git. This is intentional - remember that lakeFS is the one managing the data. But wait, what is this special +input/.lakefs_ref.yaml file that Git does track?

+
is_alpaca % cat input/.lakefs_ref.yaml
+
+src: lakefs://is-alpaca/experiment-1/dataset/train/s
+at_head: 589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3
+
+

This file includes the lakeFS version of the data that the Git repository is currently pointing to.

+ +

Let’s commit the changes to Git with:

+
git commit -m "added is_alpaca dataset" 
+
+

By committing to Git, we tie the current code version of the model to the dataset version in lakeFS as it appears in +input/.lakefs_ref.yaml.

+

+ + + Experiment and Version Results + + +

+ + +

We ran the train script on the cloned input, and it generated a model. Now, let’s use the model to predict whether an axolotl is an alpaca.

+ +

A reminder - this is how an axolotl looks like - not like an alpaca!

+ +

axolotl

+ +

Here are the (surprising) results:

+
is_alpaca % ./predict.py ~/axolotl1.jpeg
+{'alpaca': 0.32112, 'not alpaca': 0.07260383}
+
+

We expected the model to provide a more concise prediction, so let’s try to improve it. To do that, we will add additional +images of axolotls to the model input directory:

+
is_alpaca % cp ~/axolotls_images/* input/not_alpaca
+
+ +

To inspect what changes we made to out dataset we will use lakectl local status.

+
is_alpaca % lakectl local status input 
+diff 'local:///ml_models/is_alpaca/input' <--> 'lakefs://is-alpaca/589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3/dataset/train/'...
+diff 'lakefs://is-alpaca/589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3/dataset/train/' <--> 'lakefs://is-alpaca/experiment-1/dataset/train/'...
+
+╔════════╦════════╦════════════════════════════╗
+║ SOURCE ║ CHANGE ║ PATH                       ║
+╠════════╬════════╬════════════════════════════╣
+║ local  ║ added  ║ not_alpaca/axolotl2.jpeg ║
+║ local  ║ added  ║ not_alpaca/axolotl3.png  ║
+║ local  ║ added  ║ not_alpaca/axolotl4.jpeg ║
+╚════════╩════════╩════════════════════════════╝
+
+ +

At this point, the dataset changes are not yet tracked by lakeFS. We will validate that by looking at the uncommitted changes +area of our experiment branch and verifying it is empty.

+ +

To commit these changes to lakeFS we will use lakectl local commit:

+
is_alpaca % lakectl local commit input -m "add images of axolotls to the training dataset"
+
+Getting branch: experiment-1
+
+diff 'local:///ml_models/is_alpaca/input' <--> 'lakefs://is-alpaca/589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3/dataset/train/'...
+upload not_alpaca/axolotl3.png              ... done! [5.04KB in 679ms]
+upload not_alpaca/axolotl2.jpeg             ... done! [38.31KB in 685ms]
+upload not_alpaca/axolotl4.jpeg             ... done! [7.70KB in 718ms]
+
+Sync Summary:
+
+Downloaded: 0
+Uploaded: 3
+Removed: 0
+
+Finished syncing changes. Perform commit on branch...
+Commit for branch "experiment-1" completed.
+
+ID: 0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6
+Message: add images of axolotls to the training dataset
+Timestamp: 2024-02-08 17:41:20 +0200 IST
+Parents: 589f87704418c6bac80c5a6fc1b52c245af347b9ad1ea8d06597e4437fae4ca3
+
+ +

Looking at the lakeFS UI we can see that the lakeFS commit includes metadata that tells us what was the code version of +the linked Git repository at the time of the commit. +git metadata in lakeFS

+ +

Inspecting the Git repository, we can see that the input/.lakefs_ref.yaml is pointing to the latest lakeFS commit 0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6.

+ +

We will now re-train our model with the modified dataset and give a try to predict whether an axolotl is an alpaca:

+
is_alpaca % ./predict.py ~/axolotl1.jpeg
+{'alpaca': 0.12443, 'not alpaca': 0.47260383}
+
+

Results are indeed more accurate.

+

+ + + Sync a Local Directory with lakeFS + + +

+ + +

Now that we think that the latest version of our model generates reliable predictions, let’s validate it against a test dataset +rather than against a single picture. We will use the test dataset provided by Kaggel. +Let’s create a local testDataset directory in our git repository and populate it with the test dataset.

+ +

Now, we will use lakectl local init to sync the testDataset directory with our lakeFS repository:

+
is_alpaca % lakectl local init lakefs://is-alpaca/main/dataset/test/ testDataset 
+Location added to /is_alpaca/.gitignore
+Successfully linked local directory '/is_alpaca/testDataset' with remote 'lakefs://is-alpaca/main/dataset/test/'
+
+ +

And validate that the directory was linked successfully:

+
is_alpaca % lakectl local list                                                           
++-------------+-------------------------------------------------+------------------------------------------------------------------+
+| DIRECTORY   | REMOTE URI                                      | SYNCED COMMIT                                                    |
++-------------+-------------------------------------------------+------------------------------------------------------------------+
+| input       | lakefs://is-alpaca/main/dataset/train/          | 0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6 |
+| testDataset | lakefs://is-alpaca/main/dataset/test/           | 0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6 |
++-------------+-------------------------------------------------+------------------------------------------------------------------+
+
+ +

Now we will tell Git to track the testDataset directory with git add testDataset, and as we saw earlier Git will only track the testDataset/.lakefs_ref.yaml +for that directory rather than its content.

+ +

To see the difference between our local testDataset directory and its lakeFS location lakefs://is-alpaca/main/dataset/test/ +we will use lakectl local status:

+
is_alpaca % lakectl local status testDataset 
+
+diff 'local:///ml_models/is_alpaca/testDataset' <--> 'lakefs://is-alpaca/0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6/dataset/test/'...
+diff 'lakefs://is-alpaca/0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6/dataset/test/' <--> 'lakefs://is-alpaca/main/dataset/test/'...
+
+╔════════╦════════╦════════════════════════════════╗
+║ SOURCE ║ CHANGE ║ PATH                           ║
+╠════════╬════════╬════════════════════════════════╣
+║ local  ║ added  ║ alpaca/alpaca (1).jpg          ║
+║ local  ║ added  ║ alpaca/alpaca (10).jpg         ║
+    .        .                  .
+    .        .                  .
+    .        .                  .
+║ local  ║ added  ║ not_alpaca/not_alpaca (9).jpg  ║
+╚════════╩════════╩════════════════════════════════╝
+
+ +

We can see that multiple files were locally added to the synced directory.

+ +

To apply these changes to lakeFS we will commit them:

+
is_alpaca % lakectl local commit testDataset -m "add is_alpaca test dataset to lakeFS" 
+
+Getting branch: experiment-1
+
+diff 'local:///ml_models/is_alpaca/testDataset' <--> 'lakefs://is-alpaca/0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6/dataset/test/'...
+upload alpaca/alpaca (23).jpg            ... done! [113.81KB in 1.241s]
+upload alpaca/alpaca (26).jpg            ... done! [102.74KB in 1.4s]
+          .                                             .
+          .                                             .
+upload not_alpaca/not_alpaca (42).jpg    ... done! [886.93KB in 14.336s]
+
+Sync Summary:
+
+Downloaded: 0
+Uploaded: 77
+Removed: 0
+
+Finished syncing changes. Perform commit on branch...
+Commit for branch "experiment-1" completed.
+
+ID: c8be7f4f5c13dd2e489ae85e6f747230bfde8e50f9cd9b6af20b2baebfb576cf
+Message: add is_alpaca test dataset to lakeFS
+Timestamp: 2024-02-10 12:31:53 +0200 IST
+Parents: 0b376f01b925a075851bbaffacf104a80de04a43ed7e56054bf54c42d2c8cce6
+
+ +

Looking at the lakFS UI we see that our test data is now available at lakeFS: +test_dataset

+ +

Finally, we will Git commit the local changes to link between the Git and lakeFS repositories state.

+ +

Note: While syncing a local directory with a lakeF prefix, it is recommended to first commit the data to lakeFS and then +do a Git commit that will include the changes done to the .lakefs_ref.yaml for the synced directory. Reasoning is that +only after committing the data to lakeFS, the .lakefs_ref.yaml file points to a lakeFS commit that includes the added +content from the directory.

+

+ + + Reproduce Model Results + + +

+ + +

What if we wanted to re-run the model that predicted that an axolotl is more likely to be an alpaca? +This question translates into the question: “How do I roll back my code and data to the time before we optimized the train dataset?” +Which translates to: “What was the Git commit ID at this point?”

+ +

Searching our Git log we find this commit:

+
commit 5403ec29903942b692aabef404598b8dd3577f8a
+
+    added is_alpaca dataset
+
+ +

So, all we have to do now is git checkout 5403ec29903942b692aabef404598b8dd3577f8a and we are good to reproduce the model results!

+ +

Checkout our article about ML Data Version Control and Reproducibility at Scale to get another example for how lakeFS and Git work seamlessly together.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/migrate-away.html b/v1.46/howto/migrate-away.html new file mode 100644 index 000000000..27f6776bd --- /dev/null +++ b/v1.46/howto/migrate-away.html @@ -0,0 +1,727 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Migrating away from lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Migrating away from lakeFS + + +

+ +

+ + + Copying data from a lakeFS repository to an S3 bucket + + +

+ + +

The simplest way to migrate away from lakeFS is by copying data from a lakeFS repository to an S3 bucket +(or any other object store).

+ +

For smaller repositories, you can do this by using the AWS CLI or Rclone. +For larger repositories, running distcp with lakeFS as the source is also an option.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/mirroring.html b/v1.46/howto/mirroring.html new file mode 100644 index 000000000..5555c4b66 --- /dev/null +++ b/v1.46/howto/mirroring.html @@ -0,0 +1,1136 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Mirroring | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Mirroring + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

Mirroring is only available for lakeFS Cloud.

+
+ + +

+ + + What is lakeFS mirroring? + + +

+ + +

Mirroring in lakeFS allows replicating a lakeFS repository (“source”) into read-only copies (“mirror”) in different locations.

+ +

Unlike conventional mirroring, data isn’t simply copied between regions - lakeFS Cloud tracks the state of each commit, advancing the commit log on the mirror only once a commit has been fully replicated and all data is available.

+ +

mirroring architecture

+ + +

+ + + Uses cases + + +

+ +

+ + + Disaster recovery + + +

+ + +

Typically, object stores provide a replication/batch copy API to allow for disaster recovery: as new objects are written, they are asynchronously copied to other geo locations.

+ +

In the case of regional failure, users can rely on the other geolocations which should contain relatively-up-to-date state.

+ +

The problem is reasoning about what managed to arrive by the time of disaster and what hasn’t:

+
    +
  • have all the necessary files for a given dataset arrived?
  • +
  • in cases there are dependencies between datasets, are all dependencies also up to date?
  • +
  • what is currently in-flight or haven’t even started replicating yet?
  • +
+ +

Reasoning about these is non-trivial, especially in the face of a regional disaster, however ensuring business continuity might require that we have these answers.

+ +

Using lakeFS mirroring makes it much easier to answer: we are guaranteed that the latest commit that exists in the replica is in a consistent state and is fully usable, even if it isn’t the absolute latest commit - it still reflects a known, consistent, point in time.

+

+ + + Data Locality + + +

+ + +

For certain workloads, it might be cheaper to have data available in multiple regions: Expensive hardware such as GPUs might fluctuate in price, so we’d want to pick the region that currently offers the best pricing. The difference could easily offset to cost of the replicated data.

+ +

The challenge is reproducibility - Say we have an ML training job that reads image files from a path in the object store. Which files existed at the time of training?

+ +

If data is constantly flowing between regions, this might be harder to answer than we think. And even if we know - how can we recreate that exact state if we want to run the process again (for example, to rebuild that model for troubleshooting).

+ +

Using consistent commits solves this problem - with lakeFS mirroring, it is guaranteed that a commit ID, regardless of location, will always contain the exact same data.

+ +

We can train our model in region A, and a month later feed the same commit ID into another region - and get back the same results.

+

+ + + Setting up mirroring + + +

+ +

+ + + Configuring bucket replication on S3 + + +

+ + +

The objects within the repository are copied using your cloud provider’s object store replication mechanism. +For AWS S3, please refer to the AWS S3 replication documentation to make sure your lakeFS repository’s storage namespace (source) is replicated to the region you’d like your mirror to be located on (target).

+ +

After setting the replication rule, new objects will be replicated to the destination bucket.

+ +

In order to replicate the existing objects, we’d need to manually copy them - however, we can use S3 batch jobs to do this.

+

+ + + Creating a lakeFS user with a “replicator” policy + + +

+ + +

On our source lakeFS installation, under Administration create a new user that will be used by the replication subsystem. +The user should have the following RBAC policy attached:

+ +
{
+   "id": "ReplicationPolicy",
+   "statement": [
+      {
+         "action": [
+            "fs:ReadRepository",
+            "fs:CreateRepository",
+            "fs:UpdateRepository",
+            "fs:DeleteRepository",
+            "fs:ListRepositories",
+            "fs:AttachStorageNamespace",
+            "fs:ReadObject",
+            "fs:WriteObject",
+            "fs:DeleteObject",
+            "fs:ListObjects",
+            "fs:CreateCommit",
+            "fs:CreateMetaRange",
+            "fs:ReadCommit",
+            "fs:ListCommits",
+            "fs:CreateBranch",
+            "fs:DeleteBranch",
+            "fs:RevertBranch",
+            "fs:ReadBranch",
+            "fs:ListBranches"
+         ],
+         "effect": "allow",
+         "resource": "*"
+      }
+   ]
+}
+
+ +

Alternatively, we can create a policy with a narrower scope, only for a specific repository and/or mirror:

+ +
{
+   "id": "ReplicationPolicy",
+   "statement": [
+      {
+         "action": [
+            "fs:ListRepositories"
+         ],
+         "effect": "allow",
+         "resource": "*"
+      },
+      {
+         "action": [
+            "fs:ReadRepository",
+            "fs:ReadObject",
+            "fs:ListObjects",
+            "fs:ReadCommit",
+            "fs:ListCommits",
+            "fs:ReadBranch",
+            "fs:ListBranches"
+         ],
+         "effect": "allow",
+         "resource": "arn:lakefs:fs:::repository/{sourceRepositoryId}"
+      },
+      {
+         "action": [
+            "fs:ReadRepository",
+            "fs:CreateRepository",
+            "fs:UpdateRepository",
+            "fs:DeleteRepository",
+            "fs:AttachStorageNamespace",
+            "fs:ReadObject",
+            "fs:WriteObject",
+            "fs:DeleteObject",
+            "fs:ListObjects",
+            "fs:CreateCommit",
+            "fs:CreateMetaRange",
+            "fs:ReadCommit",
+            "fs:ListCommits",
+            "fs:CreateBranch",
+            "fs:DeleteBranch",
+            "fs:RevertBranch",
+            "fs:ReadBranch",
+            "fs:ListBranches"
+         ],
+         "effect": "allow",
+         "resource": "arn:lakefs:fs:::repository/{mirrorId}"
+      },
+      {
+         "action": [
+            "fs:AttachStorageNamespace"
+         ],
+         "effect": "allow",
+         "resource": "arn:lakefs:fs:::namespace/{DestinationStorageNamespace}"
+      }
+   ]
+}
+
+ +

Once a user has been created and the replication policy attached to it, create an access key and secret to be used by the mirroring process.

+

+ + + Authorizing the lakeFS Mirror process to use the replication user + + +

+ + +

Please contact Treeverse customer success to connect the newly created user with the mirroring process

+

+ + + Configuring repository replication + + +

+ + +

Replication has a stand-alone HTTP API. In this example, we’ll use cURL, but feel free to use any HTTP client or library:

+ +
curl --location 'https://<ORGANIZATION_ID>.<SOURCE_REGION>.lakefscloud.io/service/replication/v1/repositories/<SOURCE_REPO>/mirrors' \
+--header 'Content-Type: application/json' \
+-u <ACCESS_KEY_ID>:<SECRET_ACCESS_KEY> \
+-X POST \
+--data '{
+    "name": "<MIRROR_NAME>",
+    "region": "<MIRROR_REGION>",
+    "storage_namespace": "<MIRROR_STORAGE_NAMESPACE>"
+}'
+
+

Using the following parameters:

+ +
    +
  • ORGANIZATION_ID - The ID as it appears in the URL of your lakeFS installation (e.g. https://my-org.us-east-1.lakefscloud.io/)
  • +
  • SOURCE_REGION - The region where our source repository is hosted
  • +
  • SOURCE_REPO - Name of the repository acting as our replication source. It should exist
  • +
  • ACCESS_KEY_ID & SECRET_ACCESS_KEY - Credentials for your lakeFS user (make sure you have the necessary RBAC permissions as listed below)
  • +
  • MIRROR_NAME - Name used for the read-only mirror to be created on the destination region
  • +
  • MIRROR_STORAGE_NAMESPACE - Location acting as the replication target for the storage namespace of our source repository
  • +
+

+ + + Mirroring and Garbage Collection + + +

+ + +

Garbage collection won’t run on mirrored repositories. +Deletions from garbage collection should be replicated from the source:

+
    +
  1. Enable DELETED marker replication on the source bucket.
  2. +
  3. Create a lifecycle policy on the destination bucket to delete the objects with the DELETED marker.
  4. +
+

+ + + RBAC + + +

+ + +

These are the required RBAC permissions for working with the new cross-region replication feature:

+ +

Creating a Mirror:

+ +
+ + + + + + + + + + + + + + + + + + + + +
ActionARN
fs:CreateRepositoryarn:lakefs:fs:::repository/{mirrorId}
fs:MirrorRepositoryarn:lakefs:fs:::repository/{sourceRepositoryId}
fs:AttachStorageNamespacearn:lakefs:fs:::namespace/{storageNamespace}
+ +

Editing Mirrored Branches:

+ +
+ + + + + + + + + + + + +
ActionARN
fs:MirrorRepositoryarn:lakefs:fs:::repository/{sourceRepositoryId}
+ +

Deleting a Mirror:

+ +
+ + + + + + + + + + + + +
ActionARN
fs:DeleteRepositoryarn:lakefs:fs:::repository/{mirrorId}
+ +

Listing/Getting Mirrors for a Repository:

+ +
+ + + + + + + + + + + + +
ActionARN
fs:ListRepositories*
+

+ + + Other replication operations + + +

+ +

+ + + Listing all mirrors for a repository + + +

+ + +
curl --location 'https://<ORGANIZATION_ID>.<SOURCE_REGION>.lakefscloud.io/service/replication/v1/repositories/<SOURCE_REPO>/mirrors' \
+-u <ACCESS_KEY_ID>:<SECRET_ACCESS_KEY> -s
+
+

+ + + Getting a specific mirror + + +

+ + +
url --location 'https://<ORGANIZATION_ID>.<SOURCE_REGION>.lakefscloud.io/service/replication/v1/repositories/<SOURCE_REPO>/mirrors/<MIRROR_ID>' \
+-u <ACCESS_KEY_ID>:<SECRET_ACCESS_KEY> -s
+
+

+ + + Deleting a specific mirror + + +

+ + +
curl --location --request DELETE 'https://<ORGANIZATION_ID>.<SOURCE_REGION>.lakefscloud.io/service/replication/v1/repositories/<SOURCE_REPO>/mirrors/<MIRROR_ID>' \
+-u <ACCESS_KEY_ID>:<SECRET_ACCESS_KEY>
+
+

+ + + Limitations + + +

+ + +
    +
  1. Mirroring is currently only supported on AWS S3 and lakeFS Cloud for AWS
  2. +
  3. Read-only mirrors cannot be written to. Mirroring is one-way, from source to destination(s)
  4. +
  5. Currently, only branches are mirrored. Tags and arbitrary commits that do not belong to any branch are not replicated
  6. +
  7. lakeFS Hooks will only run on the source repository, not its replicas
  8. +
  9. Replication is still asynchronous: reading from a branch will always return a valid commit that the source has pointed to, but it is not guaranteed to be the latest commit the source branch is pointing to.
  10. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/private-link.html b/v1.46/howto/private-link.html new file mode 100644 index 000000000..c7f776512 --- /dev/null +++ b/v1.46/howto/private-link.html @@ -0,0 +1,912 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Private Link | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Private Link + + +

+ +

lakeFS Cloud

+ +

Private Link enables lakeFS Cloud to interact with your infrastructure using private networking.

+ + +

+ + + Supported Vendors + + +

+ + +

At the moment, we support Private-Link with AWS and Azure. If you are looking for Private Link for GCP please contact us.

+ +
+ +
+

+ + + Access Methods + + +

+ + +

There are two types of Private Link implementation:

+ +
    +
  • +

    Front-End Access refers to API and UI access. Use this option if you’d like your lakeFS application to be exposed only to your infrastructure and not to the whole internet.

    +
  • +
  • +

    Back-End Access refers to the network communication between the lakeFS clusters we host, and your infrastructure. Use this option if you’d like lakeFS to communicate with your servers privately and not over the internet.

    +
  • +
+ +

The two types of access are not mutually exclusive nor are they dependent on each other.

+ + +

+ + + Front-End Access + + +

+ + +

Prerequisites:

+
    +
  • Administrator access to your AWS account
  • +
  • In order for us to communicate with your account privately, we’ll need to create a service endpoint on our end first.
  • +
+ +

Steps:

+
    +
  1. Login to your AWS account
  2. +
  3. Go to AWS VPC Service
  4. +
  5. Filter the relevant VPC & Navigate to Endpoints
  6. +
  7. Click Create endpoint
  8. +
  9. Fill in the following: +
      +
    • Name: lakefs-cloud
    • +
    • Service category: Other endpoint services
    • +
    • Service name: input from Treeverse team (see prerequisites)
    • +
    • Click Verify service
    • +
    • Pick the VPC you’d like to expose this service to.
    • +
    • Click Create endpoint
    • +
    +
  10. +
+ +

Now you can access your infrastructure privately using the endpoint DNS name. If you would like to change the DNS name to a friendly one please contact support@treeverse.io.

+

+ + + Back-End Access + + +

+ + +

Prerequisites:

+
    +
  • Administrator access to your AWS account
  • +
+ +

Steps:

+
    +
  1. Login to your AWS account
  2. +
  3. Go to AWS VPC Service
  4. +
  5. Filter the relevant VPC & Navigate to Endpoints
  6. +
  7. Click endpoint service
  8. +
  9. Fill in the following: +
      +
    • Name: lakefs-cloud
    • +
    • Load Balancer Type: Network
    • +
    • Available load balancers: pick the load balancer you’d like lakefs-cloud to send events to.
    • +
    • Click Create
    • +
    +
  10. +
  11. Pick the newly created Endpoint Service from within the Endpoint Services page.
  12. +
  13. Navigate to the Allow principals tab.
  14. +
  15. Click Allow principals
  16. +
  17. Fill in the following ARN: arn:aws:iam::924819537486:root
  18. +
  19. Click Allow principals
  20. +
+ +

That’s it on your end! Now, we’ll need the service name you’ve just created in order to associate it with our infrastructure, once we do, we’ll be ready to use the back-end access privately.

+
+ +
+

Azure Private Link enables secure access to Azure services from a private endpoint within your virtual network. +By using Azure Private Link with lakeFS, you can securely access lakeFS services without exposing traffic to the public internet. +In this manual, we will guide you through the steps to enable Azure Private Link to your lakeFS instance.

+

+ + + Register your Azure subscription + + +

+ + +

To automatically approve private endpoint connections to the lakeFS network, please provide us with your subscription. If required, you can register multiple subscriptions.

+ + + +

Once your subscription is in our trusted subscriptions navigate to the Azure portal and do the following steps:

+
    +
  1. Navigate to the private endpoint
  2. +
  3. Click Create
  4. +
  5. On the first step (basics): +
      +
    • Select your subscription
    • +
    • Specify the desired resource group used to access lakeFS
    • +
    • Provide a name for your private endpoint instance
    • +
    • Specify the region of your lakeFS instance
    • +
    +
  6. +
  7. On the second step (Resource) +
      +
    • In connection method select connect to an Azure resource by resource ID or alias
    • +
    • Insert the alias provided by us into the Resource ID or alias
    • +
    • No need to add a request message
    • +
    +
  8. +
  9. Continue with the steps and run Review + Create
  10. +
+

+ + + Create a DNS entry for your private endpoint + + +

+ + +

Update your DNS server to resolve your account URL (which will be provided by us) to the Private Link IP address. +You can add the DNS entry to your on-premises DNS server or private DNS on your VNet, to access lakeFS services.

+
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/protect-branches.html b/v1.46/howto/protect-branches.html new file mode 100644 index 000000000..b1ff009cb --- /dev/null +++ b/v1.46/howto/protect-branches.html @@ -0,0 +1,787 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Branch Protection | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Branch Protection Rules + + +

+ + +

Define branch protection rules to prevent direct changes and commits to specific branches. +Only merges are allowed into protected branches. Together with the power of pre-merge hooks, +you can run validations on your data before it reaches your important branches and is exposed to consumers.

+ +

You can create rules for a specific branch or any branch that matches a name pattern you specify with glob syntax (supporting ? and * wildcards).

+

+ + + How it works + + +

+ + +

When at least one protection rule applies to a branch, the branch is protected. The following operations will fail on protected branches:

+
    +
  1. Object write operations: upload and delete objects.
  2. +
  3. Branch operations: commit and reset uncommitted changes.
  4. +
+ +

To operate on a protected branch, merge commits from other branches into +it. Use pre-merge hooks to validate the changes before +they are merged.

+ +

Reverting a previous commit using lakectl branch revert is allowed on a protected branch.

+

+ + + Managing branch protection rules + + +

+ + +

This section explains how to use the lakeFS UI to manage rules. You can also use the command line and API.

+

+ + + Reaching the branch protection rules page + + +

+ + +
    +
  1. On lakeFS, navigate to the main page of the repository.
  2. +
  3. Click on the Settings tab.
  4. +
  5. In the left menu, click Branches.
  6. +
+

+ + + Adding a rule + + +

+ + +

To add a new rule, click the Add button. In the dialog, enter the branch name pattern and then click Create.

+ +

Adding a branch protection rule

+

+ + + Deleting a rule + + +

+ + +

To delete a rule, click the Delete button next to it.

+ +

Deleting a branch protection rule

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/pull-requests.html b/v1.46/howto/pull-requests.html new file mode 100644 index 000000000..dc471d25d --- /dev/null +++ b/v1.46/howto/pull-requests.html @@ -0,0 +1,795 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Pull Requests | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Pull Requests + + +

+ + +

A pull request is a proposal to merge a set of changes from one branch into another. +In a pull request, collaborators can review the proposed set of changes before they integrate the changes. +Pull requests display the differences, or diffs, between the content in the source branch and the content in the target branch.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Open a Pull Request
  2. +
  3. Review Changes
  4. +
  5. Merge or Close
  6. +
  7. View Pull Requests
  8. +
+ +
+

+ + + Open a Pull Request + + +

+ + +

Create a branch, and make all the necessary changes in that branch. +When your changes are ready for review, head over to the Pull Requests tab in your repository. +Choose your source branch and target branch, add a title and description (optional, and markdown is supported).

+ +

Open Pull Request

+ +

When ready, click Create Pull Request. You will be redirected to the newly created pull request page.

+

+ + + Review Changes + + +

+ + +

Run validation checks or automated data quality tests to ensure that the changes meet your standards.

+ +

Review Pull Request

+ +

Every Pull Request is assigned a unique ID. You can share the Pull Request’s URL with others to review the change.

+ +

As with any lakeFS reference, reviewers can take the source branch, query, test and modify it as necessary prior to merging.

+

+ + + Merge or Close + + +

+ + +

Once the review is complete and all checks have passed, click the Merge pull request button to merge the changes into the target branch.

+ +

Merged Pull Request

+ +

The data is now updated in a controlled and transparent manner.

+ +

If the changes are not ready to be merged, you can close the pull request without merging the changes, by clicking the Close pull request button.

+

+ + + View Pull Requests + + +

+ + +

You can view all open and closed pull requests in the Pull Requests tab in your repository. +The tabs (Open, Closed) allow you to filter the list of pull requests according to their status.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/scim.html b/v1.46/howto/scim.html new file mode 100644 index 000000000..3dcaae02a --- /dev/null +++ b/v1.46/howto/scim.html @@ -0,0 +1,868 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +System for Cross-domain Identity Management (SCIM) | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + System for Cross-domain Identity Management (SCIM) + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +

lakeFS Cloud includes an SCIM v2.0 compliant server, which can integrate with SCIM clients (IdPs) to automate provisioning/de-provisioning of users and groups.

+ + +

+ + + Officially Supported Clients (IdPs), Capabilities, and Limitations + + +

+ +

+ + + Supported Clients (IdPs) + + +

+ + +

Currently, the lakeFS Cloud SCIM server has been tested and validated with Entra ID (a.k.a Azure AD). However, with SCIM v2.0 being an accepted standard, any SCIM-compliant IdP should be able to integrate with lakeFS Cloud.

+

+ + + Capabilities + + +

+ +

+ + + User Provisioning + + +

+ + +
    +
  • Create users: Users and members of groups assigned to the application will be provisioned in lakeFS Cloud
  • +
  • Update user attributes: Changes to supported user attributes are synced to lakeFS Cloud
  • +
  • Deactivate users: Deactivating a user or removing their assignment to the application will disable them in lakeFS Cloud
  • +
  • User adoption: Users that are already found in lakeFS Cloud will be “adopted” by the IdP and not re-created
  • +
+

+ + + Group Provisioning + + +

+ + +
    +
  • Create groups: Groups assigned to the application are created in lakeFS Cloud and any group user members are created and added to the group in lakeFS Cloud
  • +
  • Update group name: When a synced group is renamed in the IdP, it will be renamed in lakeFS Cloud
  • +
  • Add/remove members: When members are added/removed from an assigned group, they will be added/removed from the group in lakeFS Cloud
  • +
  • Group adoption: Groups that already exist in lakeFS Cloud will be “adopted” by the IdP and not re-created
  • +
+ + + +

The lakeFS Cloud SCIM server requires the minimum set of user attributes required for provisioning. The required attributes are a sub-set of the basic user profile, which is exchanged during federated authentication/SSO login. User consent is requested by the IdP upon first login to lakeFS Cloud.

+

+ + + Known Limitations + + +

+ + +
    +
  • User and group policies can only be managed in lakeFS
    +This means groups and users newly created via SCIM only have basic read permissions. The lakeFS UI or API must be used to attach policies to those users and groups. However, if a user is created and added to an existing group with an attached policy, that user will receive the permissions allowed by the policy attached to the group.
  • +
  • Only direct group memberships are provisioned via SCIM
    +Both Okta and Entra ID only support syncing direct group membership via SCIM. This means that if you assign a group to the application, only its user members will be provisioned via SCIM. SCIM provisioning will not cascade to member groups and their members, and so forth.
  • +
+

+ + + Enabling SCIM in lakeFS Cloud + + +

+ + +

To enable SCIM support in lakeFS Cloud, you need to log into the cloud admin. In the cloud admin, SCIM settings are under Access > Settings. SCIM is not enabled by default, so to enable SCIM for the organization, click the Setup Provisioning Button.

+ +

lakeFS Cloud SCIM Settings

+ +

Clicking the button will enable SCIM for the organization and provide the details you’ll need to set up your IdP to work with lakeFS Cloud SCIM.

+ +

lakeFS Cloud SCIM Configuration

+ +

To set up your IdP, you’ll need the lakeFS Cloud SCIM provisioning endpoint and you’ll also need to generate an integration token. When creating a new integration token, you can optionally provide a description for future reference.

+ +
+

Note: The token value is only presented once, right after creation. Make sure to copy the token, as its value isn’t stored and cannot be retrieved after the initial creation.

+
+

+ + + Setting Up SCIM Provisioning in Entra ID (a.k.a Azure AD) + + +

+ + +
+

Note: This guide assumes you’ve already set up an Entra ID enterprise application for federated authentication to lakeFS Cloud.

+
+ +

In the Entra ID admin dashboard, go to Enterprise Applications and choose the lakeFS Cloud enterprise application from the list. Then click Provisioning in the sidebar and then Get Started.

+ +
    +
  1. In the provisioning settings set mode to Automatic
  2. +
  3. In Tenant URL enter the URL from the lakeFS Cloud provisioning settings
  4. +
  5. In Secret Token paste the token you copied in the previous step. If you haven’t created a token yet, you may do so now
  6. +
  7. Click Test Connection
  8. +
  9. If the test fails, please ensure you’ve entered the correct SCIM endpoint URL from lakeFS Cloud and copied the token correctly. Otherwise, click “Save” at the top of the settings panel
  10. +
+ +
+

Note: lakeFS Cloud is designed to work with the default attribute mapping for users and groups provided by Entra ID. +If your organization has customized the user and/or group entities in Entra ID, you might want to set mappings in accordance with those. +You can find details of how this is done in the Entra ID documentation.
+Incorrectly modifying these mappings can break provisioning functionality, so it’s advised to do so cautiously and only when necessary.

+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/sizing-guide.html b/v1.46/howto/sizing-guide.html new file mode 100644 index 000000000..62cbd9420 --- /dev/null +++ b/v1.46/howto/sizing-guide.html @@ -0,0 +1,1434 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Sizing Guide | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Sizing guide + + +

+ + +

Note: For a scalable managed lakeFS service with guaranteed SLAs, try lakeFS Cloud

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. System Requirements
  2. +
  3. Scaling factors
  4. +
  5. Benchmarks
  6. +
  7. Important metrics
  8. +
  9. Reference architectures
  10. +
+ +
+

+ + + System Requirements + + +

+ +

+ + + Operating Systems and ISA + + +

+ +

lakeFS can run on MacOS and Linux. Windows binaries are available but not rigorously tested - +we don’t recommend deploying lakeFS to production on Windows. +x86_64 and arm64 architectures are supported for both MacOS and Linux.

+

+ + + Memory and CPU requirements + + +

+ +

lakeFS servers require a minimum of 512mb of RAM and 1 CPU core. +For high throughput, additional CPUs help scale requests across different cores. +“Expensive” operations such as large diff or commit operations can take advantage of multiple cores.

+

+ + + Network + + +

+ +

If using the data APIs such as the S3 Gateway, +lakeFS will require enough network bandwidth to support the planned concurrent network upload/download operations. +For most cloud providers, more powerful machines (i.e., more expensive and usually containing more CPU cores) also provide increased network bandwidth.

+ +

If using only the metadata APIs (for example, only using the Hadoop/Spark clients), network bandwidth is minimal, +at roughly 1Kb per request.

+

+ + + Disk + + +

+ +

lakeFS greatly benefits from fast local disks. +A lakeFS instance doesn’t require any strong durability guarantees from the underlying storage, +as the disk is only ever used as a local caching layer for lakeFS metadata and not for long-term storage. +lakeFS is designed to work with ephemeral disks - +these are usually based on NVMe and are tied to the machine’s lifecycle. +Using ephemeral disks lakeFS can provide a very high throughput/cost ratio, +probably the best that could be achieved on a public cloud, so we recommend those.

+ +

A local cache of at least 512 MiB should be provided. +For large installations (managing >100 concurrently active branches, with >100M objects per commit), +we recommend allocating at least 10 GiB - since it’s a caching layer over a relatively slow storage (the object store), +see Important metrics below to understand how to size this: it should be big enough to hold all commit metadata for actively referenced commits.

+

+ + + lakeFS KV Store + + +

+ + +

lakeFS uses a key-value database to manage branch references, authentication and authorization information +and to keep track of currently uncommitted data across branches.
+Please refer to the relevant driver tab for best practices, requirements and benchmarks.

+

+ + + Storage + + +

+ +

The dataset stored in the metadata store is relatively modest as most metadata is pushed down into the object store. +Required storage is mostly a factor of the amount of uncommitted writes across all branches at any given point in time: +in the range of 150 MiB per every 100,000 uncommitted writes.

+ +

We recommend starting at 10 GiB for a production deployment, as it will likely be more than enough.

+ +
+ +
+ +

RAM
+Since the data size is small, it’s recommended to provide enough memory to hold the vast majority of that data in RAM. +Cloud providers will save you the need to tune this parameter - it will be set to a fixed percentage the chosen instance’s available RAM (25% on AWS RDS, 30% on Google Cloud SQL). +It is recommended that you check with your selected cloud provider for configuration and provisioning information for you database. +For self-managed database instances follow these best practices

+ +

Ideally, configure the shared_buffers +of your PostgreSQL instances to be large enough to contain the currently active dataset. +Pick a database instance with enough RAM to accommodate this buffer size at roughly x4 the size given for shared_buffers. For example, if an installation has ~500,000 uncommitted writes at any given time, it would require about 750 MiB of shared_buffers +that would require about 3 GiB of RAM.

+ +

CPU
+PostgreSQL CPU cores help scale concurrent requests. 1 CPU core for every 5,000 requests/second is ideal.

+
+
+

lakeFS will create a DynamoDB table for you, defaults to on-demand capacity setting. No need to specify how much read and write throughput you expect your application to perform, as DynamoDB instantly accommodates your workloads as they ramp up or down.

+ +

You can customize the table settings to provisioned capacity which allows you to manage and optimize your costs by allocating read/write capacity in advance (see Benchmarks)

+ +

Notes:

+
    +
  • Using DynamoDB on-demand capacity might generate unwanted costs if the table is abused, if you’d like to cap your costs, make sure to change the table to use provisioned capacity instead.
  • +
  • lakeFS doesn’t manage the DynamoDB’s table lifecycle, we’ve included the table creation in order to help evaluating the system with minimal effort, any change to the table beyond the table creation - will need to be handled manually or by 3rd party tools.
  • +
+ +

RAM
+Managed by AWS.

+ +

CPU
+Managed by AWS.

+
+
+

+ + + Scaling factors + + +

+ + +

Scaling lakeFS, like most data systems, moves across two axes: throughput of requests (amount per given timeframe) and latency (time to complete a single request).

+

+ + + Understanding latency and throughput considerations + + +

+ + +

Most lakeFS operations are designed to be very low in latency. +Assuming a well-tuned local disk cache (see Storage above), +most critical path operations +(writing objects, requesting objects, deleting objects) are designed to complete in <25ms at p90. +Listing objects obviously requires accessing more data, but should always be on-par with what the underlying object store can provide, +and in most cases, it’s actually faster. +At the worst case, for directory listing with 1,000 common prefixes returned, expect a latency of 75ms at p90.

+ +

Managing branches (creating them, listing them and deleting them) are all constant-time operations, generally taking <30ms at p90.

+ +

Committing and merging can take longer, as they are proportional to the amount of changes introduced. +This is what makes lakeFS optimal for large Data Lakes - +the amount of changes introduced per commit usually stays relatively stable while the entire data set usually grows over time. +This means lakeFS will provide predictable performance: +committing 100 changes will take roughly the same amount of time whether the resulting commit contains 500 or 500 million objects.

+ +

See Data Model for more information.

+ +

Scaling throughput depends very much on the amount of CPU cores available to lakeFS. +In many cases, it’s easier to scale lakeFS across a fleet of smaller cloud instances (or containers) +than scale up with machines that have many cores. In fact, lakeFS works well in both cases. +Most critical path operations scale very well across machines.

+

+ + + Benchmarks + + +

+ + +
+ +
+

+ + + PostgresSQL + + +

+ +

All benchmarks below were measured using 2 x c5ad.4xlarge instances +on AWS us-east-1. +Similar results can be achieved on Google Cloud using a c2-standard-16 machine type, with an attached local SSD. +On Azure, you can use a Standard_F16s_v2 virtual machine.

+ +

The PostgreSQL instance that was used is a db.m6g.2xlarge +(8 vCPUs, 32 GB RAM). Equivalent machines on Google Cloud or Azure should yield similar results.

+ +

The example repository we tested against contains the metadata of a large lakeFS installation, +where each commit contains ~180,000,000 objects (representing ~7.5 Petabytes of data).

+ +

All tests are reproducible using the lakectl abuse command, +so use it to properly size and tune your setup. All tests are accompanied by the relevant lakectl abuse command that generated them.

+

+ + + Random reads + + +

+ + +

This test generates random read requests to lakeFS, +in a given commit. Paths are requested randomly from a file containing a set of preconfigured (and existing) paths.

+ +

command executed:

+ +
lakectl abuse random-read \
+    --from-file randomly_selected_paths.txt \
+    --amount 500000 \
+    --parallelism 128 \
+    lakefs://example-repo/<commit hash>
+
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and commit hash.

+ +

Result Histogram (raw):

+ +
Histogram (ms):
+1	0
+2	0
+5	37945
+7	179727
+10	296964
+15	399682
+25	477502
+50	499625
+75	499998
+100	499998
+250	500000
+350	500000
+500	500000
+750	500000
+1000	500000
+5000	500000
+min	3
+max	222
+total	500000
+
+ +

So 50% of all requests took <10ms, while 99.9% of them took <50ms

+ +

throughput:

+ +

Average throughput during the experiment was 10851.69 requests/second

+

+ + + Random Writes + + +

+ + +

This test generates random write requests to a given lakeFS branch. +All the paths are pre-generated and don’t overwrite each other (as overwrites are relatively rare in a Data Lake setup).

+ +

command executed:

+ +
lakectl abuse random-write \
+    --amount 500000 \
+    --parallelism 64 \
+    lakefs://example-repo/main
+
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and branch.

+ +

Result Histogram (raw):

+ +
Histogram (ms):
+1	0
+2	0
+5	30715
+7	219647
+10	455807
+15	498144
+25	499535
+50	499742
+75	499784
+100	499802
+250	500000
+350	500000
+500	500000
+750	500000
+1000	500000
+5000	500000
+min	3
+max	233
+total	500000
+
+ +

So, 50% of all requests took <10ms, while 99.9% of them took <25ms.

+ +

throughput:

+ +

The average throughput during the experiment was 7595.46 requests/second.

+

+ + + Branch creation + + +

+ + +

This test creates branches from a given reference.

+ +

command executed:

+ +
lakectl abuse create-branches \
+    --amount 500000 \
+    --branch-prefix "benchmark-" \
+    --parallelism 256 \
+    lakefs://example-repo/<commit hash>
+
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and commit hash.

+ +

Result Histogram (raw):

+ +
Histogram (ms):
+1	0
+2	1
+5	5901
+7	39835
+10	135863
+15	270201
+25	399895
+50	484932
+75	497180
+100	499303
+250	499996
+350	500000
+500	500000
+750	500000
+1000	500000
+5000	500000
+min	2
+max	304
+total	500000
+
+ +

So, 50% of all requests took <15ms, while 99.9% of them took <100ms.

+ +

throughput:

+ +

The average throughput during the experiment was 7069.03 requests/second.

+
+
+

+ + + DynamoDB + + +

+ +

All benchmarks below were measured using m5.xlarge instance on AWS us-east-1.

+ +

The DynamoDB table that was used was provisioned with 500/1000 read/write capacity.

+ +

The example repository we tested against contains the metadata of a large lakeFS installation, where each commit contains ~100,000,000 objects (representing ~3.5 Petabytes of data).

+ +

All tests are reproducible using the lakectl abuse command, so use it to properly size and tune your setup. All tests are accompanied by the relevant lakectl abuse command that generated them.

+

+ + + Random reads + + +

+ + +

This test generates random read requests to lakeFS, +in a given commit. Paths are requested randomly from a file containing a set of preconfigured (and existing) paths.

+ +

command executed:

+ +
lakectl abuse random-read \
+    --from-file randomly_selected_paths.txt \
+    --amount 500000 \
+    --parallelism 128 \
+    lakefs://example-repo/<commit hash>
+
+ +

Result Histogram (raw): Provisioned read capacity units = 1000 +Provisioned write capacity units = 1000

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	0
+25	122
+50	47364
+75	344489
+100	460404
+250	497912
+350	498016
+500	498045
+750	498111
+1000 498176
+5000 499478
+min	18
+max	52272
+total 500000
+
+ +

Result Histogram (raw): Provisioned read capacity units = 500 +Provisioned write capacity units = 500

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	1
+25	2672
+50	239661
+75	420171
+100	470146
+250	486603
+350	486715
+500	486789
+750	487443
+1000	488113
+5000	493201
+min	14
+max	648085
+total	499998
+
+

+ + + Random Writes + + +

+ + +

This test generates random write requests to a given lakeFS branch. +All the paths are pre-generated and don’t overwrite each other (as overwrites are relatively rare in a Data Lake setup).

+ +

command executed:

+ +
lakectl abuse random-write \
+    --amount 500000 \
+    --parallelism 64 \
+    lakefs://example-repo/main
+
+ +

Result Histogram (raw): Provisioned read capacity units = 1000 +Provisioned write capacity units = 1000

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	0
+25	24
+50	239852
+75	458504
+100	485225
+250	493687
+350	493872
+500	493960
+750	496239
+1000	499194
+5000	500000
+min	23
+max	4437
+total	500000
+
+

Result Histogram (raw): Provisioned read capacity units = 500 +Provisioned write capacity units = 500

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	0
+25	174
+50	266460
+75	462641
+100	484486
+250	490633
+350	490856
+500	490984
+750	492973
+1000 495605
+5000 498920
+min	21
+max	50157
+total 500000
+
+

+ + + Branch creation + + +

+ + +

This test creates branches from a given reference.

+ +

command executed:

+ +
lakectl abuse create-branches \
+    --amount 500000 \
+    --branch-prefix "benchmark-" \
+    --parallelism 256 \
+    lakefs://example-repo/<commit hash>
+
+ +

Result Histogram (raw): Provisioned read capacity units = 1000 +Provisioned write capacity units = 1000

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	0
+25	0
+50	628
+75	26153
+100	58099
+250	216160
+350	307078
+500	406165
+750	422898
+1000	431332
+5000	475848
+min	41
+max	430725
+total	490054
+
+ +

Result Histogram (raw): Provisioned read capacity units = 500 +Provisioned write capacity units = 500

+ +
Histogram (ms):
+1	0
+2	0
+5	0
+7	0
+10	0
+15	0
+25	0
+50	3132
+75	155570
+100	292745
+250	384224
+350	397258
+500	431141
+750	441360
+1000 445597
+5000 469538
+min	39
+max	760626
+total 497520
+
+ +
+
+

+ + + Important metrics + + +

+ + +

lakeFS exposes metrics using the Prometheus protocol. +Every lakeFS instance exposes a /metrics endpoint that could be used to extract them.

+ +

Here are a few notable metrics to keep track of when sizing lakeFS:

+ +

api_requests_total - Tracks throughput of API requests over time.

+ +

api_request_duration_seconds - Histogram of latency per operation type.

+ +

gateway_request_duration_seconds - Histogram of latency per S3 Gateway operation.

+ +
+ +
+ +

dynamo_request_duration_seconds - Time spent doing DynamoDB requests.

+ +

dynamo_consumed_capacity_total - The capacity units consumed by operation.

+ +

dynamo_failures_total - The total number of errors while working for kv store.

+ +
+
+

+ + + Reference architectures + + +

+ + +

Below are a few example architectures for lakeFS deployment.

+

+ + + Reference Architecture: Data Science/Research environment + + +

+ + +

Use case: Manage Machine learning or algorithms development. +Use lakeFS branches to achieve both isolation and reproducibility of experiments. +Data being managed by lakeFS is both structured tabular data, +as well as unstructured sensor and image data used for training. +Assuming a team of 20-50 researchers, with a dataset size of 500 TiB across 20M objects.

+ +

Environment: lakeFS will be deployed on Kubernetes. +managed by AWS EKS +with PostgreSQL on AWS RDS Aurora

+ +

Sizing: Since most of the work is done by humans (vs. automated pipelines), most experiments tend to be small in scale, +reading and writing 10s to 1000s of objects. +The expected amount of branches active in parallel is relatively low, around 1-2 per user, +each representing a small amount of uncommitted changes at any given point in time. +Let’s assume 5,000 uncommitted writes per branch = ~500k.

+ +

To support the expected throughput, a single moderate lakeFS instance should be more than enough, +since requests per second would be on the order of 10s to 100s. +For high availability, we’ll deploy 2 pods with 1 CPU core and 1 GiB of RAM each.

+ +

Since the PostgreSQL instance is expected to hold a very small dataset +(at 500k, expected dataset size is 150MiB (for 100k records) * 5 = 750MiB). +To ensure we have enough RAM to hold this, we’ll need 3 GiB of RAM, so, a very moderate Aurora instance db.t3.large (2 vCPUs, 8 GB RAM) will be more than enough. +An equivalent database instance on GCP or Azure should give similar results.

+ +

ML and Research lakeFS reference architecture

+

+ + + Reference Architecture: Automated Production Pipelines + + +

+ + +

Use case: Manage multiple concurrent data pipelines using Apache Spark and Airflow. +Airflow DAGs start by creating a branch for isolation and for CI/CD. +Data being managed by lakeFS is structured, tabular data. The total dataset size is 10 PiB, spanning across 500M objects. +The expected throughput is 10k reads/second + 2k writes per second across 100 concurrent branches.

+ +

Environment: lakeFS will be deployed on Kubernetes. +managed by AWS EKS +with PostgreSQL on AWS RDS

+ +

Sizing: Data pipelines tend to be bursty in nature: +reading in a lot of objects concurrently, doing some calculation or aggregation, and then writing many objects concurrently. +The expected amount of branches active in parallel is high, +with many Airflow DAGs running per day, each representing a moderate amount of uncommitted changes at any given point in time. +Let’s assume 1,000 uncommitted writes/branch * 2500 branches = ~2.5M records.

+ +

To support the expected throughput, looking the benchmarking numbers above, +we’re doing roughly 625 requests/core, so 24 cores should cover our peak traffic. We can deploy 6 * 4 CPU core pods.

+ +

On to the PostgreSQL instance - at 500k, the expected dataset size is 150MiB (for 100k records) * 25 = 3750 MiB. +To ensure we have enough RAM to hold this, we’ll need at least 15 GiB of RAM, so we’ll go with a db.r5.xlarge (4 vCPUs, 32GB RAM) Aurora instance. +An equivalent database instance on GCP or Azure should give similar results.

+ +

Automated pipelines lakeFS reference architecture

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/unity-delta-sharing.html b/v1.46/howto/unity-delta-sharing.html new file mode 100644 index 000000000..ce5395c0b --- /dev/null +++ b/v1.46/howto/unity-delta-sharing.html @@ -0,0 +1,926 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Unity Delta Sharing | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Unity Delta Sharing + + +

+ +

lakeFS Cloud

+ +
+

Please note, as of June 15th 2024, the Unity Delta Sharing feature will be removed. To integrate lakeFS with Unity Catalog, refer to the Unity integration docs.

+
+

+ + + Introduction + + +

+ + +

lakeFS Unity Delta Sharing provides a read-only experience from Unity Catalog for lakeFS customers. Currently, this is available as a private +preview. It provides full read-only functionality for Unity Catalog. It does not provide a “self-service” experience to set up the service.

+

+ + + Setup + + +

+ + +

This guide explains how to set up and use lakeFS Delta Sharing. Currently, you will have to configure lakeFS Delta Sharing in collaboration with +Treeverse Customer Success. Once set up is complete, you will of course be able to use lakeFS Delta Sharing on existing and on new tables without +further assistance.

+

+ + + 1. Collect data and initial setup + + +

+ + +
    +
  • +

    Select a Delta Sharing configuration URL. This is a single location on lakeFS to hold the top-level configuration of lakeFS Delta Sharing across all +repositories of your organization. Typically, it will have the form lakefs://REPO/main/lakefs_delta_sharing.yaml for one of your +repositories. A longer path may be supplied - however, we do recommend keeping it on the main branch, as this object represents state +for the entire installation.

    +
  • +
  • +

    Create a user lakefs-delta-sharing-service for lakeFS Delta Sharing, and an access key for that user. It should have at least read permissions for +the configuration URL and for all repositories and all data accesses by Unity. lakeFS Delta Sharing will these credentials to communicate with lakeFS.

    +
  • +
+ +

Communicate these items to Customer Success:

+ +
    +
  • Configuration URL
  • +
  • Access key ID and secret access key for user lakefs-delta-sharing-service.
  • +
+ +

Note: All YAML files extensions used in this guide must be yaml. Do not use a yml extension instead.

+

+ + + 2. Initial configuration + + +

+ + +

Select a secret authorization token to share Unity catalog. Unity catalog will use to authenticate to the lakeFS Delta Sharing server.
+You might use this command on Linux:

+ +
head -c 50 /dev/random | base64
+
+ +

Create a file lakefs_delta_sharing.yaml and place it at the config URL selected above. It should look like this:

+ +
authorization_token: "GENERATED TOKEN"
+# Map lakeFS repositories to Unity shares
+repositories:
+    - id: sample-repo
+      share_name: undev
+      # List branches and prefixes to export.  Each of these branches (and only
+      # these branches) will be available as a schema on Unity.
+      branches:
+          - main
+          - staging
+          - dev_*
+    - id: repo2
+      share_name: share_two
+      branches:
+      - "*"
+
+ +

Note that a plain “*” line must be quoted in YAML.

+ +

Upload it to your config URL. For instance if the config URL is lakefs://repo/main/lakefs_delta_sharing.yaml, you might use:

+ +
lakectl fs upload -s ./lakefs_delta_sharing.yaml lakefs://repo/main/lakefs_delta_sharing.yaml
+
+

+ + + 3. Connect Unity to lakeFS Delta Sharing! + + +

+ + +

You now need to configure Unity to use the lakeFS Delta Sharing server. Create a share provider file config.share.json; see the Delta Sharing manual:

+ +
   {
+     "shareCredentialsVersion": 1,
+     "endpoint": "https://ORG_ID.REGION.lakefscloud.io/service/delta-sharing/v1",
+     "bearerToken": "GENERATED TOKEN",
+     "expirationTime": "2030-01-01T00:00:00.0Z"
+   }
+
+ +

“GENERATED TOKEN” is the secret authorization token use above.

+ +

Install the databricks cli. We will use it to create Delta Share on Unity. Follow the +instructions to configure it.

+ +

Run the provider creation command:

+
   databricks unity-catalog providers create \
+       --name lakefs-cloud \
+       --recipient-profile-json-file config.share.json
+
+ +

Go to “Data » Delta Sharing” on the DataBricks environment. Once Treeverse have configured lakeFS Delta Sharing on your account with your config URL, +the “lakefs-cloud” provider should appear under “Shared with me”.

+ +

lakeFS-Cloud provider appearing on DataBricks Delta Sharing / Shared with me

+ +

Click the provider to see its shares.

+ +

lakeFS-Cloud provider, showing share and create catalog

+ +

You can now create a catalog from these shares.

+ +

lakeFS-Cloud provider, create catalog from share

+ +

And you can see schemas for each of the branches that you configured in the share. Here branch name dev_experiment1 matches the pattern dev_* that +we defined in the configuration object lakefs-delta-sharing.yaml, so it appears as a schema.

+ +

lakeFS-Cloud provider, every configured branch is a schema

+ +

At this point you have configured Delta Sharing on lakeFS, and DataBricks to communicate with lakeFS delta sharing. No further Treeverse involvement is required.
+Updates to lakefs_delta_sharing.yaml will update within a minute of uploading a new version.

+

+ + + 4. Configure tables + + +

+ + +

Everything is ready: lakeFS repositories are configured as shares, and branches are configured as schemas. Now you can define tables! Once a +repository is shared, its tables are configured as a table descriptor object on the repository on the path _lakefs_tables/TABLE.yaml.

+

+ + + Delta Lake tables + + +

+ + +

Delta Lake format includes full metadata, so you only need to configure the prefix:

+ +
name: users
+type: delta
+path: path/to/users/
+
+ +

Note: The filename of the ‘yaml’ file containing the table definition must match the ‘name’ of the table itself. In the example above, ‘_lakefs_tables/users.yaml’.

+ +

When placed inside _lakefs_tables/users.yaml this defines a table users on the prefix path/to/users/ (so path/to/users/ holds the prefix _delta_log).

+

+ + + Hive tables + + +

+ + +

Hive metadata server tables are essentially just a set of objects that share a prefix, with no table metadata stored on the object store. You need to configure prefix, partitions, and schema.

+ +
name: clicks
+type: hive
+path: path/to/clicks/
+partition_columns: ['year']
+schema:
+  type: struct
+  fields:
+    - name: year
+      type: integer
+      nullable: false
+      metadata: {}
+    - name: page
+      type: string
+      nullable: false
+      metadata: {}
+    - name: site
+      type: string
+      nullable: true
+      metadata:
+        comment: a comment about this column
+
+ +

Useful types recognized by DataBricks Photon include integer, long, short, string, double, float, date, and timestamp.
+For exact type mappings, and whether to specify a field as nullable: false, refer to DataBricks Photon documentation.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/howto/upgrade.html b/v1.46/howto/upgrade.html new file mode 100644 index 000000000..63831114b --- /dev/null +++ b/v1.46/howto/upgrade.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/howto/virtual-host-addressing.html b/v1.46/howto/virtual-host-addressing.html new file mode 100644 index 000000000..dabf61113 --- /dev/null +++ b/v1.46/howto/virtual-host-addressing.html @@ -0,0 +1,793 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +S3 Virtual-host addressing (advanced) | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Configuring lakeFS to use S3 Virtual-Host addressing + + +

+ +

+ + + Understanding virtual-host addressing + + +

+ + +

Some systems require S3 endpoints (such as lakeFS’s S3 Gateway) to support virtual-host style addressing.

+ +

lakeFS supports this, but requires some configuration in order to extract the bucket name (used as the lakeFS repository ID) from the host address.

+ +

For example:

+ +
GET http://foo.example.com/some/location
+
+ +

There are two ways to interpret the URL above:

+
    +
  • as a virtual-host URL where the endpoint URL is example.com, the bucket name is foo, and the path is /some/location
  • +
  • as a path-based URL where the endpoint is foo.example.com, the bucket name is some and the path is location
  • +
+ +

By default, lakeFS reads URLs as path-based. To read the URL as a virtual-host request, lakeFS requires additional configuration which includes +defining an explicit set of DNS records for the lakeFS S3 gateway.

+

+ + + Adding an explicit S3 domain name to the S3 Gateway configuration + + +

+ + +

The first step would be to tell the lakeFS installation which hostnames are used for the S3 Gateway. This should be a different DNS record from the one used for e.g. the UI or API.

+ +

Typically, if the lakeFS installation is served under lakefs.example.com, a good choice would be s3.lakefs.example.com.

+ +

This could be done using either an environment variable:

+ +
LAKEFS_GATEWAYS_S3_DOMAIN_NAME="s3.lakefs.example.com"
+
+ +

Or by adding the gateways.s3.domain_name setting to the lakeFS config.yaml file:

+ +
---
+database:
+  connection_string: "..."
+
+...
+
+# This section defines an explict S3 gateway address that supports virtual-host addressing
+gateways:
+  s3:
+    domain_name: s3.lakefs.example.com
+
+ +

For more information on how to configure lakeFS, check out the configuration reference

+

+ + + Setting up the appropriate DNS records + + +

+ + +

Once our lakeFS installation is configured with an explicit S3 gateway endpoint address, we need to define 2 DNS records and have them point at our lakeFS installation. +This requires 2 CNAME records:

+ +
    +
  1. s3.lakefs.example.com - CNAME to lakefs.example.com. This would be used as the S3 endpoint when configuring clients and will serve as our bare domain.
  2. +
  3. *.s3.lakefs.example.com - Also a CNAME to lakefs.example.com. This will resolve virtual-host requests such as example-repo.s3.lakefs.example.com that lakeFS would now know how to parse.
  4. +
+ +
+

For more information on how to configure these, see the official documentation of your DNS provider.

+

On AWS, This could also be done using ALIAS records for a load balancer.

+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/index.html b/v1.46/index.html new file mode 100644 index 000000000..44ab19790 --- /dev/null +++ b/v1.46/index.html @@ -0,0 +1,928 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Welcome to lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ +
+ +

+ + + Welcome to the Lake! + + +

+ + +

+ +

lakeFS brings software engineering best practices and applies them to data.

+ +

lakeFS provides version control over the data lake, and uses Git-like semantics to create and access those versions. If you know git, you’ll be right at home with lakeFS.

+ +

With lakeFS, you can use concepts on your data lake such as branch to create an isolated version of the data, commit to create a reproducible point in time, and merge in order to incorporate your changes in one atomic action.

+ + +

+ + + How Do I Get Started? + + +

+ + +

The hands-on quickstart guides you through some core features of lakeFS.

+ +

These include branching, merging, and rolling back changes to data.

+ +

You can use the 30-day free trial of lakeFS Cloud if you want to try out lakeFS without installing anything.

+

+ + + Key lakeFS Features + + +

+ + +
    +
  • It is format-agnostic.
  • +
  • It works with numerous data tools and platforms.
  • +
  • Your data stays in place.
  • +
  • It eliminates the need for data duplication using zero-copy branching.
  • +
  • It maintains high performance over data lakes of any size.
  • +
  • It includes configurable garbage collection capabilities.
  • +
  • It is proven in production and has an active community.
  • +
+ +

lakeFS integration into data lake

+

+ + + How Does lakeFS Work With Other Tools? + + +

+ + +

lakeFS is an open source project that supports managing data in AWS S3, Azure Blob Storage, Google Cloud Storage (GCS) and any other object storage with an S3 interface. It integrates seamlessly with popular data frameworks such as Spark, Hive Metastore, dbt, Trino, Presto, and many others and includes an S3 compatibility layer.

+ +

For more details and a full list see the integrations pages.

+ +

+ +

+ +

lakeFS maintains compatibility with the S3 API to minimize adoption +friction. You can use it as a drop-in replacement for S3 from the perspective of +any tool interacting with a data lake.

+ +

For example, take the common operation of reading a collection of data from an object storage into a Spark DataFrame. For data outside a lakeFS repo, the code will look like this:

+ +
df = spark.read.parquet("s3a://my-bucket/collections/foo/")
+
+ +

After adding the data collections into my-bucket to a repository, the same operation becomes:

+ +
df = spark.read.parquet("s3a://my-repo/main-branch/collections/foo/")
+
+ +

You can use the same methods and syntax you are already using to read and write data when using a lakeFS repository. This simplifies the adoption of lakeFS - minimal changes are needed to get started, making further changes an incremental process.

+

+ + + lakeFS is Git for Data + + +

+ + +

Git conquered the world of code because it had best supported engineering best practices required by developers, in particular:

+ +
    +
  • Collaborate during development.
  • +
  • Develop and Test in isolation
  • +
  • Revert code repository to a sable version in case of an error
  • +
  • Reproduce and troubleshoot issues with a given version of the code
  • +
  • Continuously integrate and deploy new code (CI/CD)
  • +
+ +

lakeFS provides these exact benefits, that are data practitioners are missing today, and enables them a clear intuitive Git-like interface to easily manage their data like they manage code. Through its versioning engine, lakeFS enables the following built-in operations familiar from Git:

+ +
    +
  • branch: a consistent copy of a repository, isolated from other branches and their changes. Initial creation of a branch is a metadata operation that does not duplicate objects.
  • +
  • commit: an immutable checkpoint containing a complete snapshot of a repository.
  • +
  • merge: performed between two branches — merges atomically update one branch with the changes from another.
  • +
  • revert: returns a repo to the exact state of a previous commit.
  • +
  • tag: a pointer to a single immutable commit with a readable, meaningful name.
  • +
+ +

See the object model for an in-depth +definition of these, and the CLI reference for the +full list of commands.

+ +

Incorporating these operations into your data lake pipelines provides the same collaboration and organizational benefits you get when managing application code with source control.

+

+ + + The lakeFS Promotion Workflow + + +

+ + +

Here’s how lakeFS branches and merges improve the universal process of updating collections with the latest data.

+ +

lakeFS promotion workflow

+ +
    +
  1. First, create a new branch from main to instantly generate a complete “copy” of your production data.
  2. +
  3. Apply changes or make updates on the isolated branch to understand their impact prior to exposure.
  4. +
  5. And finally, perform a merge from the feature branch back to main to atomically promote the updates into production.
  6. +
+ +

Following this pattern, lakeFS facilitates a streamlined data deployment workflow that consistently produces data assets you can have total confidence in.

+

+ + + How Can lakeFS Help Me? + + +

+ + +

lakeFS helps you maintain a tidy data lake in several ways, including:

+

+ + + Isolated Dev/Test Environments with zero-copy branching + + +

+ + +

lakeFS makes creating isolated dev/test environments for ETL testing instantaneous, and through its use of zero-copy branching, cheap. This enables you to test and validate code changes on production data without impacting it, as well as run analysis and experiments on production data in an isolated clone.

+ +

👉🏻 Read more

+

+ + + Reproducibility: What Did My Data Look Like at a Point In Time? + + +

+ + +

Being able to look at data as it was at a given point is particularly useful in at least two scenarios:

+ +
    +
  1. +

    Reproducibility of ML experiments

    + +

    ML experimentation is iterative, requiring the ability to reproduce specific results. With lakeFS, you can version all aspects of an ML experiment, including the data. This enables:

    + +

    Data Lineage: Track the transformation of data from raw datasets to the final version used in experiments, ensuring transparency and traceability.

    + +

    Zero-Copy Branching: Minimize storage use by creating lightweight branches of your data, allowing for easy experimentation across different versions.

    + +

    Easy Integration: Seamlessly integrate with ML tools like MLFlow, linking experiments directly to the exact data versions used, making reproducibility straightforward.

    + +

    lakeFS enhances your ML workflow by ensuring that all versions of data are easily accessible, traceable, and reproducible.

    +
  2. +
  3. +

    Troubleshooting production problems

    + +

    Data engineers are often asked to validate the data. A user might report inconsistencies, question the accuracy, or simply report it to be incorrect.

    + +

    Since the data continuously changes, it is challenging to understand its state at the time of the error.

    + +

    With lakeFS you can create a branch from a commit to debug an issue in isolation.

    +
  4. +
+ +

👉🏻 Read More

+

+ + + Rollback of Data Changes and Recovery from Data Errors + + +

+ + +

Human error or misconfigurations can lead to erroneous data making its way into production or critical data being accidentally deleted. Traditional backups are often inadequate for recovery in these situations, as they may be outdated and require time-consuming object-level sifting.

+ +

With lakeFS, you can avoid these inefficiencies by committing snapshots of data at well-defined times. This allows for instant recovery: simply identify a good historical commit and restore or copy from it with a single operation.

+ +

👉🏻 Read more

+

+ + + Establishing data quality guarantees - CI/CD for data + + +

+ + +

The best way to deal with mistakes is to avoid them. A data source that is ingested into the lake introducing low-quality data should be blocked before exposure if possible.

+ +

With lakeFS, you can achieve this by tying data quality tests to commit and merge operations via lakeFS hooks.

+ +

👉🏻 Read more

+

+ + + Next Step + + +

+ + +

Try lakeFS on the cloud or run it locally

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/airbyte.html b/v1.46/integrations/airbyte.html new file mode 100644 index 000000000..6d8ba09e3 --- /dev/null +++ b/v1.46/integrations/airbyte.html @@ -0,0 +1,812 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Airbyte | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Airbyte + + +

+ + +

Airbyte is an open-source platform for syncing data from applications, APIs, and databases to +warehouses, lakes, and other destinations. You can use Airbyte’s connectors to get your data pipelines to consolidate +many input sources.

+ +

The integration between Airbyte and lakeFS brings resilience and manageability when you use Airbyte +connectors to sync data to your S3 buckets by leveraging lakeFS branches and atomic commits and merges.

+

+ + + Use cases + + +

+ + +

You can take advantage of lakeFS consistency guarantees and Data Lifecycle Management when ingesting data to S3 using lakeFS:

+ +
    +
  1. Consolidate many data sources to a single branch and expose them to consumers simultaneously when merging to the main branch.
  2. +
  3. Test incoming data for breaking schema changes using lakeFS hooks.
  4. +
  5. Prevent consumers from reading partial data from connectors which failed half-way through sync.
  6. +
  7. Experiment with ingested data on a branch before exposing it.
  8. +
+

+ + + S3 Connector + + +

+ + +

lakeFS exposes an S3 Gateway that enables applications to communicate +with lakeFS the same way they would with Amazon S3. +You can use Airbyte’s S3 Connector to upload data to lakeFS.

+ +
+

Note

+ +

If using Airbyte OSS, please ensure you are using S3 destination connector version 0.3.17 or higher. +Previous connector versions are not supported.

+
+

+ + + Configuring lakeFS using the connector + + +

+ + +

Set the following parameters when creating a new Destination of type S3:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
NameValueExample
EndpointThe lakeFS S3 gateway URLhttps://cute-axolotol.lakefs-demo.io
S3 Bucket NameThe lakeFS repository where the data will be writtenexample-repo
S3 Bucket PathThe branch and the path where the data will be writtenmain/data/from/airbyte Where main is the branch name, and data/from/airbyte is the path under the branch.
S3 Bucket RegionNot applicable to lakeFS, use us-east-1us-east-1
S3 Key IDThe lakeFS access key id used to authenticate to lakeFS.AKIAlakefs12345EXAMPLE
S3 Access KeyThe lakeFS secret access key used to authenticate to lakeFS.abc/lakefs/1234567bPxRfiCYEXAMPLEKEY
+ +

The UI configuration will look as follows:

+ +

S3 Destination Connector Configuration

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/airflow.html b/v1.46/integrations/airflow.html new file mode 100644 index 000000000..d2bf61a01 --- /dev/null +++ b/v1.46/integrations/airflow.html @@ -0,0 +1,875 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Apache Airflow | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Apache Airflow + + +

+ + +

Apache Airflow is a platform that allows users to programmatically author, schedule, and monitor workflows.

+ + + +

To run Airflow with lakeFS, you need to follow a few steps.

+

+ + + Create a lakeFS connection on Airflow + + +

+ + +

To access the lakeFS server and authenticate with it, create a new Airflow +Connection +of type HTTP and add it to your DAG. You can do that using the Airflow UI +or the CLI. Here’s an example Airflow command that does just that:

+ +
airflow connections add conn_lakefs --conn-type=HTTP --conn-host=http://<LAKEFS_ENDPOINT> \
+    --conn-extra='{"access_key_id":"<LAKEFS_ACCESS_KEY_ID>","secret_access_key":"<LAKEFS_SECRET_ACCESS_KEY>"}'
+
+

+ + + Install the lakeFS Airflow package + + +

+ + +

You can use pip to install the package

+ +
pip install airflow-provider-lakefs
+
+

+ + + Use the package + + +

+ +

+ + + Operators + + +

+ + +

The package exposes several operations to interact with a lakeFS server:

+ +
    +
  1. +

    CreateBranchOperator creates a new lakeFS branch from the source branch (main by default).

    + +
    task_create_branch = CreateBranchOperator(
    +   task_id='create_branch',
    +   repo='example-repo',
    +   branch='example-branch',
    +   source_branch='main'
    +)
    +
    +
  2. +
  3. +

    CommitOperator commits uncommitted changes to a branch.

    + +
    task_commit = CommitOperator(
    +    task_id='commit',
    +    repo='example-repo',
    +    branch='example-branch',
    +    msg='committing to lakeFS using airflow!',
    +    metadata={'committed_from": "airflow-operator'}
    +)
    +
    +
  4. +
  5. +

    MergeOperator merges 2 lakeFS branches.

    + +
    task_merge = MergeOperator(
    +  task_id='merge_branches',
    +  source_ref='example-branch',
    +  destination_branch='main',
    +  msg='merging job outputs',
    +  metadata={'committer': 'airflow-operator'}
    +)
    +
    +
  6. +
+

+ + + Sensors + + +

+ + +

Sensors are also available that allow synchronizing a running DAG with external operations:

+ +
    +
  1. +

    CommitSensor waits until a commit has been applied to the branch

    + +
    task_sense_commit = CommitSensor(
    +    repo='example-repo',
    +    branch='example-branch',
    +    task_id='sense_commit'
    +)
    +
    +
  2. +
  3. +

    FileSensor waits until a given file is present on a branch.

    + +
    task_sense_file = FileSensor(
    +    task_id='sense_file',
    +    repo='example-repo',
    +    branch='example-branch',
    +    path="file/to/sense"
    +)
    +
    +
  4. +
+

+ + + Example + + +

+ + +

This example DAG +in the airflow-provider-lakeFS repository shows how to use all of these.

+

+ + + Performing other operations + + +

+ + +

Sometimes an operator might not be supported by airflow-provider-lakeFS yet. You can access lakeFS directly by using:

+ +
    +
  • SimpleHttpOperator to send API requests to lakeFS.
  • +
  • BashOperator with lakectl commands. +For example, deleting a branch using BashOperator: +
    commit_extract = BashOperator(
    + task_id='delete_branch',
    + bash_command='lakectl branch delete lakefs://example-repo/example-branch',
    + dag=dag,
    +)
    +
    +
  • +
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and branch.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/athena.html b/v1.46/integrations/athena.html new file mode 100644 index 000000000..ee0a5b4b8 --- /dev/null +++ b/v1.46/integrations/athena.html @@ -0,0 +1,862 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Amazon Athena | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Amazon Athena + + +

+ + +

Deprecated Feature: Having heard the feedback from the community, we are planning to replace the below manual steps with an automated process. +You can read more about it here.

+ +

Amazon Athena is an interactive query service that makes it easy to analyze data in Amazon S3 using standard SQL.

+ +

Amazon Athena works directly above S3 and can’t access lakeFS. Tables created using Athena aren’t readable by lakeFS. +However, tables stored in lakeFS (that were created with glue/hive) can be queried by Athena.

+ +

To support querying data from lakeFS with Amazon Athena, we will use create-symlink, one of the metastore commands in lakectl. +create-symlink receives a source table, destination table, and the table location. It performs two actions:

+
    +
  1. It creates partitioned directories with symlink files in the underlying S3 bucket.
  2. +
  3. It creates a table in Glue catalog with symlink format type and location pointing to the created symlinks.
  4. +
+ +

Note +.lakectl.yaml file should be configured with the proper hive/glue credentials. For more information

+ +

create-symlink receives a table in glue or hive pointing to lakeFS and creates a copy of the table in glue. +The table data will use the SymlinkTextInputFormat, which will point to the lakeFS repository storage namespace. You will be able to query your data with Athena without copying any data. However, the symlinks table will only show the data that existed during +the copy. If the table changed in lakeFS, you need to run create-symlink again for your changed to be reflected in Athena.

+

+ + + Example: + + +

+ + +

Let’s assume that some time ago, we created a hive table my_table that is stored in lakeFS repo example under branch main, using the command:

+
CREATE EXTERNAL TABLE `my_table`(
+   `id` bigint, 
+   `key` string 
+)
+PARTITIONED BY (YEAR INT, MONTH INT)
+LOCATION 's3://example/main/my_table';
+WITH (format = 'PARQUET', external_location 's3a://example/main/my_table' );
+
+ +

The repository example has the S3 storage space s3://my-bucket/my-repo-prefix/. +After inserting some data into it, the object structure under lakefs://example/main/my_table looks as follows:

+ +

lakefs_table.png

+ +

To query that table with Athena, you need to use the create-symlink command as follows:

+ +
lakectl metastore create-symlink \
+  --repo example \
+  --branch main \
+  --path my_table \
+  --from-client-type hive \
+  --from-schema default \
+  --from-table my_table \
+  --to-schema default \ 
+  --to-table my_table  
+
+ +

The command will generate two notable outputs:

+ +
    +
  1. For each partition, the command will create a symlink file:
  2. +
+ +
aws s3 ls s3://my-bucket/my-repo-prefix/my_table/ --recursive
+2021-11-23 17:46:29          0 my-repo-prefix/my_table/symlinks/example/main/my_table/year=2021/month=11/symlink.txt
+2021-11-23 17:46:29         60 my-repo-prefix/my_table/symlinks/example/main/my_table/year=2021/month=12/symlink.txt
+2021-11-23 17:46:30         60 my-repo-prefix/my_table/symlinks/example/main/my_table/year=2022/month=1/symlink.txt
+
+ +

An example content of a symlink file, where each line represents a single object of the specific partition:

+ +
s3://my-bucket/my-repo-prefix/5bdc62da516944b49889770d98274227
+s3://my-bucket/my-repo-prefix/64262fbf3d6347a79ead641d2b2baee6
+s3://my-bucket/my-repo-prefix/64486c8de6484de69f12d7d26804c93e
+s3://my-bucket/my-repo-prefix/b0165d5c5b13473d8a0f460eece9eb26
+
+ +
    +
  1. A glue table pointing to the symlink directories structure:
  2. +
+ +
aws glue get-table --name my_table --database-name default
+
+{
+  "Table": {
+    "Name": "my_table",
+    "DatabaseName": "default",
+    "Owner": "anonymous",
+    "CreateTime": "2021-11-23T17:46:30+02:00",
+    "UpdateTime": "2021-11-23T17:46:30+02:00",
+    "LastAccessTime": "1970-01-01T02:00:00+02:00",
+    "Retention": 0,
+    "StorageDescriptor": {
+      "Columns": [
+        {
+          "Name": "id",
+          "Type": "bigint",
+          "Comment": ""
+        },
+        {
+          "Name": "key",
+          "Type": "string",
+          "Comment": ""
+        }
+      ],
+      "Location": "s3://my-bucket/my-repo-prefix/symlinks/example/main/my_table",
+      "InputFormat": "org.apache.hadoop.hive.ql.io.SymlinkTextInputFormat",
+      "OutputFormat": "org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat",
+      "Compressed": false,
+      "NumberOfBuckets": -1,
+      "SerdeInfo": {
+        "Name": "default",
+        "SerializationLibrary": "org.apache.hadoop.hive.serde2.lazy.LazySimpleSerDe",
+        "Parameters": {
+          "serialization.format": "1"
+        }
+      },
+      "StoredAsSubDirectories": false
+    },
+    "PartitionKeys": [
+      {
+        "Name": "year",
+        "Type": "int",
+        "Comment": ""
+      },
+      {
+        "Name": "month",
+        "Type": "int",
+        "Comment": ""
+      }
+    ],
+    "ViewOriginalText": "",
+    "ViewExpandedText": "",
+    "TableType": "EXTERNAL_TABLE",
+    "Parameters": {
+      "EXTERNAL": "TRUE",
+      "bucketing_version": "2",
+      "transient_lastDdlTime": "1637681750"
+    },
+    "CreatedBy": "arn:aws:iam::************:user/********",
+    "IsRegisteredWithLakeFormation": false,
+    "CatalogId": "*********"
+  }
+}
+
+ +

You can now safely use Athena to query my_table.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/aws_cli.html b/v1.46/integrations/aws_cli.html new file mode 100644 index 000000000..7ac653a5a --- /dev/null +++ b/v1.46/integrations/aws_cli.html @@ -0,0 +1,892 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +AWS CLI | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with AWS CLI + + +

+ +

lakeFS exposes an S3-compatible API, so you can use the AWS S3 CLI to interact with objects in your repositories.

+

+ + + Table of contents + + +

+ + +
    +
  1. Configuration
  2. +
  3. Path convention
  4. +
  5. Usage
  6. +
  7. Examples
      +
    1. List directory
    2. +
    3. Copy from lakeFS to lakeFS
    4. +
    5. Copy from lakeFS to a local path
    6. +
    7. Copy from a local path to lakeFS
    8. +
    9. Delete file
    10. +
    11. Delete directory
    12. +
    +
  8. +
  9. Adding an alias
  10. +
+

+ + + Configuration + + +

+ + +

You would like to configure an AWS profile for lakeFS.

+ +

To configure the lakeFS credentials, run:

+
aws configure --profile lakefs
+
+

You will be prompted to enter the AWS Access Key ID and the AWS Secret Access Key.

+ +

It should look like this:

+
aws configure --profile lakefs
+# output:  
+# AWS Access Key ID [None]: AKIAIOSFODNN7EXAMPLE    
+# AWS Secret Access Key [None]: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+# Default region name [None]: 
+# Default output format [None]:
+
+

+ + + Path convention + + +

+ + +

When accessing objects in S3, you will need to use the lakeFS path convention: + s3://[REPOSITORY]/[BRANCH]/PATH/TO/OBJECT

+

+ + + Usage + + +

+ + +

After configuring the credentials, this is what a command should look:

+
aws s3 --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  ls s3://example-repo/main/example-directory
+
+ +

You can use an alias to make it shorter and more convenient.

+

+ + + Examples + + +

+ +

+ + + List directory + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 ls s3://example-repo/main/example-directory
+
+

+ + + Copy from lakeFS to lakeFS + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 cp s3://example-repo/main/example-file-1 s3://example-repo/main/example-file-2
+
+

+ + + Copy from lakeFS to a local path + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 cp s3://example-repo/main/example-file-1 /path/to/local/file
+
+

+ + + Copy from a local path to lakeFS + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 cp /path/to/local/file s3://example-repo/main/example-file-1
+
+

+ + + Delete file + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 rm s3://example-repo/main/example-directory/example-file
+
+

+ + + Delete directory + + +

+ + +
aws --profile lakefs \
+  --endpoint-url https://lakefs.example.com \
+  s3 rm s3://example-repo/main/example-directory/ --recursive
+
+

+ + + Adding an alias + + +

+ + +

To make the command shorter and more convenient, you can create an alias:

+ +
alias awslfs='aws --endpoint https://lakefs.example.com --profile lakefs'
+
+ +

Now, the ls command using the alias will be as follows:

+
awslfs s3 ls s3://example-repo/main/example-directory
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/boto.html b/v1.46/integrations/boto.html new file mode 100644 index 000000000..7899a7d60 --- /dev/null +++ b/v1.46/integrations/boto.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/cloudera.html b/v1.46/integrations/cloudera.html new file mode 100644 index 000000000..c5c58a12b --- /dev/null +++ b/v1.46/integrations/cloudera.html @@ -0,0 +1,717 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Cloudera | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Cloudera Spark + + +

+ + +

Use the lakeFS Hadoop FileSystem to integrate lakeFS with Cloudera Spark.

+ +

Review Cloudera Partner Listing for the Cloudera certification of lakeFS integration with Cloudera Data Platform (CDP) and Cloudera Spark.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/databricks.html b/v1.46/integrations/databricks.html new file mode 100644 index 000000000..a9389e216 --- /dev/null +++ b/v1.46/integrations/databricks.html @@ -0,0 +1,899 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Databricks | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Databricks + + +

+ + + +

+ + + Overview + + +

+ + +

Databricks is a unified, open analytics platform for building, deploying, sharing, +and maintaining enterprise-grade data, analytics, and AI solutions at scale.

+ +

In this document, we will cover the various Databricks products and how they integrate +with lakeFS.

+ +

+ +

+

+ + + Databricks Compute Options + + +

+ + +

Databricks offers several compute options +for running workloads. All of these options can be used with lakeFS. +At a basic level, Databricks compute products are Spark clusters that run on top +of cloud infrastructure and offer different configuration options. +From a lakeFS integration perspective, the main difference is how you configure +the storage operations that perform read/writes to lakeFS.

+ +

lakeFS storage operations will either use the lakeFS Hadoop Filesystem, +which utilizes the lakeFS OpenAPI, +or the s3a Filesystem, which uses the lakeFS S3 Gateway. +In short, the lakeFS S3 Gateway is the fastest way to get started, but it routes +all traffic through the lakeFS server. +The lakeFS Hadoop Filesystem requires more setup, but all data transfers will occur +directly on the bucket. The lakeFS Hadoop Filesystem can write to storage +using the s3a filesystem or using pre-signed +URLs generated by the lakeFS server. To read more about the alternatives, see +the Spark integration page.

+

+ + + All-Purpose Compute + + +

+ + +

Provisioned compute used to analyze data in notebooks.

+ +

When you create a Databricks compute cluster, you can configure it to use lakeFS +with the lakeFS Hadoop Filesystem (see Databricks installation guide) +or the lakeFS S3 Gateway. The lakeFS S3 Gateway can be configured +in the notebook or during +cluster setup (Advanced Options -> Spark -> Spark config).

+

+ + + Jobs Compute + + +

+ + +

Provisioned compute used to run automated jobs. +The Databricks job scheduler automatically creates a job compute whenever a job is configured to run on new compute.

+ +

To use lakeFS with Databricks jobs, a compute cluster needs to be configured in the cluster setup, +just like with All-Purpose compute.

+ +

{.note} +Note +Serverless compute for Databricks jobs is currently not supported.

+

+ + + SQL Warehouses + + +

+ + +

Classic & Pro warehouses are used to run SQL commands on data objects in the SQL editor or interactive notebooks. +Serverless warehouses do the same, except that they are on-demand elastic compute.

+ +

All warehouses do not allow the installation of external jars, such as the lakeFS Hadoop Filesystem. +To use SQL warehouses with lakeFS, utilize the lakeFS S3 Gateway.

+

+ + + Unity Catalog + + +

+ + +

Unity Catalog is Databricks’ metastore that provides centralized access control, +auditing, lineage, and data discovery capabilities across Databricks workspaces.

+ +

lakeFS can be used with Unity Catalog to provide a versioned view of the data and +the Unity Catalog metadata.

+ +

lakeFS support for Unity Catalog differs between lakeFS OSS and lakeFS Enterprise & Cloud.

+

+ + + Catalog Exports lakeFS + + +

+ + +

Leveraging the external tables feature within Unity Catalog, +you can register a Delta Lake table exported from lakeFS and access it through the unified catalog.

+ +

{.note} +Note +lakeFS Catalog exporters offer read-only table exports.

+ +

Catalog Exports relies on lakeFS Actions and offers +a way to export changes from lakeFS to Unity Catalog.

+ +

For the full guide on how to use Catalog Exports with Unity Catalog, see the documentation.

+

+ + + lakeFS for Databricks Enterprise + + +

+ + +

lakeFS for Databricks provides a seamless integration between lakeFS and Unity Catalog. +Its primary benefits over the integration offered by lakeFS open-source are:

+
    +
  • Table Write support
  • +
  • Native Unity Catalog interaction: Instead of reading/writing from lakeFS path, +use SQL to directly access data stored in Unity catalog
  • +
  • Advanced Serverless warehouse support: lakeFS can work with anything serverless - SQL warehouse or Serverless notebooks.
  • +
+ +

For more information, visit the lakeFS for Databricks page.

+

+ + + Delta Lake + + +

+ + +

lakeFS supports Delta Lake tables, and it provides a versioned view of the data and the Delta Lake metadata. +Read the Delta Lake docs for more information.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/delta.html b/v1.46/integrations/delta.html new file mode 100644 index 000000000..bcdb10ce4 --- /dev/null +++ b/v1.46/integrations/delta.html @@ -0,0 +1,950 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Delta Lake | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Delta Lake + + +

+ + +

Delta Lake is an open-source storage framework designed to improve performance and provide transactional guarantees to data lake tables.

+ +

Because lakeFS is format-agnostic, you can save data in Delta format within a lakeFS repository and benefit from the advantages of both technologies. Specifically:

+ +
    +
  1. ACID operations can span across multiple Delta tables.
  2. +
  3. CI/CD hooks can validate Delta table contents, schema, or even referential integrity.
  4. +
  5. lakeFS supports zero-copy branching for quick experimentation with full isolation.
  6. +
+ + +

+ + + Delta Lake Tables from the lakeFS Perspective + + +

+ + +

lakeFS is a data versioning tool, functioning at the object level. This implies that, by default, lakeFS remains agnostic +to whether the objects within a Delta table location represent a table, table metadata, or data. As per the Delta Lake protocol, +any modification to a table—whether it involves adding data or altering table metadata—results in the creation of a new object +in the table’s transaction log. +Typically, residing under the _delta_log path, relative to the root of the table’s directory. This new object has an incremented version compared to its predecessor.

+ +

Consequently, when making changes to a Delta table within the lakeFS environment, these changes are reflected as changes +to objects within the table location. For instance, inserting a record into a table named “my-table,” which is partitioned +by ‘category’ and ‘country,’ is represented in lakeFS as added objects within the table prefix (i.e., the table data) and the table transaction log. +Record addition

+ +

Similarly, when performing a metadata operation such as renaming a table column, new objects are appended to the table transaction log, +indicating the schema change. +Record addition

+

+ + + Using Delta Lake with lakeFS from Apache Spark + + +

+ + +

Given the native integration between Delta Lake and Spark, it’s most common that you’ll interact with Delta tables in a Spark environment.

+ +

To configure a Spark environment to read from and write to a Delta table within a lakeFS repository, you need to set the proper credentials and endpoint in the S3 Hadoop configuration, like you’d do with any Spark environment.

+ +

Once set, you can interact with Delta tables using regular Spark path URIs. Make sure that you include the lakeFS repository and branch name:

+ +
df.write.format("delta").save("s3a://<repo-name>/<branch-name>/path/to/delta-table")
+
+ +

Note: If using the Databricks Analytics Platform, see the integration guide for configuring a Databricks cluster to use lakeFS.

+ +

To see the integration in action see this notebook in the lakeFS Samples Repository.

+

+ + + Using Delta Lake with lakeFS from Python + + +

+ + +

The delta-rs library provides bindings for Python. This means that you can use Delta Lake and lakeFS directly from Python without needing Spark. Integration is done through the lakeFS S3 Gateway

+ +

The documentation for the deltalake Python module details how to read, write, and query Delta Lake tables. To use it with lakeFS use an s3a path for the table based on your repository and branch (for example, s3a://delta-lake-demo/main/my_table/) and specify the following storage_options:

+ +
storage_options = {"AWS_ENDPOINT": <your lakeFS endpoint>,
+                   "AWS_ACCESS_KEY_ID": <your lakeFS access key>,
+                   "AWS_SECRET_ACCESS_KEY": <your lakeFS secret key>,
+                   "AWS_REGION": "us-east-1",
+                   "AWS_S3_ALLOW_UNSAFE_RENAME": "true"
+                  }
+
+ +

If your lakeFS is not using HTTPS (for example, you’re just running it locally) then add the option

+ +
                   "AWS_STORAGE_ALLOW_HTTP": "true"
+
+ +

To see the integration in action see this notebook in the lakeFS Samples Repository.

+

+ + + Exporting Delta Lake tables from lakeFS into Unity Catalog + + +

+ + +

This option is for users who are managing Delta Lake tables with lakeFS and access them through Databricks Unity Catalog. lakeFS offers +a Data Catalog Export functionality that provides read-only access to your Delta tables from within Unity catalog. Using the data catalog exporters,
+you can work on Delta tables in isolation and easily explore them within the Unity Catalog.

+ +

Once exported, you can query the versioned table data with:

+
SELECT * FROM my_catalog.main.my_delta_table
+
+

Here, ‘main’ is the name of the lakeFS branch from which the delta table was exported.

+ +

To enable Delta table exports to Unity catalog use the Unity catalog integration guide.

+ + +

+ + + Limitations + + +

+ +

+ + + Multi-Writer Support in lakeFS for Delta Lake Tables + + +

+ + +

lakeFS currently supports a single writer for Delta Lake tables. Attempting to utilize multiple writers for writing to a Delta table may result in two types of issues:

+
    +
  1. Merge Conflicts: These conflicts arise when multiple writers modify a Delta table on different branches, and an attempt is made to merge these branches. +Merge conflicts
  2. +
  3. Concurrent File Overwrite: This issue occurs when multiple writers concurrently modify a Delta table on the same branch. +Concurrent file overwrite
  4. +
+ +

Note: lakeFS currently lacks its own implementation for a LogStore, and the default Log store used does not control concurrency.

+

To address these limitations, consider following best practices for implementing multi-writer support.

+

+ + + Best Practices + + +

+ +

+ + + Implementing Multi-Writer Support through lakeFS Branches and Merges + + +

+ + +

To achieve safe multi-writes to a Delta Lake table on lakeFS, we recommend following these best practices:

+
    +
  1. Isolate Changes: Make modifications to your table in isolation. Each set of changes should be associated with a dedicated lakeFS branch, branching off from the main branch.
  2. +
  3. Merge Atomically: After making changes in isolation, try to merge them back into the main branch. This approach guarantees that the integration of changes is cohesive.
  4. +
+ +

The workflow involves:

+
    +
  • Creating a new lakeFS branch from the main branch for any table change.
  • +
  • Making modifications in isolation.
  • +
  • Attempting to merge the changes back into the main branch.
  • +
  • Iterating the process in case of a merge failure due to conflicts.
  • +
+ +

The diagram below provides a visual representation of how branches and merges can be utilized to manage concurrency effectively: +Multi writers workaround

+

+ + + Follow Vacuum by Garbage Collection + + +

+ + +

To delete unused files from a table directory while working with Delta Lake over lakeFS you need to first use Delta lake +Vacuum to soft-delete the files, and then use +lakeFS Garbage Collection to hard-delete them from the storage.

+ +

Note: lakeFS enables you to recover from undesired vacuum runs by reverting the changes done by a vacuum run before running Garbage Collection.

+

+ + + When running lakeFS inside your VPC (on AWS) + + +

+ + +

When lakeFS runs inside your private network, your Databricks cluster needs to be able to access it. +This can be done by setting up a VPC peering between the two VPCs +(the one where lakeFS runs and the one where Databricks runs). For this to work on Delta Lake tables, you would also have to disable multi-cluster writes with:

+ +
spark.databricks.delta.multiClusterWrites.enabled false
+
+

+ + + Using multi cluster writes (on AWS) + + +

+ + +

When using multi-cluster writes, Databricks overrides Delta’s S3-commit action. +The new action tries to contact lakeFS from servers on Databricks’ own AWS account, which of course won’t be able to access your private network. +So, if you must use multi-cluster writes, you’ll have to allow access from Databricks’ AWS account to lakeFS. +If you are trying to achieve that, please reach out on Slack and the community will try to assist.

+

+ + + Further Reading + + +

+ + +

See Guaranteeing Consistency in Your Delta Lake Tables With lakeFS post on the lakeFS blog to learn how to +guarantee data quality in a Delta table by utilizing lakeFS branches.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/distcp.html b/v1.46/integrations/distcp.html new file mode 100644 index 000000000..07c435e7e --- /dev/null +++ b/v1.46/integrations/distcp.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/dremio.html b/v1.46/integrations/dremio.html new file mode 100644 index 000000000..43187bf43 --- /dev/null +++ b/v1.46/integrations/dremio.html @@ -0,0 +1,738 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Dremio | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Dremio + + +

+ +

Dremio is a next-generation data lake engine that liberates your data with live, +interactive queries directly on cloud data lake storage, including S3 and lakeFS.

+

+ + + Configuration + + +

+ +

Starting from version 3.2.3, Dremio supports Minio as an experimental S3-compatible plugin. +Similarly, you can connect lakeFS with Dremio.

+ +

Suppose you already have both lakeFS and Dremio deployed, and want to use Dremio to query your data in the lakeFS repositories. +You can follow the steps listed below to configure on Dremio UI:

+ +
    +
  1. click Add Data Lake.
  2. +
  3. Under File Stores, choose Amazon S3.
  4. +
  5. Under Advanced Options, check Enable compatibility mode (experimental).
  6. +
  7. Under Advanced Options > Connection Properties, add fs.s3a.path.style.access and set the value to true.
  8. +
  9. Under Advanced Options > Connection Properties, add fs.s3a.endpoint and set lakeFS S3 endpoint to the value.
  10. +
  11. Under the General tab, specify the access_key_id and secret_access_key provided by lakeFS server.
  12. +
  13. Click Save, and now you should be able to browse lakeFS repositories on Dremio.
  14. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/duckdb.html b/v1.46/integrations/duckdb.html new file mode 100644 index 000000000..32883fd69 --- /dev/null +++ b/v1.46/integrations/duckdb.html @@ -0,0 +1,867 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +DuckDB | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with DuckDB + + +

+ + +

DuckDB is an in-process SQL OLAP database management system. You can access data in lakeFS from DuckDB, as well as use DuckDB from within the web interface of lakeFS

+ + +

+ + + Accessing lakeFS from DuckDB + + +

+ +

+ + + Configuration + + +

+ + +

Querying data in lakeFS from DuckDB is similar to querying data in S3 from DuckDB. It is done using the httpfs extension connecting to the S3 Gateway that lakeFS provides.

+ +

If not loaded already, install and load the HTTPFS extension:

+ +
INSTALL httpfs;
+LOAD httpfs;
+
+ +

Then run the following to configure the connection.

+ +
-- "s3_region" is the S3 region on which your bucket resides. If local storage, or not S3, then just set it to "us-east-1".
+SET s3_region='us-east-1';
+-- the host (and port, if necessary) of your lakeFS server
+SET s3_endpoint='lakefs.example.com';
+-- the access credentials for your lakeFS user
+SET s3_access_key_id='AKIAIOSFODNN7EXAMPLE'; 
+-- the access credentials for your lakeFS user
+SET s3_secret_access_key='wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY'; 
+SET s3_url_style='path';
+
+-- Uncomment in case the endpoint listen on non-secure, for example running lakeFS locally.
+-- SET s3_use_ssl=false;
+
+

+ + + Querying Data + + +

+ + +

Once configured, you can query data using the lakeFS S3 Gateway using the following URI pattern:

+ +
s3://<REPOSITORY NAME>/<REFERENCE ID>/<PATH TO DATA>
+
+ +

Since the S3 Gateway implemenets all S3 functionality required by DuckDB, you can query using globs and patterns, including support for Hive-partitioned data.

+ +

Example:

+ +
SELECT * 
+FROM parquet_scan('s3://example-repo/main/data/population/by-region/*.parquet', HIVE_PARTITIONING=1) 
+ORDER BY name;
+
+

+ + + Writing Data + + +

+ + +

No special configuration required for writing to a branch. Assuming the configuration above and write permissions to a dev branch, +a write operation would look like any DuckDB write:

+ +
CREATE TABLE sampled_population AS SELECT * 
+FROM parquet_scan('s3://example-repo/main/data/population/by-region/*.parquet', HIVE_PARTITIONING=1) 
+USING SAMPLE reservoir(50000 ROWS) REPEATABLE (100);
+
+COPY sampled_population TO 's3://example-repo/main/data/population/sample.parquet'; -- actual write happens here
+
+

+ + + Using DuckDB in Python with lakefs-spec + + +

+ + +

Python users can use DuckDB by leveraging the lakefs-spec package.

+ +

Note This library is a third-party package and not maintained by the lakeFS developers; please file issues and bug reports directly +in the lakefs-spec repository.

+ +

Using lakefs-spec, querying lakeFS could be done using pre-signed URLs, allowing for efficient and secure I/O, where the data files are read directly from the underlying object store.

+ +

+import duckdb
+from fsspec import filesystem
+
+duckdb.register_filesystem(filesystem('lakefs'))
+
+duckdb.sql("SELECT * FROM 'lakefs://example-repo/main/data/population/sample.parquet'")
+
+

+ + + Using DuckDB in the lakeFS web UI + + +

+ + +

The lakeFS web UI includes DuckDB in the Object viewer page.

+ +

DuckDB query editor on the lakeFS objects page

+ +

Using this you can query objects in lakeFS directly using a lakefs path:

+ +
lakefs://<repository>/<branch>/object/path/foo.parquet
+
+ +

The DuckDB query editor is provided by DuckDB WASM. It renders and provides querying capabilities for any objects of the following types:

+ +
    +
  • Parquet
  • +
  • CSV
  • +
  • TSV
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/emr.html b/v1.46/integrations/emr.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/integrations/emr.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/git.html b/v1.46/integrations/git.html new file mode 100644 index 000000000..d2bb88494 --- /dev/null +++ b/v1.46/integrations/git.html @@ -0,0 +1,755 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Git | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Integrating lakeFS with Git + + +

+ + +

Integrating code and data version control systems allows you to associate code versions with data versions, facilitating +the reproduction of complex environments with multiple components. Consequently, lakeFS, a data version control system, +seamlessly integrates with Git. This combination establishes a robust foundation for versioning both your code and data, +fostering a streamlined and reproducible development process.

+

+ + + Use Cases + + +

+ +

+ + + Develop Reproducible ML Models + + +

+ + +

Maintain a comprehensive record of both model code versions and input data versions to ensure the reproducibility of ML +model results.

+ +

The common way to develop reproducible ML models with lakeFS is to use the +lakectl local command. See Working with lakeFS Data Locally +to understand how to use lakectl local in conjunction with Git to develop reproducible ML models.

+

+ + + Develop Reproducible ETL Pipelines + + +

+ + +

Track code versions for each step in ETL pipelines along with the corresponding data versions of their inputs and outputs. +This approach allows straight forward troubleshooting and reproduction of data errors.

+ +

Check out this lakeFS sample +that demonstrates how Git, Airflow, and lakeFS can be integrated to result in reproducible ETL pipelines.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/glue_etl.html b/v1.46/integrations/glue_etl.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/integrations/glue_etl.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/glue_hive_metastore.html b/v1.46/integrations/glue_hive_metastore.html new file mode 100644 index 000000000..9ff8bd0ec --- /dev/null +++ b/v1.46/integrations/glue_hive_metastore.html @@ -0,0 +1,964 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Glue / Hive metastore | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with the Glue/Hive Metastore + + +

+ + +

Deprecated Feature: Having heard the feedback from the community, we are planning to replace the below manual steps with an automated process. +You can read more about it here.

+ + +

+ + + About Glue / Hive Metastore + + +

+ + +

This part explains about how Glue/Hive Metastore work with lakeFS.

+ +

Glue and Hive Metastore store metadata related to Hive and other services (such as Spark and Trino). +They contain metadata such as the location of the table, information about columns, partitions and many more.

+

+ + + Without lakeFS + + +

+ +

To query the table my_table, Spark will:

+
    +
  • Request the metadata from Hive metastore (steps 1,2),
  • +
  • Use the location from the metadata to access the data in S3 (steps 3,4). +metastore with S3
  • +
+

+ + + With lakeFS + + +

+ +

When using lakeFS, the flow stays exactly the same. Note that the location of the table my_table now contains the branch s3://example/main/path/to/table +metastore with S3

+

+ + + Managing Tables With lakeFS Branches + + +

+ +

+ + + Motivation + + +

+ +

When creating a table in Glue/Hive metastore (using a client such as Spark, Hive, Presto), we specify the table location. +Consider the table my_table that was created with the location s3://example/main/path/to/table.

+ +

Suppose you created a new branch called DEV with main as the source branch. +The data from s3://example/main/path/to/table is now accessible in s3://example/DEV/path/to/table. +The metadata is not managed in lakeFS, meaning you don’t have any table pointing to s3://example/DEV/path/to/table.

+ +

To address this, lakeFS introduces lakectl metastore commands. The case above can be handled using the copy command: it creates a copy of my_table with data located in s3://example/DEV/path/to/table. Note that this is a fast, metadata-only operation.

+

+ + + Configurations + + +

+ +

The lakectl metastore commands can run on Glue or Hive metastore. +Add the following to the lakectl configuration file (by default ~/.lakectl.yaml):

+

+ + + Hive + + +

+ + +
metastore:
+  type: hive
+  hive:
+    uri: hive-metastore:9083
+
+

+ + + Glue + + +

+ + +
metastore:
+  type: glue
+  glue:
+    catalog_id: 123456789012
+    region: us-east-1
+    profile: default # optional, implies using a credentials file
+    credentials:
+      access_key_id: AKIAIOSFODNN7EXAMPLE
+      secret_access_key: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+
+ +

Note: It’s recommended to set type and catalog-id/metastore-uri in the lakectl configuration file.

+

+ + + Suggested Model + + +

+ + +

For simplicity, we recommend creating a schema for each branch. That way, you can use the same table name across different schemas.

+ +

For example: +after creating branch example_branch, also create a schema named example_branch. +For a table named my_table under the schema main, create a new table under the same name and under the schema example_branch. You now have two my_table, one in the main schema and one in the branch schema.

+

+ + + Commands + + +

+ + +

Metastore tools support three commands: copy, diff, and create-symlink. +copy and diff can work both on Glue and on Hive. +create-symlink works only on Glue.

+ +

Note: If to-schema or to-table are not specified, the destination branch and source table names will be used as per the suggested model.

+ +

Note: Metastore commands can only run on tables located in lakeFS. You should not use tables that aren’t located in lakeFS.

+

+ + + Copy + + +

+ + +

The copy command creates a copy of a table pointing to the defined branch. +In case the destination table already exists, the command will only merge the changes.

+ +

Example:

+ +

Suppose we created the table inventory on branch main on schema default.

+
CREATE EXTERNAL TABLE `inventory`(
+        `inv_item_sk` int,
+        `inv_warehouse_sk` int,
+        `inv_quantity_on_hand` int)
+    PARTITIONED BY (
+        `inv_date_sk` int) STORED AS ORC
+    LOCATION
+        's3a://my_repo/main/path/to/table';
+
+ +

We create a new lakeFS branch example_branch:

+ +
lakectl branch create lakefs://my_repo/example_branch --source lakefs://my_repo/main
+
+ +

The data from s3://my_repo/main/path/to/table is now accessible in s3://my_repo/DEV/path/to/table. +To query the data in s3://my_repo/DEV/path/to/table, you would like to create a copy of the table inventory in schema example_branch pointing to the new branch.

+ +
lakectl metastore copy --from-schema default --from-table inventory --to-schema example_branch --to-table inventory --to-branch example_branch
+
+ +

After running this command, query the table example_branch.inventory to get the data from s3://my_repo/DEV/path/to/table

+

+ + + Copy Partition + + +

+ + +

After adding a partition to the branch table, you may want to copy the partition to the main table. +For example, for the new partition 2020-08-01, run the following to copy the partition to the main table:

+ +
lakectl metastore copy --type hive --from-schema example_branch --from-table inventory --to-schema default --to-table inventory --to-branch main -p 2020-08-01
+
+ +

For a table partitioned by more than one column, specify the partition flag for every column. For example, for the partition (year='2020',month='08',day='01'):

+ +
lakectl metastore copy --from-schema example_branch --from-table branch_inventory --to-schema default --to-branch main -p 2020 -p 08 -p 01
+
+

+ + + Diff + + +

+ + +

Provides a two-way diff between two tables. +Shows added+ , removed- and changed~ partitions and columns.

+ +

Example:

+ +

Suppose you made some changes on the copied table inventory on schema example_branch and now want to view the changes before merging back to inventory on schema default.

+ +

Hive:

+
lakectl metastore diff --type hive --address thrift://hive-metastore:9083 --from-schema example_branch --from-table branch --to-schema default --to-table inventory
+
+ +

The output will look like this:

+ +
Columns are identical
+Partitions
+- 2020-07-04
++ 2020-07-05
++ 2020-07-06
+~ 2020-07-08
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/glue_metastore.html b/v1.46/integrations/glue_metastore.html new file mode 100644 index 000000000..092c3d9fe --- /dev/null +++ b/v1.46/integrations/glue_metastore.html @@ -0,0 +1,1077 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Glue Data Catalog | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with the Glue Catalog + + +

+ + + +

+ + + Overview + + +

+ + +

The integration between Glue and lakeFS is based on Data Catalog Exports.

+ +

This guide describes how to use lakeFS with the Glue Data Catalog. +You’ll be able to query your lakeFS data by specifying the repository, branch and commit in your SQL query. +Currently, only read operations are supported on the tables. +You will set up the automation required to work with lakeFS on top of the Glue Data Catalog, including:

+
    +
  1. Create a table descriptor under _lakefs_tables/<your-table>.yaml. This will represent your table schema.
  2. +
  3. Write an exporter script that will: +
      +
    • Mirror your branch’s state into Hive Symlink files readable by Athena.
    • +
    • Export the table descriptors from your branch to the Glue Catalog.
    • +
    +
  4. +
  5. Set up lakeFS hooks to trigger the above script when specific events occur.
  6. +
+

+ + + Example: Using Athena to query lakeFS data + + +

+ +

+ + + Prerequisites + + +

+ + +

Before starting, make sure you have:

+
    +
  1. An active lakeFS installation with S3 as the backing storage, and a repository in this installation.
  2. +
  3. A database in Glue Data Catalog (lakeFS does not create one).
  4. +
  5. AWS Credentials with permission to manage Glue, Athena Query and S3 access.
  6. +
+

+ + + Add table descriptor + + +

+ + +

Let’s define a table, and commit it to lakeFS. +Save the YAML below as animals.yaml and upload it to lakeFS.

+ +
lakectl fs upload lakefs://catalogs/main/_lakefs_tables/animals.yaml -s ./animals.yaml && \
+lakectl commit lakefs://catalogs/main -m "added table"
+
+ +
name: animals
+type: hive
+# data location root in lakeFS
+path: tables/animals
+# partitions order
+partition_columns: ['type', 'weight']
+schema:
+  type: struct
+  # all the columns spec
+  fields:
+    - name: type
+      type: string
+      nullable: true
+      metadata:
+        comment: axolotl, cat, dog, fish etc
+    - name: weight
+      type: integer
+      nullable: false
+      metadata: {}
+    - name: name
+      type: string
+      nullable: false
+      metadata: {}
+
+

+ + + Write some table data + + +

+ + +

Insert data into the table path, using your preferred method (e.g. Spark), and commit upon completion. +This example uses CSV files, and the files added to lakeFS should look like this:

+ +

lakeFS Uploaded CSV Files

+

+ + + The exporter script + + +

+ + +

Upload the following script to your main branch under scripts/animals_exporter.lua (or a path of your choice).

+ +
+

For code references check symlink_exporter and glue_exporter docs.

+
+ +
local aws = require("aws")
+local symlink_exporter = require("lakefs/catalogexport/symlink_exporter")
+local glue_exporter = require("lakefs/catalogexport/glue_exporter")
+-- settings 
+local access_key = args.aws.aws_access_key_id
+local secret_key = args.aws.aws_secret_access_key
+local region = args.aws.aws_region
+local table_path = args.table_source -- table descriptor 
+local db = args.catalog.db_name -- glue db
+local table_input = args.catalog.table_input -- table input (AWS input spec) for Glue
+-- export symlinks 
+local s3 = aws.s3_client(access_key, secret_key, region)
+local result = symlink_exporter.export_s3(s3, table_path, action, {debug=true})
+-- register glue table
+local glue = aws.glue_client(access_key, secret_key, region)
+local res = glue_exporter.export_glue(glue, db, table_path, table_input, action, {debug=true})
+
+

+ + + Configure Action Hooks + + +

+ + +

Hooks serve as the mechanism that triggers the execution of the exporter. +For more detailed information on how to configure exporter hooks, you can refer to Running an Exporter.

+ +
+

The args.catalog.table_input argument in the Lua script is assumed to be passed from the action arguments, that way the same script can be reused for different tables. Check the example to construct the table input in the lua code.

+
+ +
+ +
+

+ + + Single hook with CSV Table + + +

+ + +

Upload to _lakefs_actions/animals_glue.yaml:

+ +
name: Glue Exporter
+on:
+  post-commit:
+    branches: ["main"]
+hooks:
+  - id: animals_table_glue_exporter
+    type: lua
+    properties:
+      script_path: "scripts/animals_exporter.lua"
+      args:
+        aws:
+          aws_access_key_id: "<AWS_ACCESS_KEY_ID>"
+          aws_secret_access_key: "<AWS_SECRET_ACCESS_KEY>"
+          aws_region: "<AWS_REGION>"
+        table_source: '_lakefs_tables/animals.yaml'
+        catalog:
+          db_name: "my-glue-db"
+          table_input:
+            StorageDescriptor: 
+              InputFormat: "org.apache.hadoop.hive.ql.io.SymlinkTextInputFormat"
+              OutputFormat: "org.apache.hadoop.hive.ql.io.HiveIgnoreKeyTextOutputFormat"
+              SerdeInfo:
+                SerializationLibrary: "org.apache.hadoop.hive.serde2.OpenCSVSerde"
+                Parameters:
+                  separatorChar: ","
+            Parameters: 
+              classification: "csv"
+              "skip.header.line.count": "1"
+
+ +
+
+

+ + + Spark Parquet Example + + +

+ + +

When working with Parquet files, upload the following to _lakefs_actions/animals_glue.yaml:

+ +
name: Glue Exporter
+on:
+   post-commit:
+      branches: ["main"]
+hooks:
+   - id: animals_table_glue_exporter
+     type: lua
+     properties:
+        script_path: "scripts/animals_exporter.lua"
+        args:
+           aws:
+              aws_access_key_id: "<AWS_ACCESS_KEY_ID>"
+              aws_secret_access_key: "<AWS_SECRET_ACCESS_KEY>"
+              aws_region: "<AWS_REGION>"
+           table_source: '_lakefs_tables/animals.yaml'
+           catalog:
+              db_name: "my-glue-db"
+              table_input:
+                 StorageDescriptor:
+                   InputFormat: "org.apache.hadoop.hive.ql.io.SymlinkTextInputFormat"
+                   OutputFormat: "org.apache.hadoop.hive.ql.io.IgnoreKeyTextOutputFormat"
+                   SerdeInfo:
+                       SerializationLibrary: "org.apache.hadoop.hive.ql.io.parquet.serde.ParquetHiveSerDe"
+                 Parameters:
+                   classification: "parquet"
+                   EXTERNAL: "TRUE"
+                   "parquet.compression": "SNAPPY"
+
+ +
+
+

+ + + Multiple Hooks / Inline script + + +

+ + +

The following example demonstrates how to separate the symlink and glue exporter into building blocks running in separate hooks. +It also shows how to run the lua script inline instead of a file, depending on user preference.

+ +
name: Animal Table Exporter
+on:
+  post-commit:
+    branches: ["main"]
+hooks:
+  - id: symlink_exporter
+    type: lua
+    properties:
+      args:
+        aws:
+          aws_access_key_id: "<AWS_ACCESS_KEY_ID>"
+          aws_secret_access_key: "<AWS_SECRET_ACCESS_KEY>"
+          aws_region: "<AWS_REGION>"
+        table_source: '_lakefs_tables/animals.yaml'
+      script: |
+        local exporter = require("lakefs/catalogexport/symlink_exporter")
+        local aws = require("aws")
+        local table_path = args.table_source
+        local s3 = aws.s3_client(args.aws.aws_access_key_id, args.aws.aws_secret_access_key, args.aws.aws_region)
+        exporter.export_s3(s3, table_path, action, {debug=true})
+  - id: glue_exporter
+    type: lua
+    properties:
+      args:
+        aws:
+          aws_access_key_id: "<AWS_ACCESS_KEY_ID>"
+          aws_secret_access_key: "<AWS_SECRET_ACCESS_KEY>"
+          aws_region: "<AWS_REGION>"
+        table_source: '_lakefs_tables/animals.yaml'
+        catalog:
+          db_name: "my-glue-db"
+          table_input: # add glue table input here 
+      script: |
+        local aws = require("aws")
+        local exporter = require("lakefs/catalogexport/glue_exporter")
+        local glue = aws.glue_client(args.aws.aws_access_key_id, args.aws.aws_secret_access_key, args.aws.aws_region)
+        exporter.export_glue(glue, args.catalog.db_name, args.table_source, args.catalog.table_input, action, {debug=true})  
+
+ +
+
+ +

Adding the script and the action files to the repository and commit it. This is a post-commit action, meaning it will be executed after the commit operation has taken place.

+ +
lakectl fs upload lakefs://catalogs/main/scripts/animals_exporter.lua -s ./animals_exporter.lua
+lakectl fs upload lakefs://catalogs/main/_lakefs_actions/animals_glue.yaml -s ./animals_glue.yaml
+lakectl commit lakefs://catalogs/main -m "trigger first export hook"
+
+ +

Once the action has completed its execution, you can review the results in the action logs.

+ +

Hooks log result in lakeFS UI

+

+ + + Use Athena + + +

+ + +

We can use the exported Glue table with any tool that supports Glue Catalog (or Hive compatible) such as Athena, Trino, Spark and others. +To use Athena we can simply run MSCK REPAIR TABLE and then query the tables.

+ +

In Athena, make sure that the correct database (my-glue-db in the example above) is configured, then run:

+ +
MSCK REPAIR TABLE `animals_catalogs_main_9255e5`; -- load partitions for the first time 
+SELECT * FROM `animals_catalogs_main_9255e5` limit 50;
+
+ +

Athena SQL Result

+

+ + + Cleanup + + +

+ + +

Users can use additional hooks / actions to implement a custom cleanup logic to delete the symlink in S3 and Glue Tables.

+ +
glue.delete_table(db, '<glue table name>')
+s3.delete_recursive('bucket', 'path/to/symlinks/of/a/commit/')
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/hive.html b/v1.46/integrations/hive.html new file mode 100644 index 000000000..bc22397d0 --- /dev/null +++ b/v1.46/integrations/hive.html @@ -0,0 +1,817 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Apache Hive | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Apache Hive + + +

+ + +

The Apache Hive ™ data warehouse software facilitates reading, writing, and managing large datasets residing in distributed storage using SQL. Structure can be projected onto data already in storage. A command line tool and JDBC driver are provided to connect users to Hive.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Configuration
  2. +
  3. Examples
  4. +
+ +
+

+ + + Configuration + + +

+ +

To configure Hive to work with lakeFS, you need to set the lakeFS credentials in the corresponding S3 credential fields.

+ +

lakeFS endpoint: fs.s3a.endpoint

+ +

lakeFS access key: fs.s3a.access.key

+ +

lakeFS secret key: fs.s3a.secret.key

+ +

Note +In the following examples, we set AWS credentials at runtime for clarity. In production, these properties should be set using one of Hadoop’s standard ways of Authenticating with S3.

+ +

For example, you can add the configurations to the file hdfs-site.xml:

+
<configuration>
+    ...
+    <property>
+        <name>fs.s3a.secret.key</name>
+        <value>wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.s3a.access.key</name>
+        <value>AKIAIOSFODNN7EXAMPLE</value>
+    </property>
+    <property>
+        <name>fs.s3a.endpoint</name>
+        <value>https://lakefs.example.com</value>
+    </property>
+    <property>
+       <name>fs.s3a.path.style.access</name>
+       <value>true</value>
+    </property>
+</configuration>
+
+ +

Note +In this example, we set fs.s3a.path.style.access to true to remove the need for additional DNS records for virtual hosting +fs.s3a.path.style.access that was introduced in Hadoop 2.8.0

+

+ + + Examples + + +

+ +

+ + + Example with schema + + +

+ + +
CREATE  SCHEMA example LOCATION 's3a://example/main/' ;
+CREATE TABLE example.request_logs (
+    request_time timestamp,
+    url string,
+    ip string,
+    user_agent string
+);
+
+

+ + + Example with an external table + + +

+ + +
CREATE EXTERNAL TABLE request_logs (
+    request_time timestamp,
+    url string,
+    ip string,
+    user_agent string
+) LOCATION 's3a://example/main/request_logs' ;
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/huggingface_datasets.html b/v1.46/integrations/huggingface_datasets.html new file mode 100644 index 000000000..7d22ded87 --- /dev/null +++ b/v1.46/integrations/huggingface_datasets.html @@ -0,0 +1,811 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +HuggingFace Datasets | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Versioning HuggingFace Datasets with lakeFS + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Installation
  2. +
  3. Configuration
  4. +
  5. Reading a Dataset
  6. +
  7. Saving/Loading
  8. +
+ +
+ +

HuggingFace 🤗 Datasets is a library for easily accessing and sharing datasets for Audio, Computer Vision, and Natural Language Processing (NLP) tasks.

+ +

🤗 Datasets supports access to cloud storage providers through fsspec FileSystem implementations.

+ +

lakefs-spec is a community implementation of an fsspec Filesystem that fully leverages lakeFS’ capabilities. Let’s start by installing it:

+

+ + + Installation + + +

+ + +
pip install lakefs-spec
+
+

+ + + Configuration + + +

+ + +

If you’ve already configured the lakeFS python SDK and/or lakectl, you should have a $HOME/.lakectl.yaml file that contains your access credentials and endpoint for your lakeFS environment.

+ +

Otherwise, install lakectl and run lakectl config to set up your access credentials.

+

+ + + Reading a Dataset + + +

+ + +

To read a dataset, all we have to do is use a lakefs://... URI when calling load_dataset:

+ +
>>> from datasets import load_dataset
+>>> 
+>>> dataset = load_dataset('csv', data_files='lakefs://example-repository/my-branch/data/example.csv')
+
+ +

That’s it! this should automatically load the lakefs-spec implementation that we’ve installed, which will use the $HOME/.lakectl.yaml file to read its credentials, so we don’t need to pass additional configuration.

+

+ + + Saving/Loading + + +

+ + +

Once we’ve loaded a Dataset, we can save it using the save_to_disk method as normal:

+ +
>>> dataset.save_to_disk('lakefs://example-repository/my-branch/datasets/example/')
+
+ +

At this point, we might want to commit that change to lakeFS, and tag it, so we could share it with our colleagues.

+ +

We can do it through the UI or lakectl, but let’s do it with the lakeFS Python SDK:

+ +
>>> import lakefs
+>>>
+>>> repo = lakefs.repository('example-repository')
+>>> commit = repo.branch('my-branch').commit(
+...     'saved my first huggingface Dataset!',
+...     metadata={'using': '🤗'})
+>>> repo.tag('alice_experiment1').create(commit)
+
+ +

Now, others on our team can load our exact dataset by using the tag we created:

+ +
>>> from datasets import load_from_disk
+>>>
+>>> dataset = load_from_disk('lakefs://example-repository/alice_experiment1/datasets/example/')
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/iceberg.html b/v1.46/integrations/iceberg.html new file mode 100644 index 000000000..cd570e6f3 --- /dev/null +++ b/v1.46/integrations/iceberg.html @@ -0,0 +1,978 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Apache Iceberg | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Apache Iceberg + + +

+ + + + +

To enrich your Iceberg tables with lakeFS capabilities, you can use the lakeFS implementation of the Iceberg catalog. +You will then be able to query your Iceberg tables using lakeFS references, such as branches, tags, and commit hashes:

+ +
SELECT * FROM catalog.ref.db.table
+
+

+ + + Setup + + +

+ + +
+ +
+ +

Use the following Maven dependency to install the lakeFS custom catalog:

+ +
<dependency>
+  <groupId>io.lakefs</groupId>
+  <artifactId>lakefs-iceberg</artifactId>
+  <version>0.1.4</version>
+</dependency>
+
+ +
+
+

Include the lakefs-iceberg jar in your package list along with Iceberg. For example:

+ +
.config("spark.jars.packages", "org.apache.iceberg:iceberg-spark-runtime-3.3_2.12:1.3.0,io.lakefs:lakefs-iceberg:0.1.4")
+
+
+
+

+ + + Configure + + +

+ + +
+ +
+ +

Set up the Spark SQL catalog:

+
.config("spark.sql.catalog.lakefs", "org.apache.iceberg.spark.SparkCatalog") \
+.config("spark.sql.catalog.lakefs.catalog-impl", "io.lakefs.iceberg.LakeFSCatalog") \
+.config("spark.sql.catalog.lakefs.warehouse", f"lakefs://{repo_name}") \ 
+.config("spark.sql.catalog.lakefs.cache-enabled", "false")
+
+ +

Configure the S3A Hadoop FileSystem with your lakeFS connection details. +Note that these are your lakeFS endpoint and credentials, not your S3 ones.

+ +
.config("spark.hadoop.fs.s3.impl", "org.apache.hadoop.fs.s3a.S3AFileSystem") \
+.config("spark.hadoop.fs.s3a.endpoint", "https://example-org.us-east-1.lakefscloud.io") \
+.config("spark.hadoop.fs.s3a.access.key", "AKIAIO5FODNN7EXAMPLE") \
+.config("spark.hadoop.fs.s3a.secret.key", "wJalrXUtnFEMI/K3MDENG/bPxRfiCYEXAMPLEKEY") \
+.config("spark.hadoop.fs.s3a.path.style.access", "true")
+
+ +
+ +
+
spark-shell --conf spark.sql.catalog.lakefs="org.apache.iceberg.spark.SparkCatalog" \
+   --conf spark.sql.catalog.lakefs.catalog-impl="io.lakefs.iceberg.LakeFSCatalog" \
+   --conf spark.sql.catalog.lakefs.warehouse="lakefs://example-repo" \
+   --conf spark.sql.catalog.lakefs.cache-enabled="false" \
+   --conf spark.hadoop.fs.s3.impl="org.apache.hadoop.fs.s3a.S3AFileSystem" \
+   --conf spark.hadoop.fs.s3a.endpoint="https://example-org.us-east-1.lakefscloud.io" \
+   --conf spark.hadoop.fs.s3a.access.key="AKIAIO5FODNN7EXAMPLE" \
+   --conf spark.hadoop.fs.s3a.secret.key="wJalrXUtnFEMI/K3MDENG/bPxRfiCYEXAMPLEKEY" \
+   --conf spark.hadoop.fs.s3a.path.style.access="true"
+
+
+
+

+ + + Using Iceberg tables with lakeFS + + +

+ +

+ + + Create a table + + +

+ + +

To create a table on your main branch, use the following syntax:

+ +
CREATE TABLE lakefs.main.db1.table1 (id int, data string);
+
+

+ + + Insert data into the table + + +

+ + +
INSERT INTO lakefs.main.db1.table1 VALUES (1, 'data1');
+INSERT INTO lakefs.main.db1.table1 VALUES (2, 'data2');
+
+

+ + + Create a branch + + +

+ + +

We can now commit the creation of the table to the main branch:

+ +
lakectl commit lakefs://example-repo/main -m "my first iceberg commit"
+
+ +

Then, create a branch:

+ +
lakectl branch create lakefs://example-repo/dev -s lakefs://example-repo/main
+
+

+ + + Make changes on the branch + + +

+ + +

We can now make changes on the branch:

+ +
INSERT INTO lakefs.dev.db1.table1 VALUES (3, 'data3');
+
+

+ + + Query the table + + +

+ + +

If we query the table on the branch, we will see the data we inserted:

+ +
SELECT * FROM lakefs.dev.db1.table1;
+
+ +

Results in:

+
+----+------+
+| id | data |
++----+------+
+| 1  | data1|
+| 2  | data2|
+| 3  | data3|
++----+------+
+
+ +

However, if we query the table on the main branch, we will not see the new changes:

+ +
SELECT * FROM lakefs.main.db1.table1;
+
+ +

Results in:

+
+----+------+
+| id | data |
++----+------+
+| 1  | data1|
+| 2  | data2|
++----+------+
+
+

+ + + Migrating an existing Iceberg Table to lakeFS Catalog + + +

+ + +

This is done through an incremental copy from the original table into lakeFS.

+ +
    +
  1. Create a new lakeFS repository lakectl repo create lakefs://example-repo <base storage path>
  2. +
  3. +

    Initiate a spark session that can interact with the source iceberg table and the target lakeFS catalog.

    + +

    Here’s an example of Hadoop and S3 session and lakeFS catalog with per-bucket config:

    + +
     SparkConf conf = new SparkConf();
    + conf.set("spark.hadoop.fs.s3a.path.style.access", "true");
    +
    + // set hadoop on S3 config (source tables we want to copy) for spark
    + conf.set("spark.sql.catalog.hadoop_prod", "org.apache.iceberg.spark.SparkCatalog");
    + conf.set("spark.sql.catalog.hadoop_prod.type", "hadoop");
    + conf.set("spark.sql.catalog.hadoop_prod.warehouse", "s3a://my-bucket/warehouse/hadoop/");
    + conf.set("spark.sql.extensions", "org.apache.iceberg.spark.extensions.IcebergSparkSessionExtensions");
    + conf.set("spark.hadoop.fs.s3a.bucket.my-bucket.access.key", "<AWS_ACCESS_KEY>");
    + conf.set("spark.hadoop.fs.s3a.bucket.my-bucket.secret.key", "<AWS_SECRET_KEY>");
    +
    + // set lakeFS config (target catalog and repository)
    + conf.set("spark.sql.catalog.lakefs", "org.apache.iceberg.spark.SparkCatalog");
    + conf.set("spark.sql.catalog.lakefs.catalog-impl", "io.lakefs.iceberg.LakeFSCatalog");
    + conf.set("spark.sql.catalog.lakefs.warehouse", "lakefs://example-repo");
    + conf.set("spark.sql.extensions", "org.apache.iceberg.spark.extensions.IcebergSparkSessionExtensions");
    + conf.set("spark.hadoop.fs.s3a.bucket.example-repo.access.key", "<LAKEFS_ACCESS_KEY>");
    + conf.set("spark.hadoop.fs.s3a.bucket.example-repo.secret.key", "<LAKEFS_SECRET_KEY>");
    + conf.set("spark.hadoop.fs.s3a.bucket.example-repo.endpoint"  , "<LAKEFS_ENDPOINT>");
    +
    +
  4. +
  5. +

    Create Schema in lakeFS and copy the data

    + +

    Example of copy with spark-sql:

    + +
     -- Create Iceberg Schema in lakeFS
    + CREATE SCHEMA IF NOT EXISTS <lakefs-catalog>.<branch>.<db>
    + -- Create new iceberg table in lakeFS from the source table (pre-lakeFS)
    + CREATE TABLE IF NOT EXISTS <lakefs-catalog>.<branch>.<db> USING iceberg AS SELECT * FROM <iceberg-original-table>
    +
    +
  6. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/index.html b/v1.46/integrations/index.html new file mode 100644 index 000000000..4d5954086 --- /dev/null +++ b/v1.46/integrations/index.html @@ -0,0 +1,749 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Integrations | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + Integrations with lakeFS + + +

+ + +

You can use lakeFS with a wide range of tools and frameworks.

+ +

lakeFS provides several clients directly, as well as an S3-compatible gateway. This gateway means that if you want to use something with lakeFS, so long as that technology can interface with S3, it can interface with lakeFS.

+ +

See below for detailed instructions for using different technologies with lakeFS.

+ + +

If there is a technology not listed here that you would like to use with lakeFS, please drop by our Slack and we’ll help you get started with it.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/kafka.html b/v1.46/integrations/kafka.html new file mode 100644 index 000000000..583f829a0 --- /dev/null +++ b/v1.46/integrations/kafka.html @@ -0,0 +1,733 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Apache Kafka | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Apache Kafka + + +

+ + +

Apache Kafka provides a unified, high-throughput, low-latency platform for handling real-time data feeds.

+ +

Different distributions of Kafka offer different methods for exporting data to S3 called Kafka Sink Connectors.

+ +

The most commonly used Connector for S3 is Confluent’s S3 Sink Connector.

+ +

Add the following to connector.properties file for lakeFS support:

+ +
# Your lakeFS repository
+s3.bucket.name=example-repo
+
+# Your lakeFS S3 endpoint and credentials
+store.url=https://lakefs.example.com
+aws.access.key.id=AKIAIOSFODNN7EXAMPLE
+aws.secret.access.key=wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+
+# main being the branch we want to write to
+topics.dir=main/topics 
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/kakfa.html b/v1.46/integrations/kakfa.html new file mode 100644 index 000000000..7e1bfa8ee --- /dev/null +++ b/v1.46/integrations/kakfa.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/kubeflow.html b/v1.46/integrations/kubeflow.html new file mode 100644 index 000000000..09c465342 --- /dev/null +++ b/v1.46/integrations/kubeflow.html @@ -0,0 +1,889 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Kubeflow | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Kubeflow pipelines + + +

+ +

Kubeflow is a project dedicated to making deployments of ML workflows on Kubernetes simple, portable, and scalable. +A Kubeflow pipeline is a portable and scalable definition of an ML workflow composed of steps. Each step in the pipeline is an instance of a component represented as an instance of ContainerOp.

+ + +

+ + + Add pipeline steps for lakeFS operations + + +

+ + +

To integrate lakeFS into your Kubeflow pipeline, you need to create Kubeflow components that perform lakeFS operations. +Currently, there are two methods to create lakeFS ContainerOps:

+
    +
  1. Implement a function-based ContainerOp that uses the lakeFS Python API to invoke lakeFS operations.
  2. +
  3. Implement a ContainerOp that uses the lakectl CLI docker image to invoke lakeFS operations.
  4. +
+

+ + + Function-based ContainerOps + + +

+ + +

To implement a function-based component that invokes lakeFS operations, +you should use the Python OpenAPI client lakeFS provides. See the example below that demonstrates how to make the client’s package available to your ContainerOp.

+

+ + + Example operations + + +

+ + +

Create a new branch: A function-based ContainerOp that creates a branch called example-branch based on the main branch of example-repo.

+ +
from kfp import components
+
+def create_branch(repo_name, branch_name, source_branch):
+   import lakefs_client
+   from lakefs_client import models
+   from lakefs_client.client import LakeFSClient
+
+   # lakeFS credentials and endpoint
+   configuration = lakefs_client.Configuration()
+   configuration.username = 'AKIAIOSFODNN7EXAMPLE'
+   configuration.password = 'wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY'
+   configuration.host = 'https://lakefs.example.com'
+   client = LakeFSClient(configuration)
+
+   client.branches.create_branch(repository=repo_name, branch_creation=models.BranchCreation(name=branch_name, source=source_branch))
+
+# Convert the function to a lakeFS pipeline step.
+create_branch_op = components.func_to_container_op(
+   func=create_branch,
+   packages_to_install=['lakefs_client==<lakeFS version>']) # Type in the lakeFS version you are using
+
+ +

You can invoke any lakeFS operation supported by lakeFS OpenAPI. For example, you could implement a commit and merge function-based ContainerOps. +Check out the full API reference.

+

+ + + Non-function-based ContainerOps + + +

+ + +

To implement a non-function based ContainerOp, you should use the treeverse/lakectl docker image. +With this image, you can run lakectl commands to execute the desired lakeFS operation.

+ +

For lakectl to work with Kubeflow, you will need to pass your lakeFS configurations as environment variables named:

+ +
    +
  • LAKECTL_CREDENTIALS_ACCESS_KEY_ID: AKIAIOSFODNN7EXAMPLE
  • +
  • LAKECTL_SECRET_ACCESS_KEY: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
  • +
  • LAKECTL_SERVER_ENDPOINT_URL: https://lakefs.example.com
  • +
+

+ + + Example operations + + +

+ + +
    +
  1. +

    Commit changes to a branch: A ContainerOp that commits uncommitted changes to example-branch on example-repo.

    + +
    from kubernetes.client.models import V1EnvVar
    +
    +def commit_op():
    +   return dsl.ContainerOp(
    +   name='commit',
    +   image='treeverse/lakectl',
    +   arguments=['commit', 'lakefs://example-repo/example-branch', '-m', 'commit message']).add_env_variable(V1EnvVar(name='LAKECTL_CREDENTIALS_ACCESS_KEY_ID',value='AKIAIOSFODNN7EXAMPLE')).add_env_variable(V1EnvVar(name='LAKECTL_CREDENTIALS_SECRET_ACCESS_KEY',value='wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY')).add_env_variable(V1EnvVar(name='LAKECTL_SERVER_ENDPOINT_URL',value='https://lakefs.example.com'))
    +
    +
  2. +
  3. +

    Merge two lakeFS branches: A ContainerOp that merges example-branch into the main branch of example-repo.

    + +
    def merge_op():
    +  return dsl.ContainerOp(
    +  name='merge',
    +  image='treeverse/lakectl',
    +  arguments=['merge', 'lakefs://example-repo/example-branch', 'lakefs://example-repo/main']).add_env_variable(V1EnvVar(name='LAKECTL_CREDENTIALS_ACCESS_KEY_ID',value='AKIAIOSFODNN7EXAMPLE')).add_env_variable(V1EnvVar(name='LAKECTL_CREDENTIALS_SECRET_ACCESS_KEY',value='wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY')).add_env_variable(V1EnvVar(name='LAKECTL_SERVER_ENDPOINT_URL',value='https://lakefs.example.com'))
    +
    +
  4. +
+ +

You can invoke any lakeFS operation supported by lakectl by implementing it as a ContainerOp. Check out the complete CLI reference for the list of supported operations.

+ +

Note +The lakeFS Kubeflow integration that uses lakectl is supported on lakeFS version >= v0.43.0.

+

+ + + Add the lakeFS steps to your pipeline + + +

+ + +

Add the steps created in the previous step to your pipeline before compiling it.

+

+ + + Example pipeline + + +

+ + +

A pipeline that implements a simple ETL that has steps for branch creation and commits.

+ +
def lakectl_pipeline():
+   create_branch_task = create_branch_op('example-repo', 'example-branch', 'main') # A function-based component
+   extract_task = example_extract_op()
+   commit_task = commit_op()
+   transform_task = example_transform_op()
+   commit_task = commit_op()
+   load_task = example_load_op()
+
+ +

Note +It’s recommended to store credentials as Kubernetes secrets and pass them as environment variables to Kubeflow operations using V1EnvVarSource.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/minio.html b/v1.46/integrations/minio.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/integrations/minio.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/presto.html b/v1.46/integrations/presto.html new file mode 100644 index 000000000..10389a5cd --- /dev/null +++ b/v1.46/integrations/presto.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/presto_trino.html b/v1.46/integrations/presto_trino.html new file mode 100644 index 000000000..93d0487eb --- /dev/null +++ b/v1.46/integrations/presto_trino.html @@ -0,0 +1,886 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Presto / Trino | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Presto/Trino + + +

+ + +

Presto and Trino are a distributed SQL query engines designed to query large data sets distributed over one or more heterogeneous data sources.

+ +

Querying data in lakeFS from Presto/Trino is similar to querying data in S3 from Presto/Trino. It is done using the Presto Hive connector or Trino Hive connector.

+ + + +
+

Credentials

+ +

In the following examples, we set AWS credentials at runtime for clarity. In production, these properties should be set using one of Hadoop’s standard ways of Authenticating with S3.

+
+

+ + + Configuration + + +

+ +

+ + + Configure the Hive connector + + +

+ + +

Create /etc/catalog/hive.properties with the following contents to mount the hive-hadoop2 connector as the Hive catalog, replacing example.net:9083 with the correct host and port for your Hive Metastore Thrift service:

+
connector.name=hive-hadoop2
+hive.metastore.uri=thrift://example.net:9083
+
+ +

Add the lakeFS configurations to /etc/catalog/hive.properties in the corresponding S3 configuration properties:

+
hive.s3.aws-access-key=AKIAIOSFODNN7EXAMPLE
+hive.s3.aws-secret-key=wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+hive.s3.endpoint=https://lakefs.example.com
+hive.s3.path-style-access=true
+
+

+ + + Configure Hive + + +

+ + +

Presto/Trino uses Hive Metastore Service (HMS) or a compatible implementation of the Hive Metastore such as AWS Glue Data Catalog to write data to S3. +In case you are using Hive Metastore, you will need to configure Hive as well.

+ +

In file hive-site.xml add to the configuration:

+
<configuration>
+    ...
+    <property>
+        <name>fs.s3a.access.key</name>
+        <value>AKIAIOSFODNN7EXAMPLE</value></property>
+    <property>
+        <name>fs.s3a.secret.key</name>
+        <value>wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.s3a.endpoint</name>
+        <value>https://lakefs.example.com</value>
+    </property>
+    <property>
+        <name>fs.s3a.path.style.access</name>
+        <value>true</value>
+    </property>
+</configuration>
+
+

+ + + Examples + + +

+ + +

Here are some examples based on examples from the Presto Hive connector examples and Trino Hive connector examples

+

+ + + Example with schema + + +

+ + +

Create a new schema named main that will store tables in a lakeFS repository named example branch: master:

+
CREATE SCHEMA main
+WITH (location = 's3a://example/main')
+
+ +

Create a new Hive table named page_views in the web schema stored using the ORC file format, + partitioned by date and country, and bucketed by user into 50 buckets (note that Hive requires the partition columns to be the last columns in the table):

+
CREATE TABLE main.page_views (
+  view_time timestamp,
+  user_id bigint,
+  page_url varchar,
+  ds date,
+  country varchar
+)
+WITH (
+  format = 'ORC',
+  partitioned_by = ARRAY['ds', 'country'],
+  bucketed_by = ARRAY['user_id'],
+  bucket_count = 50
+)
+
+

+ + + Example with External table + + +

+ + +

Create an external Hive table named request_logs that points at existing data in lakeFS:

+ +
CREATE TABLE main.request_logs (
+  request_time timestamp,
+  url varchar,
+  ip varchar,
+  user_agent varchar
+)
+WITH (
+  format = 'TEXTFILE',
+  external_location = 's3a://example/main/data/logs/'
+)
+
+

+ + + Example of copying a table with metastore tools: + + +

+ + +

Deprecated Feature: Having heard the feedback from the community, we are planning to replace the below manual steps with an automated process. +You can read more about it here.

+ +

Copy the created table page_views on schema main to schema example_branch with location s3a://example/example_branch/page_views/

+
lakectl metastore copy --from-schema main --from-table page_views --to-branch example_branch 
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/python.html b/v1.46/integrations/python.html new file mode 100644 index 000000000..46805a8a3 --- /dev/null +++ b/v1.46/integrations/python.html @@ -0,0 +1,1563 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Python | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Use Python to interact with your objects on lakeFS + + +

+ + + + +
+

If you are still using the legacy Python lakefs-client, it’s +time to upgrade! This client is deprecated and will be removed +soon.

+
+ +

High Level Python SDK New +We’ve just released a new High Level Python SDK library, and we’re super excited to tell you about it! Continue reading to get the +full story! +Though our previous SDK client is still supported and maintained, we highly recommend using the new High Level SDK. +For previous Python SDKs follow these links: +lakefs-sdk +legacy-sdk (Deprecated)

+ +

There are three primary ways to work with lakeFS from Python:

+ +
    +
  • Use Boto to perform object operations through the lakeFS S3 gateway.
  • +
  • Use the High Level lakeFS SDK to perform object operations, versioning and other lakeFS-specific operations.
  • +
  • Using lakefs-spec to +perform high-level file operations through a file-system-like API.
  • +
+

+ + + Using the lakeFS SDK + + +

+ +

+ + + Installing + + +

+ + +

Install the Python client using pip:

+ +
pip install lakefs
+
+

+ + + Initializing + + +

+ + +

The High Level SDK by default will try to collect authentication parameters from the environment and attempt to create a default client. +When working in an environment where lakectl is configured it is not necessary to instantiate a lakeFS client or provide it for creating the lakeFS objects. +In case no authentication parameters exist, it is also possible to explicitly create a lakeFS client

+ +

Here’s how to instantiate a client:

+ +

See here for instructions on how to log in with Python using your AWS role. This is applicable for enterprise users.

+ +
from lakefs.client import Client
+
+clt = Client(
+    host="http://localhost:8000",
+    username="AKIAIOSFODNN7EXAMPLE",
+    password="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+)
+
+ +

You can use TLS with a CA that is not trusted on the host by configuring the +client with a CA cert bundle file. It should contain concatenated CA +certificates in PEM format:

+
clt = Client(
+    host="http://localhost:8000",
+    username="AKIAIOSFODNN7EXAMPLE",
+    password="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+    # Customize the CA certificates used to verify the peer.
+    ssl_ca_cert="path/to/concatenated_CA_certificates.PEM",
+)
+
+ +

For testing SSL endpoints you may wish to use a self-signed certificate. If you do this and receive an SSL: CERTIFICATE_VERIFY_FAILED error message you might add the following configuration to your client:

+ +
clt = Client(
+    host="http://localhost:8000",
+    username="AKIAIOSFODNN7EXAMPLE",
+    password="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+    verify_ssl=False,
+)
+
+ +

This setting allows well-known “man-in-the-middle”, +impersonation, and credential stealing attacks. Never use this in any +production setting.

+ +

Optionally, to enable communication via proxies, add a proxy configuration:

+ +
clt = Client(
+    host="http://localhost:8000",
+    username="AKIAIOSFODNN7EXAMPLE",
+    password="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+    ssl_ca_cert="(if needed)",
+    proxy="<proxy server URL>",
+)
+
+

+ + + Usage Examples + + +

+ + +

Lets see how we can interact with lakeFS using the High Level SDK.

+

+ + + Creating a repository + + +

+ + +
import lakefs
+
+repo = lakefs.repository("example-repo").create(storage_namespace="s3://storage-bucket/repos/example-repo")
+print(repo)
+
+ +

If using an explicit client, create the Repository object and pass the client to it (note the changed syntax).

+ +
import lakefs
+from lakefs.client import Client
+
+clt = Client(
+    host="http://localhost:8000",
+    username="AKIAIOSFODNN7EXAMPLE",
+    password="wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+)
+
+repo = lakefs.Repository("example-repo", client=clt).create(storage_namespace="s3://storage-bucket/repos/example-repo")
+print(repo)
+
+

+ + + Output + + +

+ +
{id: 'example-repo', creation_date: 1697815536, default_branch: 'main', storage_namespace: 's3://storage-bucket/repos/example-repo'}
+
+

+ + + List repositories + + +

+ + +
import lakefs
+
+print("Listing repositories:")
+for repo in lakefs.repositories():
+    print(repo)
+
+
+

+ + + Output + + +

+ +
Listing repositories:
+{id: 'example-repo', creation_date: 1697815536, default_branch: 'main', storage_namespace: 's3://storage-bucket/repos/example-repo'}
+
+

+ + + Creating a branch + + +

+ + +
import lakefs
+
+branch1 = lakefs.repository("example-repo").branch("experiment1").create(source_reference="main")
+print("experiment1 ref:", branch1.get_commit().id)
+
+branch1 = lakefs.repository("example-repo").branch("experiment2").create(source_reference="main")
+print("experiment2 ref:", branch2.get_commit().id)
+
+

+ + + Output + + +

+ +
experiment1 ref: 7a300b41a8e1ca666c653171a364c08f640549c24d7e82b401bf077c646f8859
+experiment2 ref: 7a300b41a8e1ca666c653171a364c08f640549c24d7e82b401bf077c646f8859
+
+

+ + + List branches + + +

+ + +
import lakefs
+
+for branch in lakefs.repository("example-repo").branches():
+    print(branch)
+
+
+

+ + + Output + + +

+ +
experiment1
+experiment2
+main
+
+

+ + + IO + + +

+ + +

Great, now lets see some IO operations in action! +The new High Level SDK provide IO semantics which allow to work with lakeFS objects as if they were files in your +filesystem. This is extremely useful when working with data transformation packages that accept file descriptors and streams.

+

+ + + Upload + + +

+ + +

A simple way to upload data is to use the upload method which accepts contents as str/bytes

+ +
obj = branch1.object(path="text/sample_data.txt").upload(content_type="text/plain", data="This is my object data")
+print(obj.stats())
+
+

+ + + Output + + +

+ +
{'path': 'text/sample_data.txt', 'physical_address': 's3://storage-bucket/repos/example-repo/data/gke0ignnl531fa6k90p0/ckpfk4fnl531fa6k90pg', 'physical_address_expiry': None, 'checksum': '4a09d10820234a95bb548f14e4435bba', 'size_bytes': 15, 'mtime': 1701865289, 'metadata': {}, 'content_type': 'text/plain'}
+
+ +

Reading the data is just as simple:

+
print(obj.reader(mode='r').read())
+
+

+ + + Output + + +

+ +
This is my object data
+
+ +

Now let’s generate a “sample_data.csv” file and write it directly to a lakeFS writer object

+ +
import csv
+
+sample_data = [
+    [1, "Alice", "alice@example.com"],
+    [2, "Bob", "bob@example.com"],
+    [3, "Carol", "carol@example.com"],
+]
+
+obj = branch1.object(path="csv/sample_data.csv")
+
+with obj.writer(mode='w', pre_sign=True, content_type="text/csv") as fd:
+    writer = csv.writer(fd)
+    writer.writerow(["ID", "Name", "Email"])
+    for row in sample_data:
+        writer.writerow(row)
+
+ +

On context exit the object will be uploaded to lakeFS

+ +
print(obj.stats())
+
+

+ + + Output + + +

+ +
{'path': 'csv/sample_data.csv', 'physical_address': 's3://storage-bucket/repos/example-repo/data/gke0ignnl531fa6k90p0/ckpfk4fnl531fa6k90pg', 'physical_address_expiry': None, 'checksum': 'f181262c138901a74d47652d5ea72295', 'size_bytes': 88, 'mtime': 1701865939, 'metadata': {}, 'content_type': 'text/csv'}
+
+ +

We can also upload raw byte contents:

+ +
obj = branch1.object(path="raw/file1.data").upload(data=b"Hello Object World", pre_sign=True)
+print(obj.stats())
+
+

+ + + Output + + +

+ +
{'path': 'raw/file1.data', 'physical_address': 's3://storage-bucket/repos/example-repo/data/gke0ignnl531fa6k90p0/ckpfltvnl531fa6k90q0', 'physical_address_expiry': None, 'checksum': '0ef432f8eb0305f730b0c57bbd7a6b08', 'size_bytes': 18, 'mtime': 1701866323, 'metadata': {}, 'content_type': 'application/octet-stream'}
+
+

+ + + Uncommitted changes + + +

+ + +

Using the branch uncommmitted method will show all the uncommitted changes on that branch:

+ +
for diff in branch1.uncommitted():
+    print(diff)
+
+

+ + + Output + + +

+ + +
{'type': 'added', 'path': 'text/sample_data.txt', 'path_type': 'object', 'size_bytes': 15}
+{'type': 'added', 'path': 'csv/sample_data.csv', 'path_type': 'object', 'size_bytes': 88}
+{'type': 'added', 'path': 'raw/file1.data', 'path_type': 'object', 'size_bytes': 18}
+
+
+ +

As expected, our change appears here. Let’s commit it and attach some arbitrary metadata:

+ +
ref = branch1.commit(message='Add some data!', metadata={'using': 'python_sdk'})
+print(ref.get_commit())
+
+

+ + + Output + + +

+ +
{'id': 'c4666db80d2a984b4eab8ce02b6a60830767eba53995c26350e0ad994e15fedb', 'parents': ['a7a092a5a32a2cd97f22abcc99414f6283d29f6b9dd2725ce89f90188c5901e5'], 'committer': 'admin', 'message': 'Add some data!', 'creation_date': 1701866838, 'meta_range_id': '999bedeab1b740f83d2cf8c52548d55446f9038c69724d399adc4438412cade2', 'metadata': {'using': 'python_sdk'}}
+
+
+ +

Calling uncommitted again on the same branch, this time there should be no uncommitted files:

+ +
print(len(list(branch1.uncommitted())))
+
+

+ + + Output + + +

+ +
0
+
+

+ + + Merging changes from a branch into main + + +

+ + +

Let’s diff between your branch and the main branch:

+ +
main = repo.branch("main")
+for diff in main.diff(other_ref=branch1):
+    print(diff)
+
+

+ + + Output + + +

+ +
{'type': 'added', 'path': 'text/sample_data.txt', 'path_type': 'object', 'size_bytes': 15}
+{'type': 'added', 'path': 'csv/sample_data.csv', 'path_type': 'object', 'size_bytes': 88}
+{'type': 'added', 'path': 'raw/file1.data', 'path_type': 'object', 'size_bytes': 18}
+
+ +

Looks like we have some changes. Let’s merge them:

+ +
res = branch1.merge_into(main)
+print(res)
+# output:
+# cfddb68b7265ae0b17fafa1a2068f8414395e0a8b8bc0f8d741cbcce1e67e394
+
+ +

Let’s diff again - there should be no changes as all changes are on our main branch already:

+ +
print(len(list(main.diff(other_ref=branch1))))
+
+

+ + + Output + + +

+ +
0
+
+

+ + + Read data from main branch + + +

+ + +
import csv
+
+obj = main.object(path="csv/sample_data.csv")
+
+for row in csv.reader(obj.reader(mode='r')):
+    print(row)
+
+

+ + + Output + + +

+ +
['ID', 'Name', 'Email']
+['1', 'Alice', 'alice@example.com']
+['2', 'Bob', 'bob@example.com']
+['3', 'Carol', 'carol@example.com']
+
+

+ + + Importing data into lakeFS + + +

+ + +

The new SDK makes it much easier to import existing data from the object store into lakeFS, using the new ImportManager

+ +
import lakefs
+
+branch = lakefs.repository("example-repo").repo.branch("experiment3")
+
+# We can import data from multiple sources in a single import process
+# The following example initializes a new ImportManager and adds 2 source types; A prefix and an object.
+importer = branch.import_data(commit_message="added public S3 data") \
+    .prefix("s3://example-bucket1/path1/", destination="datasets/path1/") \
+    .object("s3://example-bucket1/path2/imported_obj", destination="datasets/path2/imported_obj")
+
+# run() is a convenience method that blocks until the import is reported as done, raising an exception if it fails.
+importer.run()
+
+
+ +

Alternatively we can call start() and status() ourselves for an async version of the above

+ +
import time
+
+# Async version
+importer.start()
+status = importer.start()
+
+while not status.completed or status.error is None:
+        time.sleep(3)  # or whatever interval you choose
+        status = importer.status()
+
+if status.error:
+    # handle!
+
+print(f"imported a total of {status.ingested_objects} objects!")
+
+
+

+ + + Output + + +

+ +
imported a total of 25478 objects!
+
+

+ + + Transactions + + +

+ + +

Transactions is a new feature in the High Level SDK. It allows performing a sequence of operations on a branch as an atomic unit, similarly to how database transactions work. +Under the hood, the transaction creates an ephemeral branch from the source branch, performs all the operation on that branch, and merges it back to the source branch once the transaction is completed. +Transactions are currently supported as a context manager only.

+ +
import lakefs
+
+branch = lakefs.repository("example-repo").repo.branch("experiment3")
+
+with branch.transact(commit_message="my transaction") as tx:
+    for obj in tx.objects(prefix="prefix_to_delete/"):  # Delete some objects
+        obj.delete()
+
+    # Create new object
+    tx.object("new_object").upload("new object data")
+
+print(len(list(branch.objects(prefix="prefix_to_delete/"))))
+print(branch.object("new_object").exists())
+
+

+ + + Output + + +

+ +
0
+True
+
+

+ + + Python SDK documentation and API reference + + +

+ + +

For the documentation of lakeFS’s Python package and full api reference, see https://pydocs-lakefs.lakefs.io

+

+ + + Using lakefs-spec for higher-level file operations + + +

+ + +

The lakefs-spec project +provides higher-level file operations on lakeFS objects with a filesystem API, +built on the fsspec project.

+ +

Note This library is a third-party package and not maintained by the lakeFS developers; please file issues and bug reports directly +in the lakefs-spec repository.

+

+ + + Installation + + +

+ + +

Install lakefs-spec directly with pip:

+ +
python -m pip install --upgrade lakefs-spec
+
+

+ + + Interacting with lakeFS through a file system + + +

+ + +

To write a file directly to a branch in a lakeFS repository, consider the following example:

+ +
from pathlib import Path
+
+from lakefs_spec import LakeFSFileSystem
+
+REPO, BRANCH = "example-repo", "main"
+
+# Prepare a local example file.
+lpath = Path("demo.txt")
+lpath.write_text("Hello, lakeFS!")
+
+fs = LakeFSFileSystem()  # will auto-discover credentials from ~/.lakectl.yaml
+rpath = f"{REPO}/{BRANCH}/{lpath.name}"
+fs.put(lpath, rpath)
+
+ +

Reading it again from remote is as easy as the following:

+ +
f = fs.open(rpath, "rt")
+print(f.readline())  # prints "Hello, lakeFS!"
+
+ +

Many more operations like retrieving an object’s metadata or checking an +object’s existence on the lakeFS server are also supported. For a full list, +see the API reference.

+ + + +

A number of Python data science projects support fsspec, with pandas being a prominent example. Reading a Parquet file from a lakeFS repository into a Pandas data frame for analysis is very easy, demonstrated on the quickstart repository sample data:

+ +
import pandas as pd
+
+# Read into pandas directly by supplying the lakeFS URI...
+lakes = pd.read_parquet(f"lakefs://quickstart/main/lakes.parquet")
+german_lakes = lakes.query('Country == "Germany"')
+# ... and store directly, again with a raw lakeFS URI.
+german_lakes.to_csv(f"lakefs://quickstart/main/german_lakes.csv")
+
+ +

A list of integrations with popular data science libraries can be found in the lakefs-spec documentation.

+

+ + + Using transactions for atomic versioning operations + + +

+ + +

As with the high-level SDK (see above), lakefs-spec also supports transactions +for conducting versioning operations on newly modified files. The following is an example of creating a commit on the repository’s main branch directly after a file upload:

+ +
from lakefs_spec import LakeFSFileSystem
+
+fs = LakeFSFileSystem()
+
+# assumes you have a local train-test split as two text files:
+# train-data.txt, and test-data.txt.
+with fs.transaction("example-repo", "main") as tx:
+    fs.put_file("train-data.txt", f"example-repo/{tx.branch.id}/train-data.txt")
+    tx.commit(message="Add training data")
+    fs.put_file("test-data.txt", f"example-repo/{tx.branch.id}/test-data.txt")
+    sha = tx.commit(message="Add test data")
+    tx.tag(sha, name="My train-test split")
+
+ +

Transactions are atomic - if an exception happens at any point of the transaction, the repository remains unchanged.

+

+ + + Further information + + +

+ + +

For more user guides, tutorials on integrations with data science tools like pandas, and more, check out the lakefs-spec documentation.

+

+ + + Using Boto + + +

+ + +

💡 To use Boto with lakeFS alongside S3, check out Boto S3 Router. It will route +requests to either S3 or lakeFS according to the provided bucket name.

+ +

lakeFS exposes an S3-compatible API, so you can use Boto to interact with your objects on lakeFS.

+

+ + + Initializing + + +

+ + +

Create a Boto3 S3 client with your lakeFS endpoint and key-pair:

+ +
import boto3
+s3 = boto3.client('s3',
+    endpoint_url='https://lakefs.example.com',
+    aws_access_key_id='AKIAIOSFODNN7EXAMPLE',
+    aws_secret_access_key='wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY')
+
+ +

The client is now configured to operate on your lakeFS installation.

+

+ + + Usage Examples + + +

+ +

+ + + Put an object into lakeFS + + +

+ + +

Use a branch name and a path to put an object in lakeFS:

+ +
with open('/local/path/to/file_0', 'rb') as f:
+    s3.put_object(Body=f, Bucket='example-repo', Key='main/example-file.parquet')
+
+ +

You can now commit this change using the lakeFS UI or CLI.

+

+ + + List objects + + +

+ + +

List the branch objects starting with a prefix:

+ +
list_resp = s3.list_objects_v2(Bucket='example-repo', Prefix='main/example-prefix')
+for obj in list_resp['Contents']:
+    print(obj['Key'])
+
+ +

Or, use a lakeFS commit ID to list objects for a specific commit:

+ +
list_resp = s3.list_objects_v2(Bucket='example-repo', Prefix='c7a632d74f/example-prefix')
+for obj in list_resp['Contents']:
+    print(obj['Key'])
+
+

+ + + Get object metadata + + +

+ + +

Get object metadata using branch and path:

+ +
s3.head_object(Bucket='example-repo', Key='main/example-file.parquet')
+# output:
+# {'ResponseMetadata': {'RequestId': '72A9EBD1210E90FA',
+#  'HostId': '',
+#  'HTTPStatusCode': 200,
+#  'HTTPHeaders': {'accept-ranges': 'bytes',
+#   'content-length': '1024',
+#   'etag': '"2398bc5880e535c61f7624ad6f138d62"',
+#   'last-modified': 'Sun, 24 May 2020 10:42:24 GMT',
+#   'x-amz-request-id': '72A9EBD1210E90FA',
+#   'date': 'Sun, 24 May 2020 10:45:42 GMT'},
+#  'RetryAttempts': 0},
+# 'AcceptRanges': 'bytes',
+# 'LastModified': datetime.datetime(2020, 5, 24, 10, 42, 24, tzinfo=tzutc()),
+# 'ContentLength': 1024,
+# 'ETag': '"2398bc5880e535c61f7624ad6f138d62"',
+# 'Metadata': {}}
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/r.html b/v1.46/integrations/r.html new file mode 100644 index 000000000..26719c4c6 --- /dev/null +++ b/v1.46/integrations/r.html @@ -0,0 +1,1006 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +R | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using R with lakeFS + + +

+ + +

R is a powerful language used widely in data science. lakeFS interfaces with R in two ways:

+ +
    +
  • To read and write data in lakeFS use standard S3 tools such as the aws.s3 library. lakeFS has a S3 gateway which presents a lakeFS repository as an S3 bucket.
  • +
  • For working with lakeFS operations such as branches and commits use the API for which can be accessed from R using the httr library.
  • +
+ +

To see examples of R in action with lakeFS please visit the lakeFS-samples repository and the sample notebooks.

+ + +

+ + + Reading and Writing from lakeFS with R + + +

+ + +

Working with data stored in lakeFS from R is the same as you would with an S3 bucket, via the S3 Gateway that lakeFS provides.

+ +

You can use any library that interfaces with S3. In this example we’ll use the aws.s3 library.

+ +
install.packages(c("aws.s3"))
+library(aws.s3)
+
+

+ + + Configuration + + +

+ + +

The R S3 client documentation includes full details of the configuration options available. A good approach for using it with lakeFS set the endpoint and authentication details as environment variables:

+ +
Sys.setenv("AWS_ACCESS_KEY_ID" = "AKIAIOSFODNN7EXAMPLE",
+           "AWS_SECRET_ACCESS_KEY" = "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+           "AWS_S3_ENDPOINT" = "lakefs.mycorp.com:8000")
+
+ +

Note: it is generally best practice to set these environment variables outside of the R script; it is done so here for convenience of the example.

+ +

In conjunction with this you must also specify region and use_https in each call of an aws.s3 function as these cannot be set globally. For example:

+ +
bucketlist(
+    region = "",
+    use_https = FALSE
+    )
+
+ +
    +
  • region should always be empty
  • +
  • use_https should be set to TRUE or FALSE depending on whether your lakeFS endpoint uses HTTPS.
  • +
+

+ + + Listing repositories + + +

+ + +

The S3 gateway exposes a repository as a bucket, and so using the aws.s3 function bucketlist will return a list of available repositories on lakeFS:

+ +
bucketlist(
+    region = "",
+    use_https = FALSE
+    )
+
+

+ + + Writing to lakeFS from R + + +

+ + +

Assuming you’re using the aws.s3 library there various functions available including s3save, s3saveRDS, and put_object. Here’s an example of writing an R object to lakeFS:

+ +
repo_name <- "example"
+branch <- "development"
+
+s3saveRDS(x=my_df, 
+          bucket = repo_name, 
+          object = paste0(branch,"/my_df.R"), 
+          region = "",
+          use_https = FALSE)
+
+ +

You can also upload local files to lakeFS using R and the put_object function:

+ +
repo_name <- "example"
+branch <- "development"
+local_file <- "/tmp/never.gonna"
+
+put_object(file = local_file, 
+           bucket = repo_name, 
+           object = paste0(branch,"/give/you/up"),
+           region = "",
+           use_https = FALSE)
+
+

+ + + Reading from lakeFS with R + + +

+ + +

As with writing data from R to lakeFS, there is a similar set of functions for reading data. These include s3load, s3readRDS, and get_object. Here’s an example of reading an R object from lakeFS:

+ +
repo_name <- "example"
+branch <- "development"
+
+my_df <- s3readRDS(bucket = repo_name, 
+                   object = paste0(branch,"/my_data.R"),
+                   region = "",
+                   use_https = FALSE)
+
+

+ + + Listing Objects + + +

+ + +

In general you should always specify a branch prefix when listing objects. Here’s an example to list the main branch in the quickstart repository:

+ +
get_bucket_df(bucket = "quickstart",
+              prefix = "main/",
+              region = "",
+              use_https = FALSE)
+
+ +

When listing objects in lakeFS there is a special case which is the repository/bucket level. When you list at this level you will get the branches returned as folders. These are not listed recursively, unless you list something under the branch. To understand more about this please refer to #5441

+

+ + + Working with Arrow + + +

+ + +

Arrow’s R library includes powerful support for data analysis, including reading and writing multiple file formats including Parquet, Arrow, CSV, and JSON. It has functionality for connecting to S3, and thus integrates perfectly with lakeFS.

+ +

To start with install and load the library

+ +
install.packages("arrow")
+library(arrow)
+
+ +

Then create an S3FileSystem object to connect to your lakeFS instance

+ +
lakefs <- S3FileSystem$create(
+    endpoint_override = "lakefs.mycorp.com:8000",
+    scheme = "http"
+    access_key = "AKIAIOSFODNN7EXAMPLE", 
+    secret_key = "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY", 
+    region = "",
+)
+
+ +

From here you can list the contents of a particular lakeFS repository and branch

+ +
lakefs$ls(path = "quickstart/main")
+
+ +

To read a Parquet from lakeFS with R use the read_parquet function

+ +
lakes <- read_parquet(lakefs$path("quickstart/main/lakes.parquet"))
+
+ +

Writing a file follows a similar pattern. Here is rewriting the same file as above but in Arrow format

+ +
write_feather(x = lakes,
+              sink = lakefs$path("quickstart/main/lakes.arrow"))
+
+

+ + + Performing lakeFS Operations using the lakeFS API from R + + +

+ + +

As well as reading and writing data, you will also want to carry out lakeFS operations from R including creating branches, committing data, and more.

+ +

To do this call the lakeFS API from the httr library. You should refer to the API documentation for full details of the endpoints and their behaviour. Below are a few examples to illustrate the usage.

+

+ + + Check the lakeFS Server Version + + +

+ + +

This is a useful API call to establish connectivity and test authentication.

+ +
library(httr)
+lakefs_api_url <- "lakefs.mycorp.com:8000/api/v1"
+lakefsAccessKey <- "AKIAIOSFODNN7EXAMPLE"
+lakefsSecretKey <- "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY"
+
+r=GET(url=paste0(lakefs_api_url, "/config/version"), 
+      authenticate(lakefsAccessKey, lakefsSecretKey))
+
+ +

The returned object r can be inspected to determine the outcome of the operation by comparing it to the status codes specified in the API. Here is some example R code to demonstrate the idea:

+ +
if (r$status_code == 200) {
+    print(paste0("✅lakeFS credentials and connectivity verified. ℹ️lakeFS version ",content(r)$version))   
+} else {
+    print("🛑 failed to get lakeFS version")
+    print(content(r)$message)
+}
+
+

+ + + Create a Repository + + +

+ + +
library(httr)
+lakefs_api_url <- "lakefs.mycorp.com:8000/api/v1"
+lakefsAccessKey <- "AKIAIOSFODNN7EXAMPLE"
+lakefsSecretKey <- "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY"
+repo_name <- "my_new_repo"
+
+# Define the payload
+body=list(name=repo_name, 
+          storage_namespace="s3://example-bucket/foo")
+
+# Call the API
+r=POST(url=paste0(lakefs_api_url, "/repositories"), 
+        authenticate(lakefsAccessKey, lakefsSecretKey),
+        body=body, encode="json")
+
+

+ + + Commit Data + + +

+ + +
library(httr)
+lakefs_api_url <- "lakefs.mycorp.com:8000/api/v1"
+lakefsAccessKey <- "AKIAIOSFODNN7EXAMPLE"
+lakefsSecretKey <- "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY"
+repo_name <- "my_new_repo"
+branch <- "example"
+
+# Define the payload
+body=list(message="add some data and charts", 
+          metadata=list(
+              client="httr", 
+              author="rmoff"))
+
+# Call the API
+r=POST(url=paste0(lakefs_api_url, "/repositories/", repo_name, "/branches/", branch, "/commits"), 
+       authenticate(lakefsAccessKey, lakefsSecretKey),
+       body=body, encode="json")
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/rclone.html b/v1.46/integrations/rclone.html new file mode 100644 index 000000000..07c435e7e --- /dev/null +++ b/v1.46/integrations/rclone.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/red_hat_openshift_ai.html b/v1.46/integrations/red_hat_openshift_ai.html new file mode 100644 index 000000000..378d6a679 --- /dev/null +++ b/v1.46/integrations/red_hat_openshift_ai.html @@ -0,0 +1,738 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Red Hat OpenShift AI | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Red Hat OpenShift AI + + +

+ +

Red Hat® OpenShift® is an enterprise-ready Kubernetes container platform with full-stack automated operations to manage hybrid cloud, multi-cloud, and edge deployments. OpenShift includes an enterprise-grade Linux operating system, container runtime, networking, monitoring, registry, and authentication and authorization solutions.

+ +

Red Hat® OpenShift® AI is a flexible, scalable artificial intelligence (AI) and machine learning (ML) platform that enables enterprises to create and deliver AI-enabled applications at scale across hybrid cloud environments. Built using open source technologies, OpenShift AI provides trusted, operationally consistent capabilities for teams to experiment, serve models, and deliver innovative apps.

+ +

OpenShift AI and lakeFS can be deployed in OpenShift cluster in 3 different architectures:

+
    +
  1. OpenShift AI, lakeFS and object storage are delpoyed in OpenShift cluster
  2. +
  3. OpenShift AI and lakeFS are deployed in OpenShift cluster while object storage is external
  4. +
  5. OpenShift AI is deployed in OpenShift cluster while lakeFS and object storage are external
  6. +
+ +

OpenShift AI and lakeFS Deployment Architecture

+ +

Refer to an example in lakeFS-samples to deploy lakeFS, MinIO and OpenShift AI tutorial (Fraud Detection demo) in OpenShift cluster. Fraud detection demo is a step-by-step guide for using OpenShift AI to train an example model in JupyterLab, deploy the model, and refine the model by using automated pipelines.

+ +

In this example, OpenShift AI is configured to connect over S3 interace to lakeFS, which will version the data in a backend MinIO instance. This is the architecture to run Fraud Detection demo with or without lakeFS: +OpenShift AI and lakeFS Deployment Architecture for the demo

+ +

lakeFS-samples also includes multiple Helm chart examples to deploy lakeFS and MinIO in different scenarios:

+
    +
  1. lakefs-local.yaml: Bring up lakeFS using local object storage. This would be useful for a quick demo where MinIO is not included.
  2. +
  3. lakefs-minio.yaml: Bring up lakeFS configured to use MinIO as backend object storage. This will be used in the lakeFS demo.
  4. +
  5. minio-direct.yaml: This file would only be used if lakeFS is not in the picture and OpenShift AI will communicate directly with MinIO. It will bring up MinIO as it is in the default Fraud Detection demo, complete with configuring MinIO storage buckets and the OpenShift AI data connections. It may serve useful in debugging an issue.
  6. +
  7. minio-via-lakefs.yaml: Bring up MinIO for the modified Fraud Detection demo that includes lakeFS, complete with configuring MinIO storage buckets, but do NOT configure the OpenShift AI data connections. This will be used in the lakeFS demo.
  8. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/sagemaker.html b/v1.46/integrations/sagemaker.html new file mode 100644 index 000000000..d5df0fe55 --- /dev/null +++ b/v1.46/integrations/sagemaker.html @@ -0,0 +1,815 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Amazon SageMaker | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Amazon SageMaker + + +

+ +

Amazon SageMaker helps to prepare, build, train and deploy ML models quickly by bringing together a broad set of capabilities purpose-built for ML.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Initializing session and client
  2. +
  3. Usage Examples
  4. +
+ +
+

+ + + Initializing session and client + + +

+ + +

Initialize a Sagemaker session and an S3 client with lakeFS as the endpoint:

+
import sagemaker
+import boto3
+
+endpoint_url = '<LAKEFS_ENDPOINT>'
+aws_access_key_id = '<LAKEFS_ACCESS_KEY_ID>'
+aws_secret_access_key = '<LAKEFS_SECRET_ACCESS_KEY>'
+repo = 'example-repo'
+
+sm = boto3.client('sagemaker',
+    endpoint_url=endpoint_url,
+    aws_access_key_id=aws_access_key_id,
+    aws_secret_access_key=aws_secret_access_key)
+
+s3_resource = boto3.resource('s3',
+    endpoint_url=endpoint_url,
+    aws_access_key_id=aws_access_key_id,
+    aws_secret_access_key=aws_secret_access_key)
+
+session = sagemaker.Session(boto3.Session(), sagemaker_client=sm, default_bucket=repo)
+session.s3_resource = s3_resource
+
+

+ + + Usage Examples + + +

+ +

+ + + Upload train and test data + + +

+ + +

Let’s use the created session for uploading data to the ‘main’ branch:

+ +
prefix = "/prefix-within-branch"
+branch = 'main'
+
+train_file = 'train_data.csv';
+train_data.to_csv(train_file, index=False, header=True)
+train_data_s3_path = session.upload_data(path=train_file, key_prefix=branch + prefix + "/train")
+
+test_file = 'test_data.csv';
+test_data_no_target.to_csv(test_file, index=False, header=False)
+test_data_s3_path = session.upload_data(path=test_file, key_prefix=branch + prefix + "/test")
+
+

+ + + Download objects + + +

+ + +

You can use the integration with lakeFS to download a portion of the data you see fit:

+ +
repo = 'example-repo'
+prefix = "/prefix-to-download"
+branch = 'main'
+localpath = './' + branch
+
+session.download_data(path=localpath, bucket=repo, key_prefix = branch + prefix)
+
+ +

Note: +Advanced AWS SageMaker features, like Autopilot jobs, are encapsulated and don’t have the option to override the S3 endpoint. +However, it is possible to export the required inputs from lakeFS to S3. +
If you’re using SageMaker features that aren’t supported by lakeFS, we’d love to hear from you.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/spark.html b/v1.46/integrations/spark.html new file mode 100644 index 000000000..707753fa0 --- /dev/null +++ b/v1.46/integrations/spark.html @@ -0,0 +1,1573 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Apache Spark | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with Apache Spark + + +

+ + +

There are several ways to use lakeFS with Spark:

+ + + +

See how SimilarWeb is using lakeFS with Spark to manage algorithm changes in data pipelines.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. S3-compatible API
  2. +
  3. lakeFS Hadoop FileSystem
  4. +
  5. Hadoop FileSystem in Presigned mode
  6. +
+ +
+

+ + + S3-compatible API + + +

+ + +

lakeFS has an S3-compatible endpoint which you can point Spark at to get started quickly.

+ +

You will access your data using S3-style URIs, e.g. s3a://example-repo/example-branch/example-table.

+ +

You can use the S3-compatible API regardless of where your data is hosted.

+

+ + + Configuration + + +

+ + +

To configure Spark to work with lakeFS, we set S3A Hadoop configuration to the lakeFS endpoint and credentials:

+ +
    +
  • fs.s3a.access.key: lakeFS access key
  • +
  • fs.s3a.secret.key: lakeFS secret key
  • +
  • fs.s3a.endpoint: lakeFS S3-compatible API endpoint (e.g. https://example-org.us-east-1.lakefscloud.io)
  • +
  • fs.s3a.path.style.access: true
  • +
+ +

Here is how to do it:

+
+ +
+
spark-shell --conf spark.hadoop.fs.s3a.access.key='AKIAlakefs12345EXAMPLE' \
+              --conf spark.hadoop.fs.s3a.secret.key='abc/lakefs/1234567bPxRfiCYEXAMPLEKEY' \
+              --conf spark.hadoop.fs.s3a.path.style.access=true \
+              --conf spark.hadoop.fs.s3a.endpoint='https://example-org.us-east-1.lakefscloud.io' ...
+
+
+
+
spark.sparkContext.hadoopConfiguration.set("fs.s3a.access.key", "AKIAlakefs12345EXAMPLE")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.endpoint", "https://example-org.us-east-1.lakefscloud.io")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.path.style.access", "true")
+
+
+
+

Add these into a configuration file, e.g. $SPARK_HOME/conf/hdfs-site.xml:

+
<?xml version="1.0"?>
+<configuration>
+    <property>
+        <name>fs.s3a.access.key</name>
+        <value>AKIAlakefs12345EXAMPLE</value>
+    </property>
+    <property>
+            <name>fs.s3a.secret.key</name>
+            <value>abc/lakefs/1234567bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.s3a.endpoint</name>
+        <value>https://example-org.us-east-1.lakefscloud.io</value>
+    </property>
+    <property>
+        <name>fs.s3a.path.style.access</name>
+        <value>true</value>
+    </property>
+</configuration>
+
+
+
+

Use the below configuration when creating the cluster. You may delete any app configuration that is not suitable for your use case:

+ +
[
+  {
+    "Classification": "spark-defaults",
+    "Properties": {
+      "spark.sql.catalogImplementation": "hive"
+    }
+  },
+  {
+    "Classification": "core-site",
+    "Properties": {
+        "fs.s3.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.path.style.access": "true",
+        "fs.s3a.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "emrfs-site",
+    "Properties": {
+        "fs.s3.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.path.style.access": "true",
+        "fs.s3a.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "presto-connector-hive",
+    "Properties": {
+        "hive.s3.aws-access-key": "AKIAIOSFODNN7EXAMPLE",
+        "hive.s3.aws-secret-key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "hive.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "hive.s3.path-style-access": "true",
+        "hive.s3-file-system-type": "PRESTO"
+    }
+  },
+  {
+    "Classification": "hive-site",
+    "Properties": {
+        "fs.s3.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.path.style.access": "true",
+        "fs.s3a.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "hdfs-site",
+    "Properties": {
+        "fs.s3.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.path.style.access": "true",
+        "fs.s3a.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "mapred-site",
+    "Properties": {
+        "fs.s3.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.path.style.access": "true",
+        "fs.s3a.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.path.style.access": "true"
+    }
+  }
+]
+
+ +

Alternatively, you can pass these configuration values when adding a step.

+ +

For example:

+ +
aws emr add-steps --cluster-id j-197B3AEGQ9XE4 \
+  --steps="Type=Spark,Name=SparkApplication,ActionOnFailure=CONTINUE, \
+  Args=[--conf,spark.hadoop.fs.s3a.access.key=AKIAIOSFODNN7EXAMPLE, \
+  --conf,spark.hadoop.fs.s3a.secret.key=wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY, \
+  --conf,spark.hadoop.fs.s3a.endpoint=https://example-org.us-east-1.lakefscloud.io, \
+  --conf,spark.hadoop.fs.s3a.path.style.access=true, \
+  s3a://<lakefs-repo>/<lakefs-branch>/path/to/jar]"
+
+ +
+
+

+ + + Per-bucket configuration + + +

+ + +

The above configuration will use lakeFS as the sole S3 endpoint. To use lakeFS in parallel with S3, you can configure Spark to use lakeFS only for specific bucket names. +For example, to configure only example-repo to use lakeFS, set the following configurations:

+ +
+ +
+
spark-shell --conf spark.hadoop.fs.s3a.bucket.example-repo.access.key='AKIAlakefs12345EXAMPLE' \
+              --conf spark.hadoop.fs.s3a.bucket.example-repo.secret.key='abc/lakefs/1234567bPxRfiCYEXAMPLEKEY' \
+              --conf spark.hadoop.fs.s3a.bucket.example-repo.endpoint='https://example-org.us-east-1.lakefscloud.io' \
+              --conf spark.hadoop.fs.s3a.path.style.access=true
+
+
+
+
spark.sparkContext.hadoopConfiguration.set("fs.s3a.bucket.example-repo.access.key", "AKIAlakefs12345EXAMPLE")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.bucket.example-repo.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.bucket.example-repo.endpoint", "https://example-org.us-east-1.lakefscloud.io")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.path.style.access", "true")
+
+
+
+

Add these into a configuration file, e.g. $SPARK_HOME/conf/hdfs-site.xml:

+
<?xml version="1.0"?>
+<configuration>
+    <property>
+        <name>fs.s3a.bucket.example-repo.access.key</name>
+        <value>AKIAlakefs12345EXAMPLE</value>
+    </property>
+    <property>
+        <name>fs.s3a.bucket.example-repo.secret.key</name>
+        <value>abc/lakefs/1234567bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.s3a.bucket.example-repo.endpoint</name>
+        <value>https://example-org.us-east-1.lakefscloud.io</value>
+    </property>
+    <property>
+        <name>fs.s3a.path.style.access</name>
+        <value>true</value>
+    </property>
+</configuration>
+
+
+
+

Use the below configuration when creating the cluster. You may delete any app configuration that is not suitable for your use case:

+ +
[
+  {
+    "Classification": "spark-defaults",
+    "Properties": {
+      "spark.sql.catalogImplementation": "hive"
+    }
+  },
+  {
+    "Classification": "core-site",
+    "Properties": {
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.bucket.example-repo.path.style.access": "true",
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.bucket.example-repo.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "emrfs-site",
+    "Properties": {
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.bucket.example-repo.path.style.access": "true",
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.bucket.example-repo.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "presto-connector-hive",
+    "Properties": {
+        "hive.s3.aws-access-key": "AKIAIOSFODNN7EXAMPLE",
+        "hive.s3.aws-secret-key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "hive.s3.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "hive.s3.path-style-access": "true",
+        "hive.s3-file-system-type": "PRESTO"
+    }
+  },
+  {
+    "Classification": "hive-site",
+    "Properties": {
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.bucket.example-repo.path.style.access": "true",
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.bucket.example-repo.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "hdfs-site",
+    "Properties": {
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.bucket.example-repo.path.style.access": "true",
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.bucket.example-repo.path.style.access": "true"
+    }
+  },
+  {
+    "Classification": "mapred-site",
+    "Properties": {
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3.bucket.example-repo.path.style.access": "true",
+        "fs.s3a.bucket.example-repo.access.key": "AKIAIOSFODNN7EXAMPLE",
+        "fs.s3a.bucket.example-repo.secret.key": "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY",
+        "fs.s3a.bucket.example-repo.endpoint": "https://example-org.us-east-1.lakefscloud.io",
+        "fs.s3a.bucket.example-repo.path.style.access": "true"
+    }
+  }
+]
+
+ +

Alternatively, you can pass these configuration values when adding a step.

+ +

For example:

+ +
aws emr add-steps --cluster-id j-197B3AEGQ9XE4 \
+  --steps="Type=Spark,Name=SparkApplication,ActionOnFailure=CONTINUE, \
+  Args=[--conf,spark.hadoop.fs.s3a.bucket.example-repo.access.key=AKIAIOSFODNN7EXAMPLE, \
+  --conf,spark.hadoop.fs.s3a.bucket.example-repo.secret.key=wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY, \
+  --conf,spark.hadoop.fs.s3a.bucket.example-repo.endpoint=https://example-org.us-east-1.lakefscloud.io, \
+  --conf,spark.hadoop.fs.s3a.path.style.access=true, \
+  s3a://<lakefs-repo>/<lakefs-branch>/path/to/jar]"
+
+ +
+
+ +

With this configuration set, you read S3A paths with example-repo as the bucket will use lakeFS, while all other buckets will use AWS S3.

+

+ + + Usage + + +

+ + +

Here’s an example for reading a Parquet file from lakeFS to a Spark DataFrame:

+ +
val repo = "example-repo"
+val branch = "main"
+val df = spark.read.parquet(s"s3a://${repo}/${branch}/example-path/example-file.parquet")
+
+ +

Here’s how to write some results back to a lakeFS path:

+
df.write.partitionBy("example-column").parquet(s"s3a://${repo}/${branch}/output-path/")
+
+ +

The data is now created in lakeFS as new changes in your branch. You can now commit these changes or revert them.

+

+ + + Configuring Azure Databricks with the S3-compatible API + + +

+ + +

If you use Azure Databricks, you can take advantage of the lakeFS S3-compatible API with your Azure account and the S3A FileSystem. +This will require installing the hadoop-aws package (with the same version as your hadoop-azure package) to your Databricks cluster.

+ +

Define your FileSystem configurations in the following way:

+ +
spark.hadoop.fs.lakefs.impl=org.apache.hadoop.fs.s3a.S3AFileSystem
+spark.hadoop.fs.lakefs.access.key=‘AKIAlakefs12345EXAMPLE’                   // The access key to your lakeFS server
+spark.hadoop.fs.lakefs.secret.key=‘abc/lakefs/1234567bPxRfiCYEXAMPLEKEY’     // The secret key to your lakeFS server
+spark.hadoop.fs.lakefs.path.style.access=true
+spark.hadoop.fs.lakefs.endpoint=‘https://example-org.us-east-1.lakefscloud.io’                 // The endpoint of your lakeFS server
+
+ +

For more details about Mounting cloud object storage on Databricks.

+

+ + + Configuring Databricks SQL Warehouse with the S3-compatible API + + +

+ + +

A SQL warehouse is a compute resource that lets you run SQL commands on data +objects within Databricks SQL.

+ +

If you use Databricks SQL warehouse, you can take advantage of the lakeFS +S3-compatible API with the S3A FileSystem.

+ +

Define your SQL Warehouse configurations in the following way:

+ +
    +
  1. +

    In the top right, select Admin Settings and then Compute, and SQL warehouses.

    +
  2. +
  3. +

    Under Data Access Properties add the following key-value pairs for +each lakeFS repository you want to access:

    +
  4. +
+ +
spark.hadoop.fs.s3a.impl shaded.databricks.org.apache.hadoop.fs.s3a.S3AFileSystem
+spark.hadoop.fs.s3a.bucket.example-repo.access.key AKIAIOSFODNN7EXAMPLE // The access key to your lakeFS server
+spark.hadoop.fs.s3a.bucket.example-repo.secret.key wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY // The secret key to your lakeFS server
+spark.hadoop.fs.s3a.bucket.example-repo.endpoint https://example-org.us-east-1.lakefscloud.io // The endpoint of your lakeFS server
+spark.hadoop.fs.s3a.bucket.example-repo.path.style.access true               
+
+ +
    +
  1. Changes are applied automatically after the SQL Warehouse restarts.
  2. +
  3. You can now use the lakeFS S3-compatible API with your SQL Warehouse, e.g.:
  4. +
+ +
SELECT * FROM delta.`s3a://example-repo/main/datasets/delta-table/` LIMIT 100
+
+

+ + + ⚠️ Experimental: Pre-signed mode for S3A + + +

+ + +

In Hadoop 3.1.4 version and above (as tested using our lakeFS Hadoop FS), it is possible to use pre-signed URLs as return values from the lakeFS S3 Gateway.

+ +

This has the immediate benefit of reducing the amount of traffic that has to go through the lakeFS server thus improving IO performance. +To read more about pre-signed URLs, see this guide.

+ +

Here’s an example Spark configuration to enable this support:

+ +
spark.hadoop.fs.s3a.impl shaded.databricks.org.apache.hadoop.fs.s3a.S3AFileSystem
+spark.hadoop.fs.s3a.bucket.example-repo.access.key AKIAIOSFODNN7EXAMPLE // The access key to your lakeFS server
+spark.hadoop.fs.s3a.bucket.example-repo.secret.key wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY // The secret key to your lakeFS server
+spark.hadoop.fs.s3a.path.style.access true
+spark.hadoop.fs.s3a.bucket.example-repo.signing-algorithm QueryStringSignerType
+spark.hadoop.fs.s3a.bucket.example-repo.user.agent.prefix s3RedirectionSupport
+
+ +

user.agent.prefix should contain the string s3RedirectionSupport but does not have to match the string exactly.

+ +

Once configured, requests will include the string s3RedirectionSupport in the User-Agent HTTP header sent with GetObject requests, resulting in lakeFS responding with a pre-signed URL. +Setting the signing-algorithm to QueryStringSignerType is required to stop S3A from signing a pre-signed URL, since the existence of more than one signature method will return an error from S3.

+ +

ℹ This feature requires a lakeFS server of version >1.18.0

+

+ + + lakeFS Hadoop FileSystem + + +

+ + +

If you’re using lakeFS on top of S3, this mode will enhance your application’s performance. +In this mode, Spark will read and write objects directly from S3, reducing the load on the lakeFS server. +It will still access the lakeFS server for metadata operations.

+ +

After configuring the lakeFS Hadoop FileSystem below, use URIs of the form lakefs://example-repo/ref/path/to/data to +interact with your data on lakeFS.

+

+ + + Installation + + +

+ + +
+ +
+ +

Add the package to your spark-submit command:

+ +
  --packages io.lakefs:hadoop-lakefs-assembly:0.2.4
+
+ +
+
+

In your cluster settings, under the Libraries tab, add the following Maven package:

+ +
io.lakefs:hadoop-lakefs-assembly:0.2.4
+
+ +

Once installed, it should look something like this:

+ +

Databricks - Adding the lakeFS client Jar

+ +
+
+ +

Add the package to your pyspark or spark-submit command:

+ +
  --packages io.lakefs:hadoop-lakefs-assembly:0.2.4
+
+ +

Add the configuration to access the S3 bucket used by lakeFS to your pyspark or spark-submit command or add this configuration at the Cloudera cluster level (see below):

+ +
  --conf spark.yarn.access.hadoopFileSystems=s3a://bucket-name
+
+ +

Add the configuration to access the S3 bucket used by lakeFS at the Cloudera cluster level:

+
    +
  1. Log in to the CDP (Cloudera Data Platform) web interface.
  2. +
  3. From the CDP home screen, click the Management Console icon.
  4. +
  5. In the Management Console, select Data Hub Clusters from the navigation pane.
  6. +
  7. Select the cluster you want to configure. Click on CM-UI link under Services: + Cloudera - Management Console
  8. +
  9. In Cloudera Manager web interface, click on Clusters from the navigation pane and click on spark_on_yarn option: + Cloudera - Cloudera Manager
  10. +
  11. Click on Configuration tab and search for spark.yarn.access.hadoopFileSystems in the search box: + Cloudera - spark_on_yarn
  12. +
  13. Add S3 bucket used by lakeFS s3a://bucket-name in the spark.yarn.access.hadoopFileSystems list: + Cloudera - hadoopFileSystems
  14. +
+
+
+

+ + + Configuration + + +

+ + +

Set the fs.lakefs.* Hadoop configurations to point to your lakeFS installation:

+
    +
  • fs.lakefs.impl: io.lakefs.LakeFSFileSystem
  • +
  • fs.lakefs.access.key: lakeFS access key
  • +
  • fs.lakefs.secret.key: lakeFS secret key
  • +
  • fs.lakefs.endpoint: lakeFS API URL (e.g. https://example-org.us-east-1.lakefscloud.io/api/v1)
  • +
+ +

Configure the lakeFS client to use a temporary token instead of static credentials:

+ +
    +
  • fs.lakefs.auth.provider: The default is basic_auth with fs.lakefs.access.key and fs.lakefs.secret.key for basic authentication. +Can be set to io.lakefs.auth.TemporaryAWSCredentialsLakeFSTokenProvider for using temporary AWS credentials, you can read more about it here.
  • +
+ +

When using io.lakefs.auth.TemporaryAWSCredentialsLakeFSTokenProvider as the auth provider the following configuration are relevant:

+ +
    +
  • fs.lakefs.token.aws.access.key: AWS assumed role access key
  • +
  • fs.lakefs.token.aws.secret.key: AWS assumed role secret key
  • +
  • fs.lakefs.token.aws.session.token: AWS assumed role temporary session token
  • +
  • fs.lakefs.token.aws.sts.endpoint: AWS STS regional endpoint for generated the presigned-url (i.e https://sts.us-west-2.amazonaws.com)
  • +
  • fs.lakefs.token.aws.sts.duration_seconds: Optional, the duration in seconds for the initial identity token (default is 60)
  • +
  • fs.lakefs.token.duration_seconds: Optional, the duration in seconds for the lakeFS token (default is set in the lakeFS configuration auth.login_duration)
  • +
  • fs.lakefs.token.sts.additional_headers: Optional, comma separated list of header:value to attach when generating presigned sts request. Default is X-Lakefs-Server-ID:fs.lakefs.endpoint.
  • +
+ +

Configure the S3A FileSystem to access your S3 storage, for example using the fs.s3a.* configurations (these are not your lakeFS credentials):

+ +
    +
  • fs.s3a.access.key: AWS S3 access key
  • +
  • fs.s3a.secret.key: AWS S3 secret key
  • +
+ +

Here are some configuration examples:

+
+ +
+
spark-shell --conf spark.hadoop.fs.s3a.access.key='AKIAIOSFODNN7EXAMPLE' \
+              --conf spark.hadoop.fs.s3a.secret.key='wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY' \
+              --conf spark.hadoop.fs.s3a.endpoint='https://s3.eu-central-1.amazonaws.com' \
+              --conf spark.hadoop.fs.lakefs.impl=io.lakefs.LakeFSFileSystem \
+              --conf spark.hadoop.fs.lakefs.access.key=AKIAlakefs12345EXAMPLE \
+              --conf spark.hadoop.fs.lakefs.secret.key=abc/lakefs/1234567bPxRfiCYEXAMPLEKEY \
+              --conf spark.hadoop.fs.lakefs.endpoint=https://example-org.us-east-1.lakefscloud.io/api/v1 \
+              --packages io.lakefs:hadoop-lakefs-assembly:0.2.4 \
+              io.example.ExampleClass
+
+
+
+ +
spark.sparkContext.hadoopConfiguration.set("fs.s3a.access.key", "AKIAIOSFODNN7EXAMPLE")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.secret.key", "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY")
+spark.sparkContext.hadoopConfiguration.set("fs.s3a.endpoint", "https://s3.eu-central-1.amazonaws.com")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.impl", "io.lakefs.LakeFSFileSystem")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.access.key", "AKIAlakefs12345EXAMPLE")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.endpoint", "https://example-org.us-east-1.lakefscloud.io/api/v1")
+
+
+
+ +
sc._jsc.hadoopConfiguration().set("fs.s3a.access.key", "AKIAIOSFODNN7EXAMPLE")
+sc._jsc.hadoopConfiguration().set("fs.s3a.secret.key", "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY")
+sc._jsc.hadoopConfiguration().set("fs.s3a.endpoint", "https://s3.eu-central-1.amazonaws.com")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.impl", "io.lakefs.LakeFSFileSystem")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.access.key", "AKIAlakefs12345EXAMPLE")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.endpoint", "https://example-org.us-east-1.lakefscloud.io/api/v1")
+
+
+
+ +

Make sure that you load the lakeFS FileSystem into Spark by running it with --packages or --jars, +and then add these into a configuration file, e.g., $SPARK_HOME/conf/hdfs-site.xml:

+ +
<?xml version="1.0"?>
+<configuration>
+    <property>
+        <name>fs.s3a.access.key</name>
+        <value>AKIAIOSFODNN7EXAMPLE</value>
+    </property>
+    <property>
+            <name>fs.s3a.secret.key</name>
+            <value>wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.s3a.endpoint</name>
+        <value>https://s3.eu-central-1.amazonaws.com</value>
+    </property>
+    <property>
+        <name>fs.lakefs.impl</name>
+        <value>io.lakefs.LakeFSFileSystem</value>
+    </property>
+    <property>
+        <name>fs.lakefs.access.key</name>
+        <value>AKIAlakefs12345EXAMPLE</value>
+    </property>
+    <property>
+        <name>fs.lakefs.secret.key</name>
+        <value>abc/lakefs/1234567bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.lakefs.endpoint</name>
+        <value>https://example-org.us-east-1.lakefscloud.io/api/v1</value>
+    </property>
+</configuration>
+
+
+
+ +

Add the following the cluster’s configuration under Configuration ➡️ Advanced options:

+ +
spark.hadoop.fs.lakefs.impl io.lakefs.LakeFSFileSystem
+spark.hadoop.fs.lakefs.access.key AKIAlakefs12345EXAMPLE
+spark.hadoop.fs.lakefs.secret.key abc/lakefs/1234567bPxRfiCYEXAMPLEKEY
+spark.hadoop.fs.s3a.access.key AKIAIOSFODNN7EXAMPLE
+spark.hadoop.fs.s3a.secret.key wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+spark.hadoop.fs.s3a.impl shaded.databricks.org.apache.hadoop.fs.s3a.S3AFileSystem
+spark.hadoop.fs.lakefs.endpoint https://example-org.us-east-1.lakefscloud.io/api/v1
+
+ +

Alternatively, follow this step by step Databricks integration tutorial, including lakeFS Hadoop File System, Python client and lakeFS SPARK client.

+
+
+ +

⚠️ If your bucket is on a region other than us-east-1, you may also need to configure fs.s3a.endpoint with the correct region. +Amazon provides S3 endpoints you can use.

+

+ + + Usage with TemporaryAWSCredentialsLakeFSTokenProvider + + +

+ + +

An initial setup is required - you must have AWS Auth configured with lakeFS. +The TemporaryAWSCredentialsLakeFSTokenProvider depends on the caller to provide AWS credentials (e.g Assumed Role Key,Secret and Token) as input to the lakeFS client.

+ +

⚠️ Configure sts.endpoint with a valid sts regional service endpoint and it must be be equal to the region that is used for authentication first place. The only exception is us-east-1 which is the default region for STS.

+ +

⚠️ Using the current provider the lakeFS token will not renew upon expiry and the user will need to re-authenticate.

+ +

PySpark example using TemporaryAWSCredentialsLakeFSTokenProvider with boto3 and AWS session credentials:

+ +
import boto3 
+
+session = boto3.session.Session()
+
+# AWS credentials used s3a to access lakeFS bucket
+sc._jsc.hadoopConfiguration().set("fs.s3a.access.key", "AKIAIOSFODNN7EXAMPLE")
+sc._jsc.hadoopConfiguration().set("fs.s3a.secret.key", "wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY")
+sc._jsc.hadoopConfiguration().set("fs.s3a.endpoint", "https://s3.us-west-2.amazonaws.com")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.impl", "io.lakefs.LakeFSFileSystem")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.endpoint", "https://example-org.us-west-2.lakefscloud.io/api/v1")
+sc._jsc.hadoopConfiguration().set("spark.hadoop.fs.s3a.path.style.access", "true")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.auth.provider", "io.lakefs.auth.TemporaryAWSCredentialsLakeFSTokenProvider")
+# AWS tempporary session credentials to use with lakeFS
+sc._jsc.hadoopConfiguration().set("fs.lakefs.token.aws.access.key", session.get_credentials().access_key)
+sc._jsc.hadoopConfiguration().set("fs.lakefs.token.aws.secret.key", session.get_credentials().secret_key)
+sc._jsc.hadoopConfiguration().set("fs.lakefs.token.aws.session.token", session.get_credentials().token)
+sc._jsc.hadoopConfiguration().set("fs.lakefs.token.aws.sts.endpoint", "https://sts.us-west-2.amazonaws.com")
+
+

+ + + Usage + + +

+ + +

Hadoop FileSystem paths use the lakefs:// protocol, with paths taking the form lakefs://<repository>/<ref>/path/to/object. +<ref> can be a branch, tag, or commit ID in lakeFS. +Here’s an example for reading a Parquet file from lakeFS to a Spark DataFrame:

+ +
val repo = "example-repo"
+val branch = "main"
+val df = spark.read.parquet(s"lakefs://${repo}/${branch}/example-path/example-file.parquet")
+
+ +

Here’s how to write some results back to a lakeFS path:

+ +
df.write.partitionBy("example-column").parquet(s"lakefs://${repo}/${branch}/output-path/")
+
+ +

The data is now created in lakeFS as new changes in your branch. You can now commit these changes or revert them.

+

+ + + Hadoop FileSystem in Presigned mode + + +

+ + +

Available starting version 0.1.13 of the FileSystem

+ +

In this mode, the lakeFS server is responsible for authenticating with your storage. +The client will still perform data operations directly on the storage. +To do so, it will use pre-signed storage URLs provided by the lakeFS server.

+ +

When using this mode, you don’t need to configure the client with access to your storage:

+ +
+ +
+
spark-shell --conf spark.hadoop.fs.lakefs.access.mode=presigned \
+              --conf spark.hadoop.fs.lakefs.impl=io.lakefs.LakeFSFileSystem \
+              --conf spark.hadoop.fs.lakefs.access.key=AKIAlakefs12345EXAMPLE \
+              --conf spark.hadoop.fs.lakefs.secret.key=abc/lakefs/1234567bPxRfiCYEXAMPLEKEY \
+              --conf spark.hadoop.fs.lakefs.endpoint=https://example-org.us-east-1.lakefscloud.io/api/v1 \
+              --packages io.lakefs:hadoop-lakefs-assembly:0.2.4
+
+
+
+ +
spark.sparkContext.hadoopConfiguration.set("fs.lakefs.access.mode", "presigned")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.impl", "io.lakefs.LakeFSFileSystem")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.access.key", "AKIAlakefs12345EXAMPLE")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+spark.sparkContext.hadoopConfiguration.set("fs.lakefs.endpoint", "https://example-org.us-east-1.lakefscloud.io/api/v1")
+
+
+
+ +
sc._jsc.hadoopConfiguration().set("fs.lakefs.access.mode", "presigned")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.impl", "io.lakefs.LakeFSFileSystem")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.access.key", "AKIAlakefs12345EXAMPLE")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.secret.key", "abc/lakefs/1234567bPxRfiCYEXAMPLEKEY")
+sc._jsc.hadoopConfiguration().set("fs.lakefs.endpoint", "https://example-org.us-east-1.lakefscloud.io/api/v1")
+
+
+
+ +

Make sure that you load the lakeFS FileSystem into Spark by running it with --packages or --jars, +and then add these into a configuration file, e.g., $SPARK_HOME/conf/hdfs-site.xml:

+ +
<?xml version="1.0"?>
+<configuration>
+    <property>
+        <name>fs.lakefs.access.mode</name>
+        <value>presigned</value>
+    </property>
+    <property>
+        <name>fs.lakefs.impl</name>
+        <value>io.lakefs.LakeFSFileSystem</value>
+    </property>
+    <property>
+        <name>fs.lakefs.access.key</name>
+        <value>AKIAlakefs12345EXAMPLE</value>
+    </property>
+    <property>
+        <name>fs.lakefs.secret.key</name>
+        <value>abc/lakefs/1234567bPxRfiCYEXAMPLEKEY</value>
+    </property>
+    <property>
+        <name>fs.lakefs.endpoint</name>
+        <value>https://example-org.us-east-1.lakefscloud.io/api/v1</value>
+    </property>
+</configuration>
+
+
+
+ +

Add the following the cluster’s configuration under Configuration ➡️ Advanced options:

+ +
spark.hadoop.fs.lakefs.access.mode presigned
+spark.hadoop.fs.lakefs.impl io.lakefs.LakeFSFileSystem
+spark.hadoop.fs.lakefs.access.key AKIAlakefs12345EXAMPLE
+spark.hadoop.fs.lakefs.secret.key abc/lakefs/1234567bPxRfiCYEXAMPLEKEY
+spark.hadoop.fs.lakefs.endpoint https://example-org.us-east-1.lakefscloud.io/api/v1
+
+
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/unity-catalog.html b/v1.46/integrations/unity-catalog.html new file mode 100644 index 000000000..9dc3cfe18 --- /dev/null +++ b/v1.46/integrations/unity-catalog.html @@ -0,0 +1,999 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Unity Catalog | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using lakeFS with the Unity Catalog + + +

+ + + +

+ + + Overview + + +

+ + +

Databricks Unity Catalog serves as a centralized data governance platform for your data lakes. +Through the Unity Catalog, you can search for and locate data assets across workspaces via a unified catalog. +Leveraging the external tables feature within Unity Catalog, you can register a Delta Lake table exported from lakeFS and +access it through the unified catalog. +The subsequent step-by-step guide will lead you through the process of configuring a Lua hook +that exports Delta Lake tables from lakeFS, and subsequently registers them in Unity Catalog.

+ +
+

Currently, Unity Catalog export feature exclusively supports AWS S3 and Azure ADLS Gen2 as the underlying storage solution. It’s planned to support other cloud providers soon.

+
+ +

Reference Guide: lakeFS + Unity Catalog Integration: Step-by-Step Tutorial

+

+ + + Prerequisites + + +

+ + +

Before starting, ensure you have the following:

+ +
    +
  1. Access to Unity Catalog
  2. +
  3. An active lakeFS installation with S3 as the backing storage, and a repository in this installation.
  4. +
  5. A Databricks SQL warehouse.
  6. +
  7. AWS Credentials with S3 access.
  8. +
  9. lakeFS credentials with access to your Delta Tables.
  10. +
+ +
+

Supported from lakeFS v1.4.0

+
+

+ + + Databricks authentication + + +

+ + +

Given that the hook will ultimately register a table in Unity Catalog, authentication with Databricks is imperative. +Make sure that:

+ +
    +
  1. You have a Databricks Service Principal.
  2. +
  3. The Service principal has token usage permissions, +and an associated token +configured.
  4. +
  5. The service principal has the Service principal: Manager privilege over itself (Workspace: Admin console -> Service principals -> <service principal> -> Permissions -> Grant access (<service principal>: +Service principal: Manager), with Workspace access and Databricks SQL access checked (Admin console -> Service principals -> <service principal> -> Configurations).
  6. +
  7. Your SQL warehouse allows the service principal to use it (SQL Warehouses -> <SQL warehouse> -> Permissions -> <service principal>: Can use).
  8. +
  9. The catalog grants the USE CATALOG, USE SCHEMA, CREATE SCHEMA permissions to the service principal(Catalog -> <catalog name> -> Permissions -> Grant -> <service principal>: USE CATALOG, USE SCHEMA, CREATE SCHEMA).
  10. +
  11. You have an External Location configured, and the service principal has the CREATE EXTERNAL TABLE permission over it (Catalog -> External Data -> External Locations -> Create location).
  12. +
+

+ + + Guide + + +

+ +

+ + + Table descriptor definition + + +

+ + +

To guide the Unity Catalog exporter in configuring the table in the catalog, define its properties in the Delta Lake table descriptor. +The table descriptor should include (at minimum) the following fields:

+
    +
  1. name: The table name.
  2. +
  3. type: Should be delta.
  4. +
  5. catalog: The name of the catalog in which the table will be created.
  6. +
  7. path: The path in lakeFS (starting from the root of the branch) in which the Delta Lake table’s data is found.
  8. +
+ +

Let’s define the table descriptor and upload it to lakeFS:

+ +

Save the following as famous-people-td.yaml:

+ +
---
+name: famous_people
+type: delta
+catalog: my-catalog-name
+path: tables/famous-people
+
+ +
+

It’s recommended to create a Unity catalog with the same name as your repository

+
+ +

Upload the table descriptor to _lakefs_tables/famous-people-td.yaml and commit:

+ +
lakectl fs upload lakefs://repo/main/_lakefs_tables/famous-people-td.yaml -s ./famous-people-td.yaml && \
+lakectl commit lakefs://repo/main -m "add famous people table descriptor"
+
+

+ + + Write some data + + +

+ + +

Insert data into the table path, using your preferred method (e.g. Spark), and commit upon completion.

+ +

We shall use Spark and lakeFS’s S3 gateway to write some data as a Delta table:

+
pyspark --packages "io.delta:delta-spark_2.12:3.0.0,org.apache.hadoop:hadoop-aws:3.3.4,com.amazonaws:aws-java-sdk-bundle:1.12.262" \
+  --conf spark.sql.extensions=io.delta.sql.DeltaSparkSessionExtension \
+  --conf spark.sql.catalog.spark_catalog=org.apache.spark.sql.delta.catalog.DeltaCatalog \
+  --conf spark.hadoop.fs.s3a.aws.credentials.provider='org.apache.hadoop.fs.s3a.SimpleAWSCredentialsProvider' \
+  --conf spark.hadoop.fs.s3a.endpoint='<LAKEFS_SERVER_URL>' \
+  --conf spark.hadoop.fs.s3a.access.key='<LAKEFS_ACCESS_KEY>' \
+  --conf spark.hadoop.fs.s3a.secret.key='<LAKEFS_SECRET_ACCESS_KEY>' \
+  --conf spark.hadoop.fs.s3a.path.style.access=true
+
+ +
data = [
+   ('James','Bond','England','intelligence'),
+   ('Robbie','Williams','England','music'),
+   ('Hulk','Hogan','USA','entertainment'),
+   ('Mister','T','USA','entertainment'),
+   ('Rafael','Nadal','Spain','professional athlete'),
+   ('Paul','Haver','Belgium','music'),
+]
+columns = ["firstname","lastname","country","category"]
+df = spark.createDataFrame(data=data, schema = columns)
+df.write.format("delta").mode("overwrite").partitionBy("category", "country").save("s3a://repo/main/tables/famous-people")
+
+

+ + + The Unity Catalog exporter script + + +

+ + +
+

For code references check delta_exporter and +unity_exporter docs.

+
+ +

Create unity_exporter.lua:

+ +
local aws = require("aws")
+local formats = require("formats")
+local databricks = require("databricks")
+local delta_export = require("lakefs/catalogexport/delta_exporter")
+local unity_export = require("lakefs/catalogexport/unity_exporter")
+
+local sc = aws.s3_client(args.aws.access_key_id, args.aws.secret_access_key, args.aws.region)
+
+-- Export Delta Lake tables export:
+local delta_client = formats.delta_client(args.lakefs.access_key_id, args.lakefs.secret_access_key, args.aws.region)
+local delta_table_details = delta_export.export_delta_log(action, args.table_defs, sc.put_object, delta_client, "_lakefs_tables")
+
+-- Register the exported table in Unity Catalog:
+local databricks_client = databricks.client(args.databricks_host, args.databricks_token)
+local registration_statuses = unity_export.register_tables(action, "_lakefs_tables", delta_table_details, databricks_client, args.warehouse_id)
+
+for t, status in pairs(registration_statuses) do
+    print("Unity catalog registration for table \"" .. t .. "\" completed with commit schema status : " .. status .. "\n")
+end
+
+ +

Upload the lua script to the main branch under scripts/unity_exporter.lua and commit:

+ +
lakectl fs upload lakefs://repo/main/scripts/unity_exporter.lua -s ./unity_exporter.lua && \
+lakectl commit lakefs://repo/main -m "upload unity exporter script"
+
+

+ + + Action configuration + + +

+ + +

Define an action configuration that will run the above script after a commit is completed (post-commit) over the main branch.

+ +

Create unity_exports_action.yaml:

+ +
---
+name: unity_exports
+on:
+  post-commit:
+     branches: ["main"]
+hooks:
+  - id: unity_export
+    type: lua
+    properties:
+      script_path: scripts/unity_exporter.lua
+      args:
+        aws:
+          access_key_id: <AWS_ACCESS_KEY_ID>
+          secret_access_key: <AWS_SECRET_ACCESS_KEY>
+          region: <AWS_REGION>
+        lakefs: # provide credentials of a user that has access to the script and Delta Table
+          access_key_id: <LAKEFS_ACCESS_KEY_ID> 
+          secret_access_key: <LAKEFS_SECRET_ACCESS_KEY>
+        table_defs: # an array of table descriptors used to be defined in Unity Catalog
+          - famous-people-td
+        databricks_host: <DATABRICKS_HOST_URL>
+        databricks_token: <DATABRICKS_SERVICE_PRINCIPAL_TOKEN>
+        warehouse_id: <WAREHOUSE_ID>
+
+ +

Upload the action configurations to _lakefs_actions/unity_exports_action.yaml and commit:

+ +
+

Once the commit will finish its run, the action will start running since we’ve configured it to run on post-commit +events on the main branch.

+
+ +
lakectl fs upload lakefs://repo/main/_lakefs_actions/unity_exports_action.yaml -s ./unity_exports_action.yaml && \
+lakectl commit lakefs://repo/main -m "upload action and run it"
+
+ +

The action has run and exported the famous_people Delta Lake table to the repo’s storage namespace, and has register +the table as an external table in Unity Catalog under the catalog my-catalog-name, schema main (as the branch’s name) and +table name famous_people: my-catalog-name.main.famous_people.

+ +

Hooks log result in lakeFS UI

+

+ + + Databricks Integration + + +

+ + +

After registering the table in Unity, you can leverage your preferred method to query the data +from the exported table under my-catalog-name.main.famous_people, and view it in the Databricks’s Catalog Explorer, or +retrieve it using the Databricks CLI with the following command:

+
databricks tables get my-catalog-name.main.famous_people
+
+ +

Unity Catalog Explorer view

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/integrations/unity_catalog.html b/v1.46/integrations/unity_catalog.html new file mode 100644 index 000000000..eecb9f880 --- /dev/null +++ b/v1.46/integrations/unity_catalog.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/integrations/vertex_ai.html b/v1.46/integrations/vertex_ai.html new file mode 100644 index 000000000..5e0531dc4 --- /dev/null +++ b/v1.46/integrations/vertex_ai.html @@ -0,0 +1,900 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Vertex AI | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Using Vertex AI with lakeFS + + +

+ + +

Vertex AI lets Google Cloud users Build, deploy, and scale machine learning (ML) models faster, with fully managed ML tools for any use case.

+ +

lakeFS Works with Vertex AI by allowing users to create repositories on GCS Buckets, then use either the Dataset API to create managed Datasets on top of lakeFS version, or by automatically exporting lakeFS object versions in a way readable by Cloud Storage Mounts.

+ + +

+ + + Using lakeFS with Vertex Managed Datasets + + +

+ + +

Vertex’s ImageDataset and VideoDataset allow creating a dataset by importing a CSV file from gcs (see gcs_source).

+ +

This CSV file contains GCS addresses of image files and their corresponding labels.

+ +

Since the lakeFS API supports exporting the underlying GCS address of versioned objects, we can generate such a CSV file when creating the dataset:

+ +
#!/usr/bin/env python
+
+# Requirements:
+# google-cloud-aiplatform>=1.31.0
+# lakefs-client>=0.107.0
+
+import csv
+from pathlib import PosixPath
+from io import StringIO
+
+import lakefs_client
+from lakefs_client.client import LakeFSClient
+from google.cloud import storage
+from google.cloud import aiplatform
+
+# lakeFS connection details
+configuration = lakefs_client.Configuration()
+configuration.username = 'AKIAIOSFODNN7EXAMPLE'
+configuration.password = 'wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY'
+configuration.host = 'https://lakefs.example.com/'
+client = LakeFSClient(configuration)
+
+# Dataset configuration
+lakefs_repo = 'my-repository'
+lakefs_ref = 'main'
+img_dataset = 'datasets/my-images/'
+
+# Vertex configuration
+import_bucket = 'underlying-gcs-bucket'
+
+# produce import file for Vertex's SDK
+buf = StringIO()
+csv_writer = csv.writer(buf)
+has_more = True
+next_offset = ""
+while has_more:
+    files = client.objects_api.list_objects(
+        lakefs_repo, lakefs_ref, prefix=img_dataset, after=next_offset)
+    for r in files.get('results'):
+        p = PosixPath(r.path)
+        csv_writer.writerow((r.physical_address, p.parent.name))
+    has_more = files.get('pagination').get('has_more')
+    next_offset = files.get('pagination').get('next_offset')
+
+# spit out CSV
+print('generated path and labels CSV')
+buf.seek(0)
+
+# Write it to storage
+storage_client = storage.Client()
+bucket = storage_client.bucket(import_bucket)
+blob = bucket.blob(f'vertex/imports/{lakefs_repo}/{lakefs_ref}/labels.csv')
+with blob.open('w') as out:
+    out.write(buf.read())
+
+print(f'wrote csv to gs: gs://{import_bucket}/vertex/imports/{lakefs_repo}/{lakefs_ref}/labels.csv')
+
+# import in vertex, as dataset
+print('Importing dataset...')
+ds = aiplatform.ImageDataset.create(
+    display_name=f'{lakefs_repo}_{lakefs_ref}_imgs',
+    gcs_source=f'gs://{import_bucket}/vertex/imports/{lakefs_repo}/{lakefs_ref}/labels.csv',
+    import_schema_uri=aiplatform.schema.dataset.ioformat.image.single_label_classification,
+    sync=True
+)
+ds.wait()
+print(f'Done! {ds.display_name} ({ds.resource_name})')
+
+

+ + + Using lakeFS with Cloud Storage Fuse + + +

+ + +

Vertex allows using Google Cloud Storage mounted as a Fuse Filesystem as custom input for training jobs.

+ +

Instead of having to copy lakeFS files for each version we want to consume, we can create symlinks by using gcsfuse’s native symlink inodes.

+ +

This process can be fully automated by using the example gcsfuse_symlink_exporter.lua Lua hook.

+ +

Here’s what we need to do:

+ +
    +
  1. Upload the example .lua file into our lakeFS repository. For this example, we’ll put it under scripts/gcsfuse_symlink_exporter.lua.
  2. +
  3. Create a new hook definition file and upload to _lakefs_actions/export_images.yaml:
  4. +
+ +
---
+# Example hook declaration: (_lakefs_actions/export_images.yaml):
+name: export_images
+
+on:
+  post-commit:
+    branches: ["main"]
+  post-merge:
+    branches: ["main"]
+  post-create-tag:
+
+hooks:
+- id: gcsfuse_export_images
+  type: lua
+  properties:
+    script_path: scripts/export_gcs_fuse.lua  # Path to the script we uploaded in the previous step
+    args:
+      prefix: "datasets/images/"  # Path we want to export every commit
+      destination: "gs://my-bucket/exports/my-repo/"  # Where should we create the symlinks?
+      mount:
+        from: "gs://my-bucket/repos/my-repo/"  # Symlinks are to a unix-mounted file
+        to: "/gcs/my-bucket/repos/my-repo/"    #  This will ensure they point to a location that exists.
+      
+      # Should be the contents of a valid credentials.json file
+      # See: https://developers.google.com/workspace/guides/create-credentials
+      # Will be used to write the symlink files
+      gcs_credentials_json_string: |
+        {
+          "client_id": "...",
+          "client_secret": "...",
+          "refresh_token": "...",
+          "type": "..."
+        }
+
+ +

Done! On the next tag creation or update to the main branch, we’ll automatically export the lakeFS version of datasets/images/ to a mountable location.

+ +

To consume the symlink-ed files, we can read them normally from the mount:

+ +
with open('/gcs/my-bucket/exports/my-repo/branches/main/datasets/images/001.jpg') as f:
+    image_data = f.read()
+
+
+ +

Previously exported commits are also readable, if we exported them in the past:

+ +
commit_id = 'abcdef123deadbeef567'
+with open(f'/gcs/my-bucket/exports/my-repo/commits/{commit_id}/datasets/images/001.jpg') as f:
+    image_data = f.read()
+
+
+

+ + + Considerations when using lakeFS with Cloud Storage Fuse + + +

+ + +

For lakeFS paths to be readable by gcsfuse, the mount option --implicit-dirs must be specified.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/licensing.html b/v1.46/licensing.html new file mode 100644 index 000000000..41dce22e2 --- /dev/null +++ b/v1.46/licensing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/posts/deprecate-py-legacy.html b/v1.46/posts/deprecate-py-legacy.html new file mode 100644 index 000000000..c64c11b8c --- /dev/null +++ b/v1.46/posts/deprecate-py-legacy.html @@ -0,0 +1,825 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +⚠️ Deprecating lakefs-client, the legacy Python 🐍 Client SDKs | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + ⚠️ Deprecating lakefs-client, the legacy Python 🐍 Client SDKs + + +

+ + +

lakefs-client is now deprecated, and we will shortly publish its last +release to PyPI.

+

+ + + What is changing? + + +

+ + +

Ever since we released lakeFS 1.0, we have supported these two SDKs to program for lakeFS with Python:

+ + +

These clients will continue.

+ +

Before lakeFS 1.0 we would publish only:

+ + + +

We have continued to publish lakefs-client since the 1.0 release, in order +to provide continued support for existing users of this legacy Python SDK +client. But there are better alternatives, and its continued presence only +confuses our users.

+ +

Now it is time to deprecate this client, and we will stop publishing it at +or after the version 1.40 release of lakeFS.

+

+ + + FAQ + + +

+ +

+ + + What will happen to existing users of lakefs-client? + + +

+ + +

All existing uses will continue to work. Obviously we will not be pulling +these published versions because of this change. We will stop publishing +new versions of these clients.

+

+ + + how do I migrate from using lakefs-client to lakefs-sdk? + + +

+ + +

Migration is covered by Migrating to 1.0. In most +cases the changes are quite simple.

+

+ + + Will new versions of lakefs-client be published? + + +

+ + +

No.

+ +

Version 1.40 will be the last release of lakeFS to include a new version of +the legacy client. In order to use any new features of the lakeFS API, +you will need to upgrade your programs to use lakefs-sdk.

+

+ + + Will programs using old versions of lakefs-client continue to work with new versions of lakeFS? + + +

+ + +

Yes.

+ +

A program that uses any 1.x version of lakefs-client still uses 1.x +versions of lakeFS APIs. Its usage of those APIs is still covered by the +lakeFS interface guarantees, so it will continue to work with future 1.x +releases.

+

+ + + What about the legacy Java SDK generated client? + + +

+ + +

That is also going away. However we see much less usage of it.

+

+ + + Where can I ask another question not covered here? + + +

+ + +

As always, our Slack htttps://lakefs.io/slack is the best place to interact +with the lakeFS community! Try asking on our #dev channel

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/posts/index.html b/v1.46/posts/index.html new file mode 100644 index 000000000..fb9469baa --- /dev/null +++ b/v1.46/posts/index.html @@ -0,0 +1,1778 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakeFS Documentation | Open Source Data Version Control for Data Lakes and Data Lakehouses + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ + + + + + +
+
+

Table of contents

+ +
+ + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/posts/security_update.html b/v1.46/posts/security_update.html new file mode 100644 index 000000000..32c1dba2d --- /dev/null +++ b/v1.46/posts/security_update.html @@ -0,0 +1,762 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Security update for the lakeFS project | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Security update for the lakeFS project + + +

+ + +

Trust is our core value, and security topics are never taken lightly.

+ +

We heard you, and after many conversations, we decided to move from decoupled security authentication and access control features to enable you to plug your own authentication and security mechanism.

+ +

We understand that focusing on building lakeFS to be the best data version control tool is the open-source project north star. +Consequently, we decided to change the architecture to a pluggable one which enables you to choose your preference without being dependent on the lakeFS solution.

+

+ + + What’s changing? + + +

+ + +

With lakeFS v0.91.0 Role-based access control and SSO (OIDC, LDAP) authentication are deprecated due to this architecture change. +Within the lakeFS UI, you will see a deprecation notice on the administration screens when trying to update and create policies.

+ +

In coming versions, we plan on making authorization and authentication pluggable, while bundling a simpler reference implementation into the core of lakeFS. +This implementation will include basic identity management with built-in users and groups, as well as a simplified authorization mechanism based on ACLs.

+ +

If you currently rely on any of the deprecated features, please contact us - we’ll do our best to support you in the transition.

+

+ + + Why are we making this architectural change? + + +

+ + +

This change is necessary as it lets you decide on access control and authentication that best fits your security needs. It matters since one can ensure security only if one controls the full environment.

+ +

lakeFS users will be able to develop and use existing identity providers and wrap lakeFS with their own authorization logic (if they so desire) - to meet their specific security needs.

+

+ + + How can I ensure security? + + +

+ + +

Now that the architecture is pluggable, you can use the solution of your choice.

+ +

Depending on your needs, you might want to take a look at lakeFS Cloud, which enables us to provide a holistic, secure solution that is SOC2 compliant and provides security guarantees.

+ +

lakeFS Cloud will continue to support and maintain more advanced authentication methods such as SAML and OIDC, and granular access control in the form of role-based access control policies. Additionally, lakeFS Cloud also provides a full, queryable auditing log to help administrators gain full visibility on data access within their lakeFS environment.


+ +

Trust is our core value, and security topics are never taken lightly.

+ +

Even though this decision was difficult, it was necessary to continue building a solid foundation for our data version control.

+ +

We remain committed to making lakeFS the best, most scalable, open source version control system for data practitioners, while not compromising on its security.

+ +

Oz & Einat

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/project/code-migrate-1.0-sdk.html b/v1.46/project/code-migrate-1.0-sdk.html new file mode 100644 index 000000000..6258e44d7 --- /dev/null +++ b/v1.46/project/code-migrate-1.0-sdk.html @@ -0,0 +1,933 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Migrating to 1.0 | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS 1.0 - Code Migration Guide + + +

+ + +

Version 1.0.0 promises API and SDK stability. By “API” we mean any access to a lakeFS REST endpoint. By “SDK” we mean auto-generated lakeFS clients: lakefs-sdk for Python and io.lakefs:sdk for Java. This guide details the steps to allow you to upgrade your code to enjoy this stability.

+ +

Avoid using APIs and SDKs labeled as experimental, internal, or deprecated. If you must use them, be prepared to adjust your application to align with any lakeFS server updates.

+ +

Your software developed without such APIs should be compatible with all minor version updates of the lakeFS server from the version you originally developed with.

+ +

If you rely on a publicly released API and SDK, it will adhere to semantic versioning. Transitioning your application to a minor SDK version update should be smooth.

+ +

The operation names and tags from the api/swagger.yml specification might differ based on the SDK or coding language in use.

+

+ + + Deleted API Operations + + +

+ + +

The following API operations have been removed:

+ +
    +
  • updatePassword
  • +
  • forgotPassword
  • +
  • logBranchCommits
  • +
  • expandTemplate
  • +
  • createMetaRange
  • +
  • ingestRange
  • +
  • updateBranchToken
  • +
+

+ + + Internal API Operations + + +

+ + +

The following operations are for internal use only and should not be used in your application code. Some deprecated operations have alternatives provided.

+ +
    +
  • setupCommPrefs
  • +
  • getSetupState
  • +
  • setup
  • +
  • getAuthCapabilities
  • +
  • uploadObjectPreflight
  • +
  • setGarbageCollectionRulesPreflight
  • +
  • createBranchProtectionRulePreflight
  • +
  • postStatsEvents
  • +
  • dumpRefs (will be replaced with a long-running API later)
  • +
  • restoreRefs (will be replaced with a long-running API later)
  • +
  • createSymlinkFile (Deprecated)
  • +
  • getStorageConfig (Deprecated. Alternative: getConfig)
  • +
  • getLakeFSVersion (Deprecated. Alternative: getConfig)
  • +
  • stageObject (Deprecated. Alternatives: get/link physical address or import)
  • +
  • internalDeleteBranchProtectionRule (Deprecated. Temporary backward support. Alternative: setBranchProtectionRules)
  • +
  • internalCreateBranchProtectionRule (Deprecated. Temporary backward support. Alternative: setBranchProtectionRules)
  • +
  • internalGetBranchProtectionRule (Deprecated. Temporary backward support. Alternative: getBranchProtectionRules)
  • +
  • internalDeleteGarbageCollectionRules (Deprecated. Temporary backward support. Alternative: deleteGCRules)
  • +
  • internalSetGarbageCollectionRules (Deprecated. Temporary backward support. Alternative: setGCRules)
  • +
  • internalGetGarbageCollectionRules (Deprecated. Temporary backward support. Alternative: getGCRules)
  • +
  • prepareGarbageCollectionCommits
  • +
  • getGarbageCollectionConfig
  • +
+

+ + + New/Updated API Operations + + +

+ + +

Here are the newly added or updated operations:

+ +
    +
  • getConfig (Retrieve lakeFS version and storage info)
  • +
  • setBranchProtectionRules (Route updated)
  • +
  • getBranchProtectionRules (Route updated)
  • +
  • getGCRules (New route introduced)
  • +
  • setGCRules (New route introduced)
  • +
  • deleteGCRules (New route introduced)
  • +
  • importStatus (Response structure updated: ‘ImportStatusResp’ to ‘ImportStatus’)
  • +
  • uploadObject (Parameters ‘if-none-match’ and ‘storageClass’ are now deprecated)
  • +
  • prepareGarbageCollectionCommits (Request body removed)
  • +
  • getOtfDiffs & otfDiff (Removed from ‘otf diff’ tag; retained in ‘experimental’ tag)
  • +
+

+ + + Migrating SDK Code for Java and JVM-based Languages + + +

+ +

+ + + Introduction + + +

+ + +

If you are using the lakeFS client for Java or for any other JVM-based language, be aware that the current package is not stable with respect to minor version upgrades. Transitioning from io.lakefs:lakefs-client to io.lakefs:sdk will necessitate rewriting your API calls to fit the new design paradigm.

+

+ + + Problem with the Old Style + + +

+ + +

Previously, API calls required developers to pass all parameters, including optional ones, in a single function call. As demonstrated in this older style:

+ +
ObjectStats objectStat = objectsApi.statObject(
+    objectLoc.getRepository(), objectLoc.getRef(), objectLoc.getPath(),
+    false, false);
+
+ +

This method posed a couple of challenges:

+ +
    +
  1. Inflexibility with Upgrades: If an optional parameter were introduced in newer versions, existing code would fail to compile.
  2. +
  3. Maintenance Difficulty: Long argument lists can be challenging to manage and understand, leading to potential mistakes and readability issues.
  4. +
+ +

Adopting the Fluent Style

+ +

In the revised SDK, API calls adopt a fluent style, making the code more modular and adaptive to changes.

+ +

Here’s an example of the new style:

+ +
ObjectStats objectStat = objectsApi
+    .statObject(
+        objectLoc.getRepository(), objectLoc.getRef(), objectLoc.getPath()
+    )
+    .userMetadata(true)
+    .execute();
+
+

+ + + Here’s a breakdown of the changes: + + +

+ + +
    +
  1. Initial Function Call: Begin by invoking the desired function with all required parameters.
  2. +
  3. Modifying Optional Parameters: Chain any modifications to optional parameters after the initial function. For instance, userMetadata is changed in the example above.
  4. +
  5. Unused Optional Parameters: You can safely ignore these. For instance, this code ignores the presign optional parameter because it never uses it.
  6. +
  7. Execution: Complete the call with the .execute() method.
  8. +
+ +

This new design offers several advantages:

+ +
    +
  • Compatibility with Upgrades: When a new optional parameter is introduced, existing code will use its default value, preserving compatibility with minor server version upgrades.
  • +
  • Improved Readability: The fluent style makes it evident which parameters are required and which ones are optional, enhancing code clarity.
  • +
+ +

When migrating your code, ensure you refactor all your API calls to adopt the new fluent style. This ensures that your application remains maintainable and is safeguarded against potential issues arising from minor SDK version upgrades.

+ +

For an illustrative example of the transition between styles, you can view the changes made in this pull request: lakeFS pull request #6529.

+

+ + + Migrating SDK Code for Python + + +

+ +

+ + + Introduction + + +

+ + +

If you are using the lakeFS client for Python, be aware that the current package is not stable with respect to minor version upgrades. Transitioning from lakefs-client to lakefs-sdk will necessitate rewriting your API calls.

+

+ + + Here’s a breakdown of the changes: + + +

+ + +
    +
  1. Modules change +
      +
    • The previous model module was renamed to models, meaning that lakefs_client.model imports should be replaced with lakefs_sdk.models imports.
    • +
    • The apis module in lakefs_client is deprecated and no longer supported. To migrate to the new api module in lakefs_sdk, you should replace all imports of lakefs_client.apis with imports of lakefs_sdk.api. We still recommend using the lakefs_sdk.LakeFSClient class instead of using the api module directly. The LakeFSClient class provides a higher-level interface to the LakeFS API and makes it easier to use LakeFS in your applications.
    • +
    +
  2. +
  3. upload_object API call: The content parameter value passed to the objects_api.upload_object method call should be either a string containing the path to the uploaded file, or bytes of data to be uploaded.
  4. +
  5. get_object API call: The return value of client.get_object(...) is a bytearray containing the content of the object.
  6. +
  7. **client.{operation}_api**: The lakefs-client package’s LakeFSClient class’s deprecation-marked operations (client.{operation}) will no longer be available in the lakefs-sdk package’s LakeFSClient class. In their place, the client.{operation}_api should be used.
  8. +
  9. Minimum Python Version: 3.7
  10. +
  11. Fetching results from response objects: Instead of fetching the required results properties from a dictionary using response_result.get_property(prop_name), the response objects will include domain specific entities, thus referring to the properties in the results of the response - response_result.prop_name. For example, instead of:
  12. +
+ +
response = lakefs_client.branches.diff_branch(repository='repo', branch='main')
+diff = response.results[0] # 'results' is a 'DiffList' object
+path = diff.get_property('path') # 'diff' is a dictionary
+
+ +

You should use:

+ +
response = lakefs_client.branches_api.diff_branch(repository='repo', branch='main')
+diff = response.results[0] # 'results' is a 'DiffList' object
+path = diff.path # 'diff' is a 'Diff' object
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/project/contributing.html b/v1.46/project/contributing.html new file mode 100644 index 000000000..a38ee69ca --- /dev/null +++ b/v1.46/project/contributing.html @@ -0,0 +1,915 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Contributing to lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Contributing to lakeFS + + +

+ + +

Thank you for your interest in contributing to our project. Whether it’s a bug report, new feature, correction, or additional documentation, we greatly value feedback and contributions from our community.

+ +

Please read through this document before submitting any issues or pull requests to ensure that we have all the necessary information to effectively respond to your bug report or contribution.

+ +

Don’t know where to start? Reach out on the #dev channel on our Slack and we will help you get started. We also recommend this free series about contributing to OSS projects.

+

+ + + Getting Started + + +

+ + +

Before you get started, we kindly ask that you:

+ +
    +
  • Check out the code of conduct.
  • +
  • Sign the lakeFS CLA when making your first pull request (individual / corporate)
  • +
  • Submit any security issues directly to security@treeverse.io.
  • +
  • Contributions should have an associated GitHub issue.
  • +
  • Before making major contributions, please reach out to us on the #dev channel on Slack. +We will make sure no one else is working on the same feature.
  • +
+

+ + + Setting up an Environment + + +

+ + +

This section was tested on macOS and Linux (Fedora 32, Ubuntu 20.04) - Your mileage may vary

+ +

Our Go release workflow holds the Go and Node.js versions we currently use under go-version and node-version compatibly. The Java workflows use Maven 3.8.x (but any recent version of Maven should work).

+ +
    +
  1. Install the required dependencies for your OS: +
      +
    1. Git
    2. +
    3. GNU make (probably best to install from your OS package manager such as apt or brew)
    4. +
    5. Docker
    6. +
    7. Go
    8. +
    9. Node.js & npm
    10. +
    11. Maven to build and test Spark client codes.
    12. +
    13. Java 8 + +
    14. +
    15. Optional - PostgreSQL 11 (useful for running and debugging locally)
    16. +
    17. Optional - Rust & Cargo (useful for building the Rust SDK)
    18. +
    19. Optional - Buf CLI (only needed if you like to update Protocol Buffer files)
    20. +
    +
  2. +
  3. +

    Clone the repository from GitHub.

    + +

    This gives you read-only access to the repository. To contribute, see the next section.

    +
  4. +
  5. +

    Build the project:

    + +
    make build
    +
    + +

    Note: make build won’t work for Windows users.

    +
  6. +
  7. +

    Make sure tests are passing. The following should not return any errors:

    + +
    make test
    +
    +
  8. +
+

+ + + Before creating a pull request + + +

+ + +
    +
  1. Review this document in full.
  2. +
  3. Make sure there’s an open issue on GitHub that this pull request addresses, and that it isn’t labeled x/wontfix.
  4. +
  5. Fork the lakeFS repository.
  6. +
  7. If you’re adding new functionality, create a new branch named feature/<DESCRIPTIVE NAME>.
  8. +
  9. If you’re fixing a bug, create a new branch named fix/<DESCRIPTIVE NAME>-<ISSUE NUMBER>.
  10. +
+

+ + + Testing your change + + +

+ + +

Once you’ve made the necessary changes to the code, make sure the tests pass:

+ +

Run unit tests:

+ +
make test
+
+ +

Check that linting rules are passing.

+ +
make checks-validator
+
+ +

You will need GNU diff to run this. On the macOS it can be installed with brew install diffutils

+ +

lakeFS uses go fmt as a style guide for Go code.

+ +

Run system-tests:

+ +
make system-tests
+
+ +

Want to dive deeper into our system tests infrastructure? Need to debug the tests? Follow this documentation.

+

+ + + Submitting a pull request + + +

+ + +

Open a GitHub pull request with your change. The PR description should include a brief explanation of your change. +You should also mention the related GitHub issue. If the issue should be automatically closed after the merge, please link it to the PR.

+ +

After submitting your pull request, GitHub Actions will automatically run tests on your changes and make sure that your updated code builds and runs on Go 1.19.x.

+ +

Check back shortly after submitting your pull request to make sure that your code passes these checks. If any of the checks come back with a red X, then do your best to address the errors.

+ +

A developer from our team will review your pull request, and may request some changes to it. After the request is approved, it will be merged to our main branch.

+

+ + + Documentation + + +

+ + +

Any contribution to the docs, whether it is in conjunction with a code contribution or as a standalone, is appreciated.

+ +

Documentation of features and changes in behaviour should be included in the pull request. +You can create separate pull requests for documentation changes only.

+ +

To learn how to contribute to the lakeFS documentation see this page, which also includes details on how to build the documentation locally.

+

+ + + CHANGELOG.md + + +

+ + +

Any user-facing change should be labeled with include-changelog. +The PR title should contain a concise summary of the feature or fix and the description should have the GitHub issue number. +When we publish a new version of lakeFS, we will add this to the relevant version section of the changelog. +If the change should not be included in the changelog, label it with exclude-changelog.

+

+ + + User-Facing Changes Examples + + +

+ +
    +
  1. UI/UX modifications: Changes to the layout, color scheme, or navigation structure.
  2. +
  3. New features: Adding functionality that users can directly interact with, unless defined as internal.
  4. +
  5. Configuration changes: Updates to settings that users can adjust.
  6. +
  7. Performance improvements: Enhancements that noticeably speed up the application.
  8. +
  9. Bug fixes.
  10. +
  11. Security updates: Changes that address vulnerabilities or privacy concerns.
  12. +
+

+ + + Non-User-Facing Changes: + + +

+ +
    +
  1. Code refactoring: Restructuring the codebase without changing its external behavior.
  2. +
  3. Backend optimizations: Improvements to server-side processes that don’t noticeably affect performance.
  4. +
  5. Database schema changes: Modifications to the data structure that don’t alter the user interface and do not require data migration.
  6. +
  7. Development tooling updates: Changes to build processes or development environments.
  8. +
  9. Internal API: Adding/Altering APIs tagged as internal.
  10. +
  11. Documentation updates.
  12. +
  13. Imported libraries: Updates to third-party libraries that don’t introduce security updates.
  14. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/project/docs/callouts.html b/v1.46/project/docs/callouts.html new file mode 100644 index 000000000..70d3c06e9 --- /dev/null +++ b/v1.46/project/docs/callouts.html @@ -0,0 +1,873 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Callouts | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Callouts in lakeFS Documentation + + +

+ + +

These are part of the just-the-docs theme, and use the Block Inline Attribute List (IAL) feature of Kramdown.

+

+ + + Syntax + + +

+ + +

Whilst the lakeFS docs typically use this syntax with the code after the block:

+ +
This is a note
+{: .note }
+
+ +

This is a note

+ +

The just-the-docs documentation puts the code before the block, which makes more sense to the human eye when reading the Markdown too.

+ +
{: .note }
+This is a note
+
+ +

This is a note

+ +

However, they render the same, as can be see above.

+

+ + + Configuration + + +

+ + +

Callouts are configured in the _config.yml file, with an identifier (such as note), a colour, and optionally a title.

+ +
callouts:
+  note:
+    color: blue
+  warning:
+    title: ⚠️ Warning
+    color: red
+  tip:
+    color: green
+  fubar:
+    color: purple
+
+ +

Here’s what the above callouts look like; notice that only warning includes a title:

+ +
{: .note }
+This is a note
+
+ +

This is a note

+ +
{: .warning }
+This is a warning
+
+ +

This is a warning

+ +
{: .fubar }
+This is illustrating that the callout names are completely discretionary and not tied to their meaning
+
+ +

This is illustrating that the callout names are completely discretionary and not tied to their meaning

+

+ + + Custom titles + + +

+ + +

You can specify custom titles per block by appending -title to the block name and putting the custom title in the first line of the quoted block:

+ +
{: .tip-title }
+> lakeFS Cloud
+>
+> For a fully-managed lakeFS solution check out https://lakefs.cloud/ today
+
+ +
+

lakeFS Cloud

+ +

For a fully-managed lakeFS solution check out https://lakefs.cloud/ today

+
+ +

You can also remove the title of a callout that includes one by default (such as warning) above:

+ +
{: .warning-title }
+> Don't press the red button
+
+ +
+

Don’t press the red button

+
+ +

Or change the default title:

+ +
{: .warning-title }
+> Here Be Dragons 🐲
+>
+> ALLES TURISTEN UND NONTEKNISCHEN LOOKENSPEEPERS! <br/>
+> DAS KOMPUTERMASCHINE IST NICHT FÜR DER GEFINGERPOKEN UND MITTENGRABEN! ODERWISE IST EASY TO SCHNAPPEN DER SPRINGENWERK, BLOWENFUSEN UND POPPENCORKEN MIT SPITZENSPARKEN.
+
+ +
+

Here Be Dragons 🐲

+ +

ALLES TURISTEN UND NONTEKNISCHEN LOOKENSPEEPERS!
+DAS KOMPUTERMASCHINE IST NICHT FÜR DER GEFINGERPOKEN UND MITTENGRABEN! ODERWISE IST EASY TO SCHNAPPEN DER SPRINGENWERK, BLOWENFUSEN UND POPPENCORKEN MIT SPITZENSPARKEN.

+
+

+ + + Multi-line Blocks + + +

+ + +

Callouts support multiple lines and/or paragraphs of content:

+ +
{: .note }
+> A multi-paragraph block
+>
+> with <br/>
+> linebreaks
+
+ +
+

A multi-paragraph block

+ +

with
+linebreaks

+
+

+ + + Reference + + +

+ + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/project/docs/index.html b/v1.46/project/docs/index.html new file mode 100644 index 000000000..c3510bcaa --- /dev/null +++ b/v1.46/project/docs/index.html @@ -0,0 +1,959 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Documentation | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Documentation + + +

+ + +

Any contribution to the docs, whether it is in conjunction with a code contribution or as a standalone, is appreciated.

+ +

Please see the contributing guide for details on contributing to lakeFS in general.

+ + + +

📝 Notice! lakeFS documentation is written using Markdown. Make sure to familiarize yourself with the Markdown Guide.

+ +

Customizing the lakeFS docs site should follow the following guidelines: Just The Docs Customization and style-guide.

+

+ + + lakeFS Documentation Philosophy + + +

+ + +

We are heavily inspired by the Diátaxis approach to documentation. At a very high-level, it defines documentation as falling into one of four categories:

+ +
    +
  • How To
  • +
  • Tutorial
  • +
  • Reference
  • +
  • Explanation
  • +
+ +

There is a lot more to it than this, and you are encouraged to read the Diátaxis website for more details. Its application to lakeFS was discussed in #6197

+

+ + + lakeFS Style Guide: + + +

+ + +
    +
  • Don’t use unnecessary tech jargon or vague/wordy constructions - keep it friendly, not condescending.
  • +
  • Be inclusive and welcoming - use gender-neutral words and pronouns when talking about abstract people like users and developers.
  • +
  • Replace complex expressions with simpler ones.
  • +
  • Keep it short - 25-30 words max per sentence. Otherwise, your readers might get lost on the way.
  • +
  • Use active voice instead of passive. For example: This feature can be used to do task X. vs. You can use this feature to do task X. The second one reads much better, right?
  • +
  • You can explain things better by including examples. Show, not tell. Use illustrations, images, gifs, code snippets, etc.
  • +
  • Establish a visual hierarchy to help people quickly find the information they need. Use text formatting to create levels of title and subtitle (such as # to ###### markdown headings). The title of every page should use the topmost heading #; all other headings on the page should use lower headers ## to ######.
  • +
+

+ + + Headings and Table of Contents + + +

+ + +

The title of the page should be H1 (# in markdown). Use headings in descending order and do not skip any.

+ +

Pages should generally have a table of contents to help the user navigate it. Use the following snippet to add it to your page:

+ +
<div class="toc-block">
+## Table of contents
+{: .no_toc .text-delta }
+
+1. TOC
+{:toc}
+{::options toc_levels="2" /}
+</div>
+
+
+ +

By default the page’s Table of Contents will include only H2 headings. If you want to include H2 and H3 then use this snippet instead:

+ +
<div class="toc-block">
+## Table of contents
+{: .no_toc .text-delta }
+
+1. TOC
+{:toc}
+{::options toc_levels="2..3" /}
+</div>
+
+
+ +

Both of these snippets invoke {:toc} which is used by Kramdown (the Markdown processor that Jekyll uses) to insert a table of contents from the headings present in the markdown.

+

+ + + Callouts 💬 + + +

+ + +

Multiple callout types are available. Please review this page for details.

+ + + +

Links should use absolute paths in conjunction with {% link %}, e.g. {% +link foo/example.md %}.

+ +

Adding a link with an anchor is a bit trickier. Create a reference [link +text][link-reference] and then define the anchor at the end of the document:

+
[link-reference]:  {% link foo.example.md %}#anchor
+
+ +

This is so that references work within the versioned documentation that is deployed.

+ +

Relative links, unless within the local folder, are discouraged as it can cause additional work when moving pages at a later date.

+

+ + + Test your changes locally + + +

+ + +

If you have the necessary dependencies installed, you can run Jekyll to build and serve the documentation from your machine using the provided Makefile target:

+ +
make docs-serve
+
+ +

The alternative is to use Docker which has the benefit of handling all the dependencies for you.

+

+ + + Docker + + +

+ + +
    +
  1. +

    Launch the Docker container:

    + +
    docker run --rm \
    +           --name lakefs_docs \
    +           -e TZ="Etc/UTC" \
    +           --publish 4000:4000 --publish 35729:35729 \
    +           --volume="$PWD/docs:/srv/jekyll:Z" \
    +           --volume="$PWD/docs/.jekyll-bundle-cache:/usr/local/bundle:Z" \
    +           --interactive --tty \
    +           jekyll/jekyll:4.2.2 \
    +           jekyll serve --livereload
    +
    + +

    If you have make installed, you can also run make docs-serve-docker instead.

    +
  2. +
  3. +

    The first time you run the container it will need to download dependencies and will take several minutes to be ready.

    + +

    Once you see the following output, the docs server is ready to open in your web browser:

    + +
    Server running... press ctrl-c to stop.
    +
    +
  4. +
  5. +

    When you make a change to a page’s source the server will automatically rebuild the page which will be shown in the server log by this entry:

    + +
    Regenerating: 1 file(s) changed at 2023-01-26 08:34:47
    +               contributing.md
    +Remote Theme: Using theme pmarsceill/just-the-docs
    +
    + +

    This can take a short while—you’ll see something like this in the server’s output when it’s done.

    + +
    ...done in 34.714073460 seconds.
    +
    + +

    Your page will automatically reload to show the changes.

    +
  6. +
+ +

If you are doing lots of work on the docs you may want to leave the Docker container in place (so that you don’t have to wait for the dependencies to load each time you re-create it). To do this replace the --rm with --detach in the docker run command, and use docker logs -f lakefs_docs to view the server log.

+ + + +

When making a pull request to lakeFS that involves a docs/* file, a GitHub action will automagically check the links. You can also run this link checker manually on your local machine:

+ +
    +
  1. +

    Build the site:

    + +
    docker run --rm \
    +         --name lakefs_docs \
    +         -e TZ="Etc/UTC" \
    +         --volume="$PWD/docs:/srv/jekyll:Z" \
    +         --volume="$PWD/docs/.jekyll-bundle-cache:/usr/local/bundle:Z" \
    +         --interactive --tty \
    +         jekyll/jekyll:3.8 \
    +         jekyll build --config _config.yml -d _site --watch
    +
    +
  2. +
  3. +

    Check the links:

    + +
    docker run --rm \
    +         --name lakefs_docs_lychee \
    +         --volume "$PWD:/data"\
    +         --volume "/tmp:/output"\
    +         --tty \
    +         lycheeverse/lychee:master \
    +         --exclude-file /data/docs/.lycheeignore \
    +         --output /output/lychee_report.md \
    +         --format markdown \
    +         /data/docs/_site
    +
    +
  4. +
  5. +

    Review the lychee_report.md in your local /tmp folder

    +
  6. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/project/index.html b/v1.46/project/index.html new file mode 100644 index 000000000..338763f57 --- /dev/null +++ b/v1.46/project/index.html @@ -0,0 +1,785 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +The lakeFS Project | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + The lakeFS Project + + +

+ + +

lakeFS provides version control over the data lake and lakehouse, and uses Git-like semantics to create and access those versions. If you know git, you’ll be right at home with lakeFS.

+ +

lakeFS is an open-source project under the Apache 2.0 license.

+ +

The project was created and is supported by Treeverse, a commercial company founded by engineers passionate about providing solutions to the evolving world of data engineering.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Our commitment to open source
  2. +
  3. Roadmap
  4. +
+ +
+

+ + + Our commitment to open source + + +

+ + +

lakeFS is an open-source project under the Apache 2.0 license. +The project was created and is supported by Treeverse, a commercial company founded by engineers passionate about providing solutions to the evolving world of data engineering.

+ +

Why did we choose to open the source of our core capabilities?

+ +

We believe in the bottom-up adoption of technologies. +We believe collaborative communities have the power to bring the best solutions to the community. +We believe that every engineer should be able to use, contribute to, and influence cutting edge technologies, so they can innovate in their domain.

+ +

What is our commitment to open source?

+ +

We created lakeFS, our open-source project, to provide a Git-like interface on top of object stores - so that you can fully take advantage of with any data application at any scale.

+ +

For that reason, we commit that the following capabilities are and will remain open-source as part of lakeFS:

+ +
    +
  • All versioning capabilities,
  • +
  • Git-Like interface for the versioning operations,
  • +
  • Support for public object store APIs,
  • +
  • Integrations with publicly available applications accessing an object store,
  • +
  • CLI, API, and GUI interfaces.
  • +
+ +

We also commit to keeping lakeFS scalable in throughput and performance.

+ +

We are deeply committed to our community of engineers who use and contribute to the project. We are and will continue to be highly responsive and shape lakeFS together to provide the data versioning capabilities we are all looking for.

+ +

What is lakeFS Cloud?

+ +

Treeverse offers lakeFS Cloud, which provides all the same benefits of the Git-like interface on top of object stores as a fully-managed service.

+ +

The vision behind lakeFS Cloud is to provide a managed data versioning and management solution for data practitioners. lakeFS Cloud will leverage the lakeFS open-source technology, integrate capabilities and unique features, and lead its users to implement best practices.

+ +

As part of our commitment to the open source values of transparency and interoperability, we believe everyone should be able to enjoy these benefits, regardless of whether or not they choose to use the managed offering.

+ +

Because of that, we will not intentionally make it harder to build these features independently on top of the open source solution.

+

+ + + Roadmap + + +

+ + +

This is the lakeFS public roadmap. As the project evolves and the community grows, this roadmap might change.

+ +

Please share your feedback on the #dev on Slack or by starting a GitHub Discussion!

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart.html b/v1.46/quickstart.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/actions-and-hooks.html b/v1.46/quickstart/actions-and-hooks.html new file mode 100644 index 000000000..3c4368b19 --- /dev/null +++ b/v1.46/quickstart/actions-and-hooks.html @@ -0,0 +1,917 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +6️⃣ Using Actions and Hooks in lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Actions and Hooks in lakeFS + + +

+ + +

When we interact with lakeFS it can be useful to have certain checks performed at stages along the way. Let’s see how actions in lakeFS can be of benefit here.

+ +

We’re going to enforce a rule that when a commit is made to any branch that begins with etl:

+ +
    +
  • the commit message must not be blank
  • +
  • there must be job_name and version metadata
  • +
  • the version must be numeric
  • +
+ +

To do this we’ll create an action. In lakeFS, an action specifies one or more events that will trigger it, and references one or more hooks to run when triggered. Actions are YAML files written to lakeFS under the _lakefs_actions/ folder of the lakeFS repository.

+ +

Hooks can be either a Lua script that lakeFS will execute itself, an external web hook, or an Airflow DAG. In this example, we’re using a Lua hook.

+

+ + + Configuring the Action + + +

+ + +
    +
  1. +

    In lakeFS create a new branch called add_action. You can do this through the UI or with lakectl:

    + +
     docker exec lakefs lakectl branch create lakefs://quickstart/add_action --source lakefs://quickstart/main
    +
    +
  2. +
  3. +

    Open up your favorite text editor (or emacs), and paste the following YAML:

    + +
    name: Check Commit Message and Metadata
    +on:
    +  pre-commit:
    +    branches:
    +    - etl**
    +hooks:
    +- id: check_metadata
    +  type: lua
    +  properties:
    +    script: |
    +        commit_message=action.commit.message
    +        if commit_message and #commit_message>0 then
    +            print("✅ The commit message exists and is not empty: " .. commit_message)
    +        else
    +            error("\n\n❌ A commit message must be provided")
    +        end
    +   
    +        job_name=action.commit.metadata["job_name"]
    +        if job_name == nil then
    +            error("\n❌ Commit metadata must include job_name")
    +        else
    +            print("✅ Commit metadata includes job_name: " .. job_name)
    +        end
    +   
    +        version=action.commit.metadata["version"]
    +        if version == nil then
    +            error("\n❌ Commit metadata must include version")
    +        else
    +            print("✅ Commit metadata includes version: " .. version)
    +            if tonumber(version) then
    +                print("✅ Commit metadata version is numeric")
    +            else
    +                error("\n❌ Version metadata must be numeric: " .. version)
    +            end
    +        end
    +
    +
  4. +
  5. +

    Save this file as /tmp/check_commit_metadata.yml

    + +
      +
    • You can save it elsewhere, but make sure you change the path below when uploading
    • +
    +
  6. +
  7. +

    Upload the check_commit_metadata.yml file to the add_action branch under _lakefs_actions/. As above, you can use the UI (make sure you select the correct branch when you do), or with lakectl:

    + +
     docker exec lakefs lakectl fs upload lakefs://quickstart/add_action/_lakefs_actions/check_commit_metadata.yml --source /tmp/check_commit_metadata.yml
    +
    +
  8. +
  9. +

    Go to the Uncommitted Changes tab in the UI, and make sure that you see the new file in the path shown:

    + +

    lakeFS Uncommitted Changes view showing a file called `check_commit_metadata.yml` under the path `_lakefs_actions/`

    + +

    Click Commit Changes and enter a suitable message to commit this new file to the branch.

    +
  10. +
  11. +

    Now we’ll merge this new branch into main. From the Compare tab in the UI compare the main branch with add_action and click Merge

    + +

    lakeFS Compare view showing the difference between `main` and `add_action` branches

    +
  12. +
+

+ + + Testing the Action + + +

+ + +

Let’s remind ourselves what the rules are that the action is going to enforce.

+ +
+

When a commit is made to any branch that begins with etl:

+
+ +
+
    +
  • the commit message must not be blank
  • +
  • there must be job_name and version metadata
  • +
  • the version must be numeric
  • +
+
+ +

We’ll start by creating a branch that’s going to match the etl pattern, and then go ahead and commit a change and see how the action works.

+ +
    +
  1. +

    Create a new branch (see above instructions on how to do this if necessary) called etl_20230504. Make sure you use main as the source branch.

    + +

    In your new branch you should see the action that you created and merged above:

    + +

    lakeFS branch etl_20230504 with object /_lakefs_actions/check_commit_metadata.yml

    +
  2. +
  3. +

    To simulate an ETL job we’ll use the built-in DuckDB editor to run some SQL and write the result back to the lakeFS branch.

    + +

    Open the lakes.parquet file on the etl_20230504 branch from the Objects tab. Replace the SQL statement with the following:

    + +
     COPY (
    +     WITH src AS (
    +         SELECT lake_name, country, depth_m,
    +             RANK() OVER ( ORDER BY depth_m DESC) AS lake_rank
    +         FROM READ_PARQUET('lakefs://quickstart/etl_20230504/lakes.parquet'))
    +     SELECT * FROM SRC WHERE lake_rank <= 10
    + ) TO 'lakefs://quickstart/etl_20230504/top10_lakes.parquet'    
    +
    +
  4. +
  5. +

    Head to the Uncommitted Changes tab in the UI and notice that there is now a file called top10_lakes.parquet waiting to be committed.

    + +

    lakeFS branch etl_20230504 with uncommitted file top10_lakes.parquet

    + +

    Now we’re ready to start trying out the commit rules, and seeing what happens if we violate them.

    +
  6. +
  7. +

    Click on Commit Changes, leave the Commit message blank, and click Commit Changes to confirm.

    + +

    Note that the commit fails because the hook did not succeed

    + +

    pre-commit hook aborted

    + +

    with the output from the hook’s code displayed

    + +

    ❌ A commit message must be provided

    + +

    lakeFS blocking an attempt to commit with no commit message

    +
  8. +
  9. +

    Do the same as the previous step, but provide a message this time:

    + +

    A commit to lakeFS with commit message in place

    + +

    The commit still fails as we need to include metadata too, which is what the error tells us

    + +

    ❌ Commit metadata must include job_name

    +
  10. +
  11. +

    Repeat the Commit Changes dialog and use the Add Metadata field to add the required metadata:

    + +

    A commit to lakeFS with commit message and metadata in place

    + +

    We’re almost there, but this still fails (as it should), since the version is not entirely numeric but includes v and ß:

    + +

    ❌ Version metadata must be numeric: v1.00ß

    + +

    Repeat the commit attempt specify the version as 1.00 this time, and rejoice as the commit succeeds

    + +

    Commit history in lakeFS showing that the commit met the rules set by the action and completed successfully.

    +
  12. +

+ +

You can view the history of all action runs from the Action tab:

+ +

Action run history in lakeFS

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/add_data.html b/v1.46/quickstart/add_data.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/add_data.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/branch.html b/v1.46/quickstart/branch.html new file mode 100644 index 000000000..3c23c0b86 --- /dev/null +++ b/v1.46/quickstart/branch.html @@ -0,0 +1,866 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +3️⃣ Create a branch | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Create a Branch + + +

+ + +

lakeFS uses branches in a similar way to Git. It’s a great way to isolate changes until, or if, we are ready to re-integrate them. lakeFS uses a zero-copy branching technique which means that it’s very efficient to create branches of your data.

+ +

Having seen the lakes data in the previous step we’re now going to create a new dataset to hold data only for lakes in Denmark. Why? Well, because :)

+ +

The first thing we’ll do is create a branch for us to do this development against. We’ll use the lakectl tool to create the branch, which we first need to configure with our credentials. In a new terminal window run the following:

+ +
docker exec -it lakefs lakectl config
+
+ +

Follow the prompts to enter the credentials that you got in the first step. Leave the Server endpoint URL as http://127.0.0.1:8000.

+ +

Now that lakectl is configured, we can use it to create the branch. Run the following:

+ +
docker exec lakefs lakectl branch create lakefs://quickstart/denmark-lakes --source lakefs://quickstart/main
+
+ +

You should get a confirmation message like this:

+ +
Source ref: lakefs://quickstart/main
+created branch 'denmark-lakes' 3384cd7cdc4a2cd5eb6249b52f0a709b49081668bb1574ce8f1ef2d956646816
+
+

+ + + Transforming the Data + + +

+ + +

Now we’ll make a change to the data. lakeFS has several native clients, as well as an S3-compatible endpoint. This means that anything that can use S3 will work with lakeFS. Pretty neat.

+ +

We’re going to use DuckDB which is embedded within the web interface of lakeFS.

+ +

From the lakeFS Objects page select the lakes.parquet file to open the DuckDB editor:

+ +

The lakeFS object viewer with embedded DuckDB to query parquet files. A query has run automagically to preview the contents of the selected parquet file.

+ +

To start with, we’ll load the lakes data into a DuckDB table so that we can manipulate it. Replace the previous text in the DuckDB editor with this:

+ +
CREATE OR REPLACE TABLE lakes AS 
+    SELECT * FROM READ_PARQUET('lakefs://quickstart/denmark-lakes/lakes.parquet');
+
+ +

You’ll see a row count of 100,000 to confirm that the DuckDB table has been created.

+ +

Just to check that it’s the same data that we saw before we’ll run the same query. Note that we are querying a DuckDB table (lakes), rather than using a function to query a parquet file directly.

+ +
SELECT   country, COUNT(*)
+FROM     lakes
+GROUP BY country
+ORDER BY COUNT(*) 
+DESC LIMIT 5;
+
+ +

The DuckDB editor pane querying the lakes table

+

+ + + Making a Change to the Data + + +

+ + +

Now we can change our table, which was loaded from the original lakes.parquet, to remove all rows not for Denmark:

+ +
DELETE FROM lakes WHERE Country != 'Denmark';
+
+ +

The DuckDB editor pane deleting rows from the lakes table

+ +

We can verify that it’s worked by reissuing the same query as before:

+ +
SELECT   country, COUNT(*)
+FROM     lakes
+GROUP BY country
+ORDER BY COUNT(*) 
+DESC LIMIT 5;
+
+ +

The DuckDB editor pane querying the lakes table showing only rows for Denmark remain

+

+ + + Write the Data back to lakeFS + + +

+ + +

The changes so far have only been to DuckDB’s copy of the data. Let’s now push it back to lakeFS. Note the path is different this time as we’re writing it to the denmark-lakes branch, not main:

+ +
COPY lakes TO 'lakefs://quickstart/denmark-lakes/lakes.parquet';
+
+ +

The DuckDB editor pane writing data back to the denmark-lakes branch

+

+ + + Verify that the Data’s Changed on the Branch + + +

+ + +

Let’s just confirm for ourselves that the parquet file itself has the new data. We’ll drop the lakes table just to be sure, and then query the parquet file directly:

+ +
DROP TABLE lakes;
+
+SELECT   country, COUNT(*)
+FROM     READ_PARQUET('lakefs://quickstart/denmark-lakes/lakes.parquet')
+GROUP BY country
+ORDER BY COUNT(*) 
+DESC LIMIT 5;
+
+ +

The DuckDB editor pane show the parquet file on denmark-lakes branch has been changed

+

+ + + What about the data in main? + + +

+ + +

So we’ve changed the data in our denmark-lakes branch, deleting swathes of the dataset. What’s this done to our original data in the main branch? Absolutely nothing! See for yourself by running the same query as above, but against the main branch:

+ +
SELECT   country, COUNT(*)
+FROM     READ_PARQUET('lakefs://quickstart/main/lakes.parquet')
+GROUP BY country
+ORDER BY COUNT(*) 
+DESC LIMIT 5;
+
+

The lakeFS object browser showing DuckDB querying lakes.parquet on the main branch. The results are the same as they were before we made the changes to the denmark-lakes branch, which is as expected.

+ +

In the next step we’ll see how to commit our changes and merge our branch back into main.

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/commit-and-merge.html b/v1.46/quickstart/commit-and-merge.html new file mode 100644 index 000000000..84c0d65ea --- /dev/null +++ b/v1.46/quickstart/commit-and-merge.html @@ -0,0 +1,768 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +4️⃣ Commit and Merge | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

In the previous step we branched our data from main into a new denmark-lakes branch, and overwrote the lakes.parquet to hold solely information about lakes in Denmark. Now we’re going to commit that change (just like Git) and merge it back to main (just like git).

+

+ + + Committing Changes in lakeFS + + +

+ + +

Having make the change to the datafile in the denmark-lakes branch, we now want to commit it. There are various options for interacting with the lakeFS API, including the web interface, a Python client, and lakectl which is what we’ll use here. Run the following from a terminal window:

+ +
docker exec lakefs lakectl commit lakefs://quickstart/denmark-lakes -m "Create a dataset of just the lakes in Denmark"
+
+ +

You will get confirmation of the commit including its hash.

+
Branch: lakefs://quickstart/denmark-lakes
+Commit for branch "denmark-lakes" completed.
+
+ID: ba6d71d0965fa5d97f309a17ce08ad006c0dde15f99c5ea0904d3ad3e765bd74
+Message: Create a dataset of just the lakes in Denmark
+Timestamp: 2023-03-15 08:09:36 +0000 UTC
+Parents: 3384cd7cdc4a2cd5eb6249b52f0a709b49081668bb1574ce8f1ef2d956646816
+
+ +

With our change committed, it’s now time to merge it to back to the main branch.

+

+ + + Merging Branches in lakeFS 🔀 + + +

+ + +

As above, we’ll use lakectl to do this too. The syntax just requires us to specify the source and target of the merge. Run this from a terminal window.

+ +
docker exec lakefs lakectl merge lakefs://quickstart/denmark-lakes lakefs://quickstart/main
+
+ +

We can confirm that this has worked by returning to the same object view of lakes.parquet as before and clicking on Execute to rerun the same query. You’ll see that the country row counts have changed, and only Denmark is left in the data:

+ +

The lakeFS object browser with a DuckDB query on lakes.parquet showing that there is only data for Denmark.

+ +

But…oh no! A slow chill creeps down your spine, and the bottom drops out of your stomach. What have you done! 😱 You were supposed to create a separate file of Denmark’s lakes - not replace the original one

+ +

Is all lost? Will our hero overcome the obstacles? No, and yes respectively!

+ +

Have no fear; lakeFS can revert changes. Tune in for the final part of the quickstart to see how.

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/first_commit.html b/v1.46/quickstart/first_commit.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/first_commit.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/index.html b/v1.46/quickstart/index.html new file mode 100644 index 000000000..722f95ebc --- /dev/null +++ b/v1.46/quickstart/index.html @@ -0,0 +1,856 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +⭐ Quickstart | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS Quickstart + + +

+ + +

Welcome to lakeFS!

+ +

lakeFS provides a “Git for data” platform enabling you to implement best practices from software engineering on your data lake, including branching and merging, CI/CD, and production-like dev/test environments.

+ +

This quickstart will introduce you to some of the core ideas in lakeFS and show what you can do by illustrating the concept of branching, merging, and rolling back changes to data. It’s laid out in five short sections:

+ +
+ +
+
+

step 1

+
+
+

+ + + +Launch + + + +

+ +

Spin up the quickstart environment locally under Docker

+
+
+ +
+
+

step 2

+
+
+

+ + + +Query + + + +

+ +

Query the pre-populated data on the main branch

+
+
+ +
+
+

step 3

+
+
+

+ + + +Branch + + + +

+ +

Make changes to the data on a new branch

+
+
+ +
+
+

step 4

+
+
+

+ + + +Merge + + + +

+ +

Merge the changed data back to the main branch

+
+
+ +
+
+

step 5

+
+
+

+ + + +Rollback + + + +

+ +

Change our mind and revert the changes

+
+
+
+ +
+
+

step 6

+
+
+

+ + + +Actions + + + +

+ +

Use Actions to trigger code when an event occurs

+
+
+ +
+
+

step 6

+
+
+

+ + + +Work Locally + + + +

+ +

Experiment with lakeFS data on a local environment

+
+
+ +

You can use the 30-day free trial of lakeFS Cloud if you want to try out lakeFS without installing anything.

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/installing.html b/v1.46/quickstart/installing.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/installing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/iso_env.html b/v1.46/quickstart/iso_env.html new file mode 100644 index 000000000..81a672a11 --- /dev/null +++ b/v1.46/quickstart/iso_env.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/lakefs_cli.html b/v1.46/quickstart/lakefs_cli.html new file mode 100644 index 000000000..6c6c5a594 --- /dev/null +++ b/v1.46/quickstart/lakefs_cli.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/launch.html b/v1.46/quickstart/launch.html new file mode 100644 index 000000000..892e74837 --- /dev/null +++ b/v1.46/quickstart/launch.html @@ -0,0 +1,772 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +1️⃣ Run lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Spin up the environment + + +

+ + +

If you don’t want to use Docker, you can use the 30-day free trial of lakeFS Cloud. Once you launch the free trial you will have access to the same content as this quickstart within the provided repository once you login.

+ +

The quickstart uses Docker to bring up the lakeFS container, pre-populate it with some data, and also provides DuckDB from where we can interact with the data. You’ll need Docker installed to run this.

+ +

Launch the lakeFS container:

+ +
docker run --name lakefs --pull always --rm --publish 8000:8000 treeverse/lakefs:latest run --quickstart
+
+ +

After a few moments you should see the lakeFS container ready to use:

+ +
│
+│ lakeFS running in quickstart mode.
+│     Login at http://127.0.0.1:8000/
+│
+│     Access Key ID    : AKIAIOSFOLQUICKSTART
+│     Secret Access Key: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
+│
+
+ +

You’re now ready to dive into lakeFS!

+ +
    +
  1. +

    Open lakeFS’s web interface at http://127.0.0.1:8000/

    +
  2. +
  3. +

    Login with the quickstart credentials.

    + +
      +
    • Access Key ID: AKIAIOSFOLQUICKSTART
    • +
    • Secret Access Key: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
    • +
    +
  4. +
  5. +

    You’ll notice that there aren’t any repositories created yet. Click the Create Sample Repository button.

    + +

    Empty lakeFS Repository list

    +
  6. +
+ +

You will see the sample repository created and the quickstart guide within it. You can follow along there, or here - it’s the same :)

+ +

The quickstart sample repo in lakeFS

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/learning-more-lakefs.html b/v1.46/quickstart/learning-more-lakefs.html new file mode 100644 index 000000000..3be4b7134 --- /dev/null +++ b/v1.46/quickstart/learning-more-lakefs.html @@ -0,0 +1,870 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Learn more about lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Learn more about lakeFS + + +

+ + +

The lakeFS quickstart is just the beginning of your lakeFS journey 🛣️

+ +

Here are some more resources to help you find out more about lakeFS.

+

+ + + Connecting lakeFS to your own object storage + + +

+ + +

Enjoyed the quickstart and want to try out lakeFS against your own data? Here’s how to run lakeFS locally as a Docker container locally connecting to an object store.

+ +
+ +
+ +

Note: Make sure the Quickstart Docker Compose from the previous steps isn’t also running as you’ll get a port conflict.

+ +
docker run --pull always -p 8000:8000 \
+   -e LAKEFS_BLOCKSTORE_TYPE='s3' \
+   -e AWS_ACCESS_KEY_ID='YourAccessKeyValue' \
+   -e AWS_SECRET_ACCESS_KEY='YourSecretKeyValue' \
+   treeverse/lakefs run --local-settings
+
+ +
+
+ +

Note: Make sure the Quickstart Docker Compose from the previous steps isn’t also running as you’ll get a port conflict.

+ +
docker run --pull always -p 8000:8000 \
+   -e LAKEFS_BLOCKSTORE_TYPE='azure' \
+   -e LAKEFS_BLOCKSTORE_AZURE_STORAGE_ACCOUNT='YourAzureStorageAccountName' \
+   -e LAKEFS_BLOCKSTORE_AZURE_STORAGE_ACCESS_KEY='YourAzureStorageAccessKey' \
+   treeverse/lakefs run --local-settings
+
+ +
+
+ +

Note: Make sure the Quickstart Docker Compose from the previous steps isn’t also running as you’ll get a port conflict.

+ +
docker run --pull always -p 8000:8000 \
+   -e LAKEFS_BLOCKSTORE_TYPE='gs' \
+   -e LAKEFS_BLOCKSTORE_GS_CREDENTIALS_JSON='YourGoogleServiceAccountKeyJSON' \
+   treeverse/lakefs run --local-settings
+
+

where you will replace YourGoogleServiceAccountKeyJSON with JSON string that contains your Google service account key.

+ +

If you want to use the JSON file that contains your Google service account key instead of JSON string (as in the previous command) then go to the directory where JSON file is stored and run the command with local parameters:

+ +
docker run --pull always -p 8000:8000 \
+   -v $PWD:/myfiles \
+   -e LAKEFS_BLOCKSTORE_TYPE='gs' \
+   -e LAKEFS_BLOCKSTORE_GS_CREDENTIALS_FILE='/myfiles/YourGoogleServiceAccountKey.json' \
+   treeverse/lakefs run --local-settings
+
+

This command will mount your present working directory (PWD) within the container and will read the JSON file from your PWD.

+ +
+
+ +

To use lakeFS with MinIO (or other S3-compatible object storage), use the following example:

+ +

Note: Make sure the Quickstart Docker Compose from the previous steps isn’t also running as you’ll get a port conflict.

+ +
docker run --pull always -p 8000:8000 \
+   -e LAKEFS_BLOCKSTORE_TYPE='s3' \
+   -e LAKEFS_BLOCKSTORE_S3_FORCE_PATH_STYLE='true' \
+   -e LAKEFS_BLOCKSTORE_S3_ENDPOINT='http://<minio_endpoint>' \
+   -e LAKEFS_BLOCKSTORE_S3_DISCOVER_BUCKET_REGION='false' \
+   -e LAKEFS_BLOCKSTORE_S3_CREDENTIALS_ACCESS_KEY_ID='<minio_access_key>' \
+   -e LAKEFS_BLOCKSTORE_S3_CREDENTIALS_SECRET_ACCESS_KEY='<minio_secret_key>' \
+   treeverse/lakefs run --local-settings
+
+ +
+
+

+ + + Deploying lakeFS + + +

+ + +

Ready to do this thing for real? The deployment guides show you how to deploy lakeFS locally (including on Kubernetes) or on AWS, Azure, or GCP.

+ +

Alternatively you might want to have a look at lakeFS Cloud which provides a fully-managed, SOC-2 compliant, lakeFS service.

+

+ + + lakeFS Samples + + +

+ + +

The lakeFS Samples GitHub repository includes some excellent examples including:

+ +
    +
  • How to implement multi-table transaction on multiple Delta Tables
  • +
  • Notebooks to show integration of lakeFS with Spark, Python, Delta Lake, Airflow and Hooks.
  • +
  • Examples of using lakeFS webhooks to run automated data quality checks on different branches.
  • +
  • Using lakeFS branching features to create dev/test data environments for ETL testing and experimentation.
  • +
  • Reproducing ML experiments with certainty using lakeFS tags.
  • +
+

+ + + lakeFS Community + + +

+ + +

The lakeFS community is important to us. Our guiding principles are:

+ +
    +
  • Fully open, in code and conversation
  • +
  • We learn and grow together
  • +
  • Compassion and respect in every interaction
  • +
+ +

We’d love for you to join our Slack group and come and introduce yourself on #announcements-and-more. Or just lurk and soak up the vibes 😎

+ +

If you’re interested in getting involved in the development of lakeFS, head over our the GitHub repo to look at the code and peruse the issues. The comprehensive contributing document should have you covered on next steps but if you’ve any questions the #dev channel on Slack will be delighted to help.

+ +

We love speaking at meetups and chatting to community members at them - you can find a list of these here.

+ +

Finally, make sure to drop by to say hi on Twitter, Mastodon, and LinkedIn 👋🏻

+

+ + + lakeFS Concepts and Internals + + +

+ + +

We describe lakeFS as “Git for data” but what does that actually mean? Have a look at the concepts and architecture guides, as well as the explanation of how merges are handled. To go deeper you might be interested in the internals of versioning and our internal database structure.

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/more_quickstart_options.html b/v1.46/quickstart/more_quickstart_options.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/more_quickstart_options.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/query.html b/v1.46/quickstart/query.html new file mode 100644 index 000000000..feecca1c5 --- /dev/null +++ b/v1.46/quickstart/query.html @@ -0,0 +1,754 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +2️⃣ Query the data | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Let’s Query Something + + +

+ + +

The lakeFS server has been loaded with a sample parquet datafile. Fittingly enough for a piece of software to help users of data lakes, the lakes.parquet file holds data about lakes around the world.

+ +

You’ll notice that the branch is set to main. This is conceptually the same as your main branch in Git against which you develop software code.

+ +

The lakeFS objects list with a highlight to indicate that the branch is set to main.

+ +

Let’s have a look at the data, ahead of making some changes to it on a branch in the following steps.

+ +

Click on lakes.parquet and notice that the built-it DuckDB runs a query to show a preview of the file’s contents.

+ +

The lakeFS object viewer with embedded DuckDB to query parquet files. A query has run automagically to preview the contents of the selected parquet file.

+ +

Now we’ll run our own query on it to look at the top five countries represented in the data.

+ +

Copy and paste the following SQL statement into the DuckDB query panel and click on Execute.

+ +
SELECT   country, COUNT(*)
+FROM     READ_PARQUET('lakefs://quickstart/main/lakes.parquet')
+GROUP BY country
+ORDER BY COUNT(*) 
+DESC LIMIT 5;
+
+ +

An embedded DuckDB query showing a count of rows per country in the dataset.

+ +

Next we’re going to make some changes to the data—but on a development branch so that the data in the main branch remains untouched.

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/repository.html b/v1.46/quickstart/repository.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/repository.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/rollback.html b/v1.46/quickstart/rollback.html new file mode 100644 index 000000000..8f69b240a --- /dev/null +++ b/v1.46/quickstart/rollback.html @@ -0,0 +1,743 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +5️⃣ Roll back Changes | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Rolling back Changes in lakeFS + + +

+ + +

Our intrepid user (you) merged a change back into the main branch and realised that they had made a mistake 🤦🏻.

+ +

The good news for them (you) is that lakeFS can revert changes made, similar to how you would in Git 😅.

+ +

From your terminal window run lakectl with the revert command:

+ +
docker exec -it lakefs lakectl branch revert lakefs://quickstart/main main --parent-number 1 --yes
+
+

You should see a confirmation of a successful rollback:

+
Branch: lakefs://quickstart/main
+commit main successfully reverted
+
+ +

Back in the object page and the DuckDB query we can see that the original file is now back to how it was: +The lakeFS object viewer with DuckDB query showing that the lakes dataset on main branch has been successfully returned to state prior to the merge.

+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/quickstart/run.html b/v1.46/quickstart/run.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/run.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/try.html b/v1.46/quickstart/try.html new file mode 100644 index 000000000..32f9baa11 --- /dev/null +++ b/v1.46/quickstart/try.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/quickstart/work-with-data-locally.html b/v1.46/quickstart/work-with-data-locally.html new file mode 100644 index 000000000..4991d6345 --- /dev/null +++ b/v1.46/quickstart/work-with-data-locally.html @@ -0,0 +1,984 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +7️⃣ Work with lakeFS data locally | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Work with lakeFS Data Locally + + +

+ + +

When working with lakeFS, there are scenarios where we need to access and manipulate data locally. An example use case for working +locally is machine learning model development. Machine learning model development is dynamic and iterative. To optimize this +process, experiments need to be conducted with speed, tracking ease, and reproducibility. Localizing model data during development +accelerates the process by enabling interactive and offline development and reducing data access latency.

+ +

lakeFS provides 2 ways to expose versioned data locally

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. lakeFS Mount
  2. +
  3. lakectl local
  4. +
+ +
+

+ + + lakeFS Mount + + +

+ + +

lakeFS Mount is available (in preview) for lakeFS Enterprise and lakeFS Cloud customers.
+You can try it out by signing up for the preview

+ + +

+ + + Getting started with lakeFS Mount + + +

+ + +

Prerequisites:

+ +
    +
  • A working lakeFS Server running either lakeFS Enterprise or lakeFS Cloud
  • +
  • You’ve installed the lakectl command line utility: this is the official lakeFS command line interface, on top of which lakeFS Mount is built.
  • +
  • lakectl is configured properly to access your lakeFS server as detailed in the configuration instructions
  • +
+

+ + + Mounting a path to a local directory: + + +

+ + +
    +
  1. +

    In lakeFS create a new branch called my-experiment. You can do this through the UI or with lakectl:

    + +
     docker exec lakefs \
    +     lakectl branch create \
    +         lakefs://quickstart/my-experiment \
    +         --source lakefs://quickstart/main
    +
    +
  2. +
  3. +

    Mount images from your quickstart repository into a local directory named my_local_dir

    +
    everest mount lakefs://quickstart/my-experiment/images my_local_dir
    +
    +
  4. +
+ +

Once complete, my_local_dir should be mounted with the specified path.

+ +
    +
  1. Verify that my_local_dir is linked to the correct path in your lakeFS remote:
  2. +
+ +
ls -l my_local_dir
+
+ +
    +
  1. To unmount the directory, simply run:
  2. +
+ +
everest umount ./my_local_dir
+
+ +

Which will unmount the path and terminate the local mount-server.

+

+ + + lakectl local + + +

+ + +

Alternatively, we can use lakectl local to bring a subset of our lakeFS data to a local directory within the lakeFS +container and edit an image dataset used for ML model development. Unlike lakeFS Mount, using lakectl local requires copying data to/from lakeFS and your local machine.

+ + + +

Reference Guide: lakeFS lakectl local for machine learning

+

+ + + Cloning a Subset of lakeFS Data into a Local Directory + + +

+ + +
    +
  1. +

    In lakeFS create a new branch called my-experiment. You can do this through the UI or with lakectl:

    + +
     docker exec lakefs lakectl branch create lakefs://quickstart/my-experiment --source lakefs://quickstart/main
    +
    +
  2. +
  3. +

    Clone images from your quickstart repository into a local directory named my_local_dir within your container:

    + +
     docker exec lakefs lakectl local clone lakefs://quickstart/my-experiment/images my_local_dir
    +
    +
  4. +
  5. +

    Verify that my_local_dir is linked to the correct path in your lakeFS remote:

    + +
     docker exec lakefs lakectl local list
    +
    + +

    You should see confirmation that my_local_dir is tracking the desired lakeFS path.:

    + +
        my_local_dir	lakefs://quickstart/my-experiment/images/8614575b5488b47a094163bd17a12ed0b82e0bcbfd22ed1856151c671f1faa53
    +
    +
  6. +
  7. +

    Verify that your local environment is up-to-date with its remote path:

    + +
     docker exec lakefs lakectl local status my_local_dir
    +
    +

    You should get a confirmation message like this showing that there is no difference between your local environment and the lakeFS remote:

    + +
    diff 'local:///home/lakefs/my_local_dir' <--> 'lakefs://quickstart/8614575b5488b47a094163bd17a12ed0b82e0bcbfd22ed1856151c671f1faa53/images/'...
    +diff 'lakefs://quickstart/8614575b5488b47a094163bd17a12ed0b82e0bcbfd22ed1856151c671f1faa53/images/' <--> 'lakefs://quickstart/my-experiment/images/'...
    +
    +No diff found.
    +
    +
  8. +
+

+ + + Making Changes to Data Locally + + +

+ + +
    +
  1. +

    Download a new image of an Axolotl and add it to the dataset cloned into my_local_dir:

    + +
     curl -L https://go.lakefs.io/43ENDyS > axolotl.png
    +   
    + docker cp axolotl.png lakefs:/home/lakefs/my_local_dir
    +
    +
  2. +
  3. Clean the dataset by removing images larger than 225 KB: +
     docker exec lakefs find my_local_dir -type f -size +225k -delete
    +
    +
  4. +
  5. Check the status of your local changes compared to the lakeFS remote path: +
     docker exec lakefs lakectl local status my_local_dir
    +
    + +

    You should get a confirmation message like this, showing the modifications you made locally:

    +
     diff 'local:///home/lakefs/my_local_dir' <--> 'lakefs://quickstart/8614575b5488b47a094163bd17a12ed0b82e0bcbfd22ed1856151c671f1faa53/images/'...
    + diff 'lakefs://quickstart/8614575b5488b47a094163bd17a12ed0b82e0bcbfd22ed1856151c671f1faa53/images/' <--> 'lakefs://quickstart/my-experiment/images/'...
    +
    + ╔════════╦══════════╦═════════════════════╗
    + ║ SOURCE ║ CHANGE   ║ PATH                ║
    + ╠════════╬══════════╬═════════════════════╣
    + ║ local  ║ modified ║ axolotl.png         ║
    + ║ local  ║ removed  ║ duckdb-main-02.png  ║
    + ║ local  ║ removed  ║ empty-repo-list.png ║
    + ║ local  ║ removed  ║ repo-contents.png   ║
    + ╚════════╩══════════╩═════════════════════╝
    +
    +
  6. +
+

+ + + Pushing Local Changes to lakeFS + + +

+ + +

Once we are done with editing the image dataset in our local environment, we will push our changes to the lakeFS remote so that +the improved dataset is shared and versioned.

+ +
    +
  1. +

    Commit your local changes to lakeFS:

    + +
     docker exec lakefs lakectl local commit -m 'Deleted images larger than 225KB in size and changed the Axolotl image' my_local_dir
    +
    + +

    In your branch, you should see the commit including your local changes:

    + +

    A lakectl local commit to lakeFS

    +
  2. +
  3. +

    Compare my-experiment branch to the main branch to visualize your changes:

    + +

    A comparison between a branch that includes local changes to the main branch

    +
  4. +
+

+ + + Bonus Challenge + + +

+ + +

And so with that, this quickstart for lakeFS draws to a close. If you’re simply having too much fun to stop then here’s an exercise for you.

+ +

Implement the requirement from the beginning of this quickstart correctly, such that you write denmark-lakes.parquet in the respective branch and successfully merge it back into main. Look up how to list the contents of the main branch and verify that it looks like this:

+ +
object          2023-03-21 17:33:51 +0000 UTC    20.9 kB         denmark-lakes.parquet
+object          2023-03-21 14:45:38 +0000 UTC    916.4 kB        lakes.parquet
+
+

+ + + Finishing Up + + +

+ + +

Once you’ve finished the quickstart, shut down your local environment with the following command:

+ +
docker stop lakefs
+
+ + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/redirects.json b/v1.46/redirects.json new file mode 100644 index 000000000..943ed34c0 --- /dev/null +++ b/v1.46/redirects.json @@ -0,0 +1 @@ +{"/reference/access-control-list.html":"/v1.46/security/access-control-lists.html","/reference/access-control-lists.html":"/v1.46/security/access-control-lists.html","/hooks/airflow.html":"/v1.46/howto/hooks/airflow.html","/using/airflow.html":"/v1.46/integrations/airflow.html","/architecture/index.html":"/v1.46/understand/architecture.html","/architecture/overview.html":"/v1.46/understand/architecture.html","/using/athena.html":"/v1.46/integrations/athena.html","/audit.html":"/v1.46/reference/auditing.html","/reference/audit.html":"/v1.46/reference/auditing.html","/cloud/auditing.html":"/v1.46/reference/auditing.html","/reference/authentication.html":"/v1.46/security/authentication.html","/deploying-aws/index.html":"/v1.46/howto/deploy/aws.html","/deploying-aws/install.html":"/v1.46/howto/deploy/aws.html","/deploying-aws/db.html":"/v1.46/howto/deploy/aws.html","/deploying-aws/lb_dns.html":"/v1.46/howto/deploy/aws.html","/setup/storage/s3.html":"/v1.46/howto/deploy/aws.html","/deploy/aws.html":"/v1.46/howto/deploy/aws.html","/using/aws_cli.html":"/v1.46/integrations/aws_cli.html","/setup/storage/blob.html":"/v1.46/howto/deploy/azure.html","/deploy/azure.html":"/v1.46/howto/deploy/azure.html","/callouts.html":"/v1.46/project/docs/callouts.html","/data_lifecycle_management/ci.html":"/v1.46/understand/data_lifecycle_management/ci.html","/use_cases/cicd_for_data.html":"/v1.46/understand/use_cases/cicd_for_data.html","/usecases/ci.html":"/v1.46/understand/use_cases/cicd_for_data.html","/usecases/cd.html":"/v1.46/understand/use_cases/cicd_for_data.html","/reference/commands.html":"/v1.46/reference/cli.html","/quickstart/lakefs_cli.html":"/v1.46/reference/cli.html","/contributing.html":"/v1.46/project/contributing.html","/integrations/distcp.html":"/v1.46/howto/copying.html","/integrations/rclone.html":"/v1.46/howto/copying.html","/data_lifecycle_management/data-devenv.html":"/v1.46/understand/data_lifecycle_management/data-devenv.html","/usecases/data-devenv.html":"/v1.46/understand/data_lifecycle_management/data-devenv.html","/using/dremio.html":"/v1.46/integrations/dremio.html","/use_cases/etl_testing.html":"/v1.46/understand/use_cases/etl_testing.html","/use_cases/iso_env.html":"/v1.46/understand/use_cases/etl_testing.html","/quickstart/iso_env.html":"/v1.46/understand/use_cases/etl_testing.html","/reference/export.html":"/v1.46/howto/export.html","/reference/external-principals-aws.html":"/v1.46/security/external-principals-aws.html","/faq.html":"/v1.46/understand/faq.html","/howto/garbage-collection/index.html":"/v1.46/howto/garbage-collection/gc.html","/howto/garbage-collection/committed.html":"/v1.46/howto/garbage-collection/gc.html","/howto/garbage-collection/uncommitted.html":"/v1.46/howto/garbage-collection/gc.html","/howto/garbage-collection/internals.html":"/v1.46/howto/garbage-collection/gc.html","/reference/garbage-collection.html":"/v1.46/howto/garbage-collection/gc.html","/howto/garbage-collection-index.html":"/v1.46/howto/garbage-collection/gc.html","/howto/garbage-collection.html":"/v1.46/howto/garbage-collection/gc.html","/reference/retention.html":"/v1.46/howto/garbage-collection/gc.html","/setup/storage/gcs.html":"/v1.46/howto/deploy/gcp.html","/deploy/gcp.html":"/v1.46/howto/deploy/gcp.html","/reference/glossary.html":"/v1.46/understand/glossary.html","/glossary.html":"/v1.46/understand/glossary.html","/using/glue_hive_metastore.html":"/v1.46/integrations/glue_hive_metastore.html","/using/glue_metastore.html":"/v1.46/integrations/glue_metastore.html","/using/hive.html":"/v1.46/integrations/hive.html","/setup/import.html":"/v1.46/howto/import.html","/use_cases/":"/v1.46/understand/use_cases/","/use_cases/index.html":"/v1.46/understand/use_cases/","/branching/recommendations.html":"/v1.46/understand/data_lifecycle_management/","/using_lakefs.html":"/v1.46/understand/data_lifecycle_management/","/setup/":"/v1.46/howto/deploy/","/setup/storage/index.html":"/v1.46/howto/deploy/","/setup/create-repo.html":"/v1.46/howto/deploy/","/deploy/":"/v1.46/howto/deploy/","/deploy/index.html":"/v1.46/howto/deploy/","/reference/hooks.html":"/v1.46/howto/hooks/","/hooks.html":"/v1.46/howto/hooks/","/hooks/overview.html":"/v1.46/howto/hooks/","/hooks/index.html":"/v1.46/howto/hooks/","/hooks/":"/v1.46/howto/hooks/","/setup/hooks.html":"/v1.46/howto/hooks/","/slack/":"https://lakefs.io/slack","/using":"/v1.46/integrations/","/quickstart.html":"/v1.46/quickstart/","/quickstart/installing.html":"/v1.46/quickstart/","/quickstart/try.html":"/v1.46/quickstart/","/quickstart/add_data.html":"/v1.46/quickstart/","/quickstart/more_quickstart_options.html":"/v1.46/quickstart/","/quickstart/repository.html":"/v1.46/quickstart/","/quickstart/run.html":"/v1.46/quickstart/","/quickstart/first_commit.html":"/v1.46/quickstart/","/README.html":"/v1.46/project/docs/","/understand/licensing.html":"/v1.46/project/","/licensing.html":"/v1.46/project/","/commitment.html":"/v1.46/project/","/roadmap.html":"/v1.46/project/","/understand/roadmap.html":"/v1.46/project/","/downloads.html":"/v1.46/","/using/kakfa.html":"/v1.46/integrations/kafka.html","/integrations/kakfa.html":"/v1.46/integrations/kafka.html","/understand/kv-in-a-nutshell.html":"/v1.46/understand/how/kv.html","/hooks/lua.html":"/v1.46/howto/hooks/lua.html","/cloud/managed-gc.html":"/v1.46/howto/garbage-collection/managed-gc.html","/reference/merge.html":"/v1.46/understand/how/merge.html","/understand/merge.html":"/v1.46/understand/how/merge.html","/deploying-aws/offboarding.html":"/v1.46/howto/migrate-away.html","/reference/offboarding.html":"/v1.46/howto/migrate-away.html","/reference/object-model.html":"/v1.46/understand/model.html","/understand/branching-model.html":"/v1.46/understand/model.html","/understand/object-model.html":"/v1.46/understand/model.html","/deploying-aws/monitor.md":"/v1.46/reference/monitor.html","/deploy/k8s.html":"/v1.46/howto/deploy/onprem.html","/deploy/docker.html":"/v1.46/howto/deploy/onprem.html","/integrations/minio.html":"/v1.46/howto/deploy/onprem.html","/using/minio.html":"/v1.46/howto/deploy/onprem.html","/deploy/onprem.html":"/v1.46/howto/deploy/onprem.html","/deploying/install.html":"/v1.46/howto/deploy/onprem.html","/reference/presigned-url.html":"/v1.46/security/presigned-url.html","/integrations/presto.html":"/v1.46/integrations/presto_trino.html","/using/presto.html":"/v1.46/integrations/presto_trino.html","/cloud/private-link.html":"/v1.46/howto/private-link.html","/data_lifecycle_management/production.html":"/v1.46/understand/data_lifecycle_management/production.html","/usecases/production.html":"/v1.46/understand/data_lifecycle_management/production.html","/reference/protected_branches.html":"/v1.46/howto/protect-branches.html","/reference/pull_requests.html":"/v1.46/howto/pull-requests.html","/using/python.html":"/v1.46/integrations/python.html","/using/boto.html":"/v1.46/integrations/python.html","/integrations/boto.html":"/v1.46/integrations/python.html","/reference/authorization.html":"/v1.46/security/rbac.html","/reference/rbac.html":"/v1.46/security/rbac.html","/reference/remote-authenticator.html":"/v1.46/security/sts-login.html","/use_cases/reproducibility.html":"/v1.46/understand/use_cases/reproducibility.html","/use_cases/rollback.html":"/v1.46/understand/use_cases/rollback.html","/using/sagemaker.html":"/v1.46/integrations/sagemaker.html","/cloud/scim.html":"/v1.46/howto/scim.html","/architecture/sizing-guide.html":"/v1.46/howto/sizing-guide.html","/understand/sizing-guide.html":"/v1.46/howto/sizing-guide.html","/integrations/emr.html":"/v1.46/integrations/spark.html","/integrations/glue_etl.html":"/v1.46/integrations/spark.html","/using/databricks.html":"/v1.46/integrations/spark.html","/using/emr.html":"/v1.46/integrations/spark.html","/using/glue_etl.html":"/v1.46/integrations/spark.html","/using/spark.html":"/v1.46/integrations/spark.html","/cloud/sso.html":"/v1.46/security/sso.html","/enterprise/sso.html":"/v1.46/security/sso.html","/cloud/standalone-gc.html":"/v1.46/howto/garbage-collection/standalone-gc.html","/integrations/unity_catalog":"/v1.46/integrations/unity-catalog.html","/cloud/unity-delta-sharing.html":"/v1.46/howto/unity-delta-sharing.html","/cloud/unity-delta-sharing-m0-users":"/v1.46/howto/unity-delta-sharing.html","/deploying-aws/upgrade.html":"/v1.46/howto/deploy/upgrade.html","/reference/upgrade.html":"/v1.46/howto/deploy/upgrade.html","/howto/upgrade.html":"/v1.46/howto/deploy/upgrade.html","/understand/architecture/data-model.html":"/v1.46/understand/how/versioning-internals.html","/understand/understand/data-model.html":"/v1.46/understand/how/versioning-internals.html","/understand/data-model.html":"/v1.46/understand/how/versioning-internals.html","/understand/versioning-internals.html":"/v1.46/understand/how/versioning-internals.html","/hooks/webhooks.html":"/v1.46/howto/hooks/webhooks.html"} \ No newline at end of file diff --git a/v1.46/reference/access-control-list.html b/v1.46/reference/access-control-list.html new file mode 100644 index 000000000..c010c2f92 --- /dev/null +++ b/v1.46/reference/access-control-list.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/access-control-lists.html b/v1.46/reference/access-control-lists.html new file mode 100644 index 000000000..c010c2f92 --- /dev/null +++ b/v1.46/reference/access-control-lists.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/api.html b/v1.46/reference/api.html new file mode 100644 index 000000000..26a07ec9a --- /dev/null +++ b/v1.46/reference/api.html @@ -0,0 +1,714 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakeFS API | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +
+ + + + + + + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/audit.html b/v1.46/reference/audit.html new file mode 100644 index 000000000..0022ee13a --- /dev/null +++ b/v1.46/reference/audit.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/auditing.html b/v1.46/reference/auditing.html new file mode 100644 index 000000000..f429d1ea1 --- /dev/null +++ b/v1.46/reference/auditing.html @@ -0,0 +1,1058 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Auditing | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Auditing + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

Auditing is only available for lakeFS Cloud and lakeFS Enterprise.

+
+ +

The lakeFS audit log allows you to view all relevant user action information in a clear and organized table, including when the action was performed, by whom, and what it was they did.

+ +

This can be useful for several purposes, including:

+ +
    +
  1. +

    Compliance - Audit logs can be used to show what data users accessed, as well as any changes they made to user management.

    +
  2. +
  3. +

    Troubleshooting - If something changes on your underlying object store that you weren’t expecting, such as a big file suddenly breaking into thousands of smaller files, you can use the audit log to find out what action led to this change.

    +
  4. +
+

+ + + Setting up access to Audit Logs on AWS S3 + + +

+ + +

The access to the Audit Logs is done via AWS S3 Access Point.

+ +

There are different ways to interact with an access point (see Using access points in AWS).

+ +

The initial setup:

+ +
    +
  1. Take note of the IAM Role ARN that will be used to access the data. This should be the user or role used by e.g. Athena.
  2. +
  3. Reach out to customer success and provide this ARN. Once receiving the ARN role, an access point will be created and you should get in response the following details: +
      +
    1. S3 Bucket (e.g. arn:aws:s3:::lakefs-audit-logs-us-east-1-production)
    2. +
    3. S3 URI to an access point (e.g. s3://arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>)
    4. +
    5. Access Point alias. You can use this alias instead of the bucket name or Access Point ARN to access data through the Access Point. (e.g. lakefs-logs-<generated>-s3alias)
    6. +
    +
  4. +
  5. Update your IAM Role policy and trust policy if required
  6. +
+ +

A minimal example for IAM policy with 2 lakeFS installations in 2 regions (us-east-1, us-west-2):

+ +
{
+    "Version": "2012-10-17",
+    "Statement": [
+        {
+            "Effect": "Allow",
+            "Action": [
+                "s3:ListBucket"
+            ],
+            "Resource": [
+                "arn:aws:s3:::lakefs-audit-logs-us-east-1-production",
+                "arn:aws:s3:::lakefs-audit-logs-us-east-1-production/*",
+                "arn:aws:s3:::lakefs-logs-<generated>-s3alias/*",
+                "arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>",
+                "arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>/*"
+            ],
+            "Condition": {
+                "StringLike": {
+                    "s3:prefix": [
+                        "etl/v1/data/region=<region_a>/organization=org-<organization>/*",
+                        "etl/v1/data/region=<region_b>/organization=org-<organization>/*"
+                    ]
+                }
+            }
+        },
+        {
+            "Effect": "Allow",
+            "Action": [
+                "s3:GetObject",
+                "s3:GetObjectVersion"
+            ],
+            "Resource": [
+                "arn:aws:s3:::lakefs-audit-logs-us-east-1-production",
+                "arn:aws:s3:::lakefs-audit-logs-us-east-1-production/etl/v1/data/region=<region_a>/organization=org-<organization>/*",
+                "arn:aws:s3:::lakefs-audit-logs-us-east-1-production/etl/v1/data/region=<region_b>/organization=org-<organization>/*",
+                "arn:aws:s3:::lakefs-logs-<generated>-s3alias/*",
+                "arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>/object/etl/v1/data/region=<region_a>/organization=org-<organization>/*",
+                "arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>/object/etl/v1/data/region=<region_b>/organization=org-<organization>/*"
+            ]
+        },
+        {
+            "Action": [
+                "kms:Decrypt"
+            ],
+            "Resource": [
+                "arn:aws:kms:us-east-1:<treeverse-id>:key/<encryption-key-id>"
+            ],
+            "Effect": "Allow"
+        }
+    ]
+}
+
+ +

Trust Policy example that allows anyone in your account to assume the role above:

+ +
{
+    "Version": "2012-10-17",
+    "Statement": [
+        {
+            "Effect": "Allow",
+            "Principal": {
+                "AWS": "arn:aws:iam::<YOUR_ACCOUNT_ID>:root"
+            },
+            "Action": "sts:AssumeRole",
+            "Condition": {}
+        }
+    ]
+}
+
+ +

Authentication is done by assuming an IAM Role:

+ +
# Assume role use AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, AWS_SESSION_TOKEN:
+aws sts assume-role --role-arn arn:aws:iam::<your-aws-account>:role/<reader-role> --role-session-name <name> 
+
+# verify role assumed
+aws sts get-caller-identity 
+
+# list objects (can be used with --recursive) with access point ARN
+aws s3 ls arn:aws:s3:us-east-1:<treeverse-id>:accesspoint/lakefs-logs-<organization>/etl/v1/data/region=<region>/organization=org-<organization>/
+
+# get object locally via s3 access point alias 
+aws s3api get-object --bucket lakefs-logs-<generated>-s3alias --key etl/v1/data/region=<region>/organization=org-<organization>/year=<YY>/month=<MM>/day=<DD>/hour=<HH>/<file>-snappy.parquet sample.parquet 
+
+

+ + + Data layout + + +

+ + +
+

The bucket name is important when creating the IAM policy but, the Access Point ARN and Alias will be the ones that are used to access the data (i.e AWS CLI, Spark etc).

+
+ +

Bucket Name: lakefs-audit-logs-us-east-1-production

+ +

Root prefix: etl/v1/data/region=<region>/organization=org-<organization-name>/

+ +

Files Path pattern: All the audit logs files are in parquet format and their pattern is: etl/v1/data/region=<region>/organization=org-<organization-name>/year=<YY>/month=<MM>/day=<DD>/hour=<HH>/*-snappy.parquet

+

+ + + Path Values + + +

+ + +

region: lakeFS installation region (e.g the region in lakeFS URL: https://..lakefscloud.io/)

+ +

organization: Found in the lakeFS URL https://<organization-name>.<region>.lakefscloud.io/. The value in the S3 path must be prefixed with org-<organization-name>

+

+ + + Partitions + + +

+ + +
    +
  • year
  • +
  • month
  • +
  • day
  • +
  • hour
  • +
+

+ + + Example + + +

+ + +

As an example paths for “Acme” organization with 2 lakeFS installations:

+ +
# ACME in us-east-1 
+etl/v1/data/region=us-east-1/organization=org-acme/year=2024/month=02/day=12/hour=13/log_abc-snappy.parquet
+
+# ACME in us-west-2 
+etl/v1/data/region=us-west-2/organization=org-acme/year=2024/month=02/day=12/hour=13/log_xyz-snappy.parquet
+
+

+ + + Schema + + +

+ + +

The files are in parquet format and can be accessed directly from Spark or any client that can read parquet files. +Using Spark’s printSchema() we can inspect the values, that’s the latest schema with comments on important columns:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
columntypedescription
data_userstringthe internal user ID for the user making the request. if using an external IdP (i.e SSO, Microsoft Entra, etc) it will be the UID represented by the IdP. (see below an example how to extract the info of external IDs in python)
data_repositorystringthe repository ID relevant for this request. Currently only returned for s3_gateway requests
data_refstringthe reference ID (tag, branch, …) relevant for this request. Currently only returned for s3_gateway requests
data_status_codeintHTTP status code returned for this request
data_service_namestringService name for the request. Could be either “rest_api” or “s3_gateway”
data_request_idstringUnique ID representing this request
data_pathstringHTTP path used for this request
data_operation_idstringLogical operation ID for this request. E.g. list_objects, delete_repository, …
data_methodstringHTTP method for the request
data_timestringdatetime representing the start time of this request, in ISO 8601 format
+

+ + + IdP users: map user IDs from audit logs to an email in lakeFS + + +

+ + +

The data_user column in each log represents the user id that performed it.

+ +
    +
  • It might be empty in cases where authentication is not required (e.g login attempt).
  • +
  • If the user is an API user created internally in lakeFS that id is also the name it was given.
  • +
  • data_user might contain an ID to an external IdP (i.e. SSO system), usually it is not human friendly, we can correlate the ID to a lakeFS email used, see an example using the Python lakefs-sdk.
  • +
+ +
import lakefs_sdk
+
+# Configure HTTP basic authorization: basic_auth
+configuration = lakefs_sdk.Configuration(
+    host = "https://<org>.<region>.lakefscloud.io/api/v1",
+    username = 'AKIA...',
+    password = '...'
+)
+
+# Print all user email and uid in lakeFS 
+# the uid is equal to the user id in the audit logs.
+with lakefs_sdk.ApiClient(configuration) as api_client:
+    auth_api = lakefs_sdk.AuthApi(api_client)
+    has_more = True
+    next_offset = ''
+    page_size = 100 
+    while has_more: 
+        resp = auth_api.list_users(prefix='', after=next_offset, amount=page_size)
+        for u in resp.results:
+            email = u.email
+            uid = u.id
+            print(f'Email: {email}, UID: {uid}')
+
+        has_more = resp.pagination.has_more 
+        next_offset = resp.pagination.next_offset
+
+

+ + + Example: Glue Notebook with Spark + + +

+ + +
from awsglue.transforms import *
+from pyspark.context import SparkContext
+from awsglue.context import GlueContext
+from awsglue.job import Job
+
+sc = SparkContext.getOrCreate()
+glueContext = GlueContext(sc)
+spark = glueContext.spark_session
+job = Job(glueContext)
+
+# connect to s3 access point 
+alias = 's3://<bucket-alias-name>'
+s3_dyf = glueContext.create_dynamic_frame.from_options(
+    format_options={},
+    connection_type="s3",
+    format="parquet",
+    connection_options={
+        "paths": [alias + "/etl/v1/data/region=<region>/organization=org-<org>/year=<YY>/month=<MM>/day=<DD>/hour=<HH>/"],
+        "recurse": True,
+    },
+    transformation_ctx="sample-ctx",
+)
+
+s3_dyf.show()
+s3_dyf.printSchema()
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/authentication.html b/v1.46/reference/authentication.html new file mode 100644 index 000000000..c5772a629 --- /dev/null +++ b/v1.46/reference/authentication.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/authorization.html b/v1.46/reference/authorization.html new file mode 100644 index 000000000..3fdf4ca19 --- /dev/null +++ b/v1.46/reference/authorization.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/cli.html b/v1.46/reference/cli.html new file mode 100644 index 000000000..c84e7ecea --- /dev/null +++ b/v1.46/reference/cli.html @@ -0,0 +1,5714 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakectl (lakeFS command-line tool) | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakectl (lakeFS command-line tool) + + +

+ + + +

+ + + Installing lakectl locally + + +

+ + +

lakectl is available for Linux, macOS, and Windows. You can also run it using Docker.

+ +

Download lakectl

+ +

Or using Homebrew for Linux/macOS:

+ +
brew tap treeverse/lakefs
+brew install lakefs
+
+

+ + + Configuring credentials and API endpoint + + +

+ + +

Once you’ve installed the lakectl command, run:

+ +
lakectl config
+# output:
+# Config file /home/janedoe/.lakectl.yaml will be used
+# Access key ID: AKIAIOSFODNN7EXAMPLE
+# Secret access key: ****************************************
+# Server endpoint URL: http://localhost:8000
+
+ +

This will setup a $HOME/.lakectl.yaml file with the credentials and API endpoint you’ve supplied. +When setting up a new installation and creating initial credentials (see Quickstart), the UI +will provide a link to download a preconfigured configuration file for you.

+ +

lakectl configuration items can each be controlled by an environment variable. The variable name will have a prefix of +LAKECTL_, followed by the name of the configuration, replacing every ‘.’ with a ‘_’. Example: LAKECTL_SERVER_ENDPOINT_URL +controls server.endpoint_url.

+

+ + + Running lakectl from Docker + + +

+ + +

If you’d rather run lakectl from a Docker container you can do so by passing configuration elements as environment variables. +Here is an example:

+ +
docker run --rm --pull always \
+          -e LAKECTL_CREDENTIALS_ACCESS_KEY_ID=AKIAIOSFODNN7EXAMPLE \
+          -e LAKECTL_CREDENTIALS_SECRET_ACCESS_KEY=xxxxx
+          -e LAKECTL_SERVER_ENDPOINT_URL=https://host.us-east-2.lakefscloud.io/ \
+          --entrypoint lakectl treeverse/lakefs \
+          repo list
+
+ +

Bear in mind that if you are running lakeFS itself locally you will need to account for this in your networking configuration of +the Docker container. That is to say, localhost to a Docker container is itself, not the host machine on which it is running.

+

+ + + Command Reference + + +

+ +

+ + + lakectl + + +

+ + +

A cli tool to explore manage and work with lakeFS

+

+ + + Synopsis + + +

+ + +

lakectl is a CLI tool allowing exploration and manipulation of a lakeFS environment

+ +
lakectl [flags]
+
+

+ + + Options + + +

+ + +
      --base-uri string      base URI used for lakeFS address parse
+  -c, --config string        config file (default is $HOME/.lakectl.yaml)
+  -h, --help                 help for lakectl
+      --log-format string    set logging output format
+      --log-level string     set logging level (default "none")
+      --log-output strings   set logging output(s)
+      --no-color             don't use fancy output colors (default value can be set by NO_COLOR environment variable)
+      --verbose              run in verbose mode
+  -v, --version              version for lakectl
+
+ +

note: The base-uri option can be controlled with the LAKECTL_BASE_URI environment variable.

+

+ + + Example usage + + +

+ + +
$ export LAKECTL_BASE_URI="lakefs://my-repo/my-branch"
+# Once set, use relative lakefs uri's:
+$ lakectl fs ls /path
+
+

+ + + lakectl actions + + +

+ + +

Manage Actions commands

+

+ + + Options + + +

+ + +
  -h, --help   help for actions
+
+

+ + + lakectl actions help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type actions help [path to command] for full details.

+ +
lakectl actions help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl actions runs + + +

+ + +

Explore runs information

+

+ + + Options + + +

+ + +
  -h, --help   help for runs
+
+

+ + + lakectl actions runs describe + + +

+ + +

Describe run results

+

+ + + Synopsis + + +

+ + +

Show information about the run and all the hooks that were executed as part of the run

+ +
lakectl actions runs describe <repository URI> <run_id> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl actions runs describe lakefs://my-repo 20230719152411arS0z6I
+
+

+ + + Options + + +

+ + +
      --after string   show results after this value (used for pagination)
+      --amount int     number of results to return. By default, all results are returned.
+  -h, --help           help for describe
+
+

+ + + lakectl actions runs help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type runs help [path to command] for full details.

+ +
lakectl actions runs help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl actions runs list + + +

+ + +

List runs

+

+ + + Synopsis + + +

+ + +

List all runs on a repository optional filter by branch or commit

+ +
lakectl actions runs list <repository URI> [--branch <branch>] [--commit <commit_id>] [flags]
+
+

+ + + Examples + + +

+ + +
lakectl actions runs list lakefs://my-repo --branch my-branch --commit 600dc0ffee
+
+

+ + + Options + + +

+ + +
      --after string    show results after this value (used for pagination)
+      --amount int      number of results to return (default 100)
+      --branch string   show results for specific branch
+      --commit string   show results for specific commit ID
+  -h, --help            help for list
+
+

+ + + lakectl actions validate + + +

+ + +

Validate action file

+

+ + + Synopsis + + +

+ + +

Tries to parse the input action file as lakeFS action file

+ +
lakectl actions validate [flags]
+
+

+ + + Examples + + +

+ + +
lakectl actions validate path/to/my/file
+
+

+ + + Options + + +

+ + +
  -h, --help   help for validate
+
+

+ + + lakectl annotate + + +

+ + +

List entries under a given path, annotating each with the latest modifying commit

+ +
lakectl annotate <path URI> [flags]
+
+

+ + + Options + + +

+ + +
      --first-parent   follow only the first parent commit upon seeing a merge commit
+  -h, --help           help for annotate
+  -r, --recursive      recursively annotate all entries under a given path or prefix
+
+

+ + + lakectl auth + + +

+ + +

Manage authentication and authorization

+

+ + + Synopsis + + +

+ + +

Manage authentication and authorization including users, groups and ACLs +This functionality is supported with an external auth service only.

+

+ + + Options + + +

+ + +
  -h, --help   help for auth
+
+

+ + + lakectl auth groups + + +

+ + +

Manage groups

+

+ + + Options + + +

+ + +
  -h, --help   help for groups
+
+

+ + + lakectl auth groups acl + + +

+ + +

Manage ACLs

+

+ + + Synopsis + + +

+ + +

manage ACLs of groups

+

+ + + Options + + +

+ + +
  -h, --help   help for acl
+
+

+ + + lakectl auth groups acl get + + +

+ + +

Get ACL of group

+ +
lakectl auth groups acl get [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for get
+      --id string   Group identifier
+
+

+ + + lakectl auth groups acl help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type acl help [path to command] for full details.

+ +
lakectl auth groups acl help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth groups acl set + + +

+ + +

Set ACL of group

+

+ + + Synopsis + + +

+ + +

Set ACL of group. permission will be attached to all repositories.

+ +
lakectl auth groups acl set [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help                help for set
+      --id string           Group identifier
+      --permission string   Permission, typically one of "Read", "Write", "Super" or "Admin"
+
+

+ + + lakectl auth groups create + + +

+ + +

Create a group

+ +
lakectl auth groups create [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for create
+      --id string   Group identifier
+
+

+ + + lakectl auth groups delete + + +

+ + +

Delete a group

+ +
lakectl auth groups delete [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for delete
+      --id string   Group identifier
+
+

+ + + lakectl auth groups help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type groups help [path to command] for full details.

+ +
lakectl auth groups help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth groups list + + +

+ + +

List groups

+ +
lakectl auth groups list [flags]
+
+

+ + + Options + + +

+ + +
      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth groups members + + +

+ + +

Manage group user memberships

+

+ + + Options + + +

+ + +
  -h, --help   help for members
+
+

+ + + lakectl auth groups members add + + +

+ + +

Add a user to a group

+ +
lakectl auth groups members add [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help          help for add
+      --id string     Group identifier
+      --user string   Username (email for password-based users, default: current user)
+
+

+ + + lakectl auth groups members help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type members help [path to command] for full details.

+ +
lakectl auth groups members help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth groups members list + + +

+ + +

List users in a group

+ +
lakectl auth groups members list [flags]
+
+

+ + + Options + + +

+ + +
      --id string      Group identifier
+      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth groups members remove + + +

+ + +

Remove a user from a group

+ +
lakectl auth groups members remove [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help          help for remove
+      --id string     Group identifier
+      --user string   Username (email for password-based users, default: current user)
+
+

+ + + lakectl auth groups policies + + +

+ + +

Manage group policies

+

+ + + Synopsis + + +

+ + +

Manage group policies. Requires an external authorization server with matching support.

+

+ + + Options + + +

+ + +
  -h, --help   help for policies
+
+

+ + + lakectl auth groups policies attach + + +

+ + +

Attach a policy to a group

+ +
lakectl auth groups policies attach [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help            help for attach
+      --id string       User identifier
+      --policy string   Policy identifier
+
+

+ + + lakectl auth groups policies detach + + +

+ + +

Detach a policy from a group

+ +
lakectl auth groups policies detach [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help            help for detach
+      --id string       User identifier
+      --policy string   Policy identifier
+
+

+ + + lakectl auth groups policies help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type policies help [path to command] for full details.

+ +
lakectl auth groups policies help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth groups policies list + + +

+ + +

List policies for the given group

+ +
lakectl auth groups policies list [flags]
+
+

+ + + Options + + +

+ + +
      --id string      Group identifier
+      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type auth help [path to command] for full details.

+ +
lakectl auth help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth policies + + +

+ + +

Manage policies

+

+ + + Options + + +

+ + +
  -h, --help   help for policies
+
+

+ + + lakectl auth policies create + + +

+ + +

Create a policy

+ +
lakectl auth policies create [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help                        help for create
+      --id string                   Policy identifier
+      --statement-document string   JSON statement document path (or "-" for stdin)
+
+

+ + + lakectl auth policies delete + + +

+ + +

Delete a policy

+ +
lakectl auth policies delete [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for delete
+      --id string   Policy identifier
+
+

+ + + lakectl auth policies help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type policies help [path to command] for full details.

+ +
lakectl auth policies help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth policies list + + +

+ + +

List policies

+ +
lakectl auth policies list [flags]
+
+

+ + + Options + + +

+ + +
      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth policies show + + +

+ + +

Show a policy

+ +
lakectl auth policies show [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for show
+      --id string   Policy identifier
+
+

+ + + lakectl auth users + + +

+ + +

Manage users

+

+ + + Options + + +

+ + +
  -h, --help   help for users
+
+

+ + + lakectl auth users create + + +

+ + +

Create a user

+ +
lakectl auth users create [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for create
+      --id string   Username
+
+

+ + + lakectl auth users credentials + + +

+ + +

Manage user credentials

+

+ + + Options + + +

+ + +
  -h, --help   help for credentials
+
+

+ + + lakectl auth users credentials create + + +

+ + +

Create user credentials

+ +
lakectl auth users credentials create [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for create
+      --id string   Username (email for password-based users, default: current user)
+
+

+ + + lakectl auth users credentials delete + + +

+ + +

Delete user credentials

+ +
lakectl auth users credentials delete [flags]
+
+

+ + + Options + + +

+ + +
      --access-key-id string   Access key ID to delete
+  -h, --help                   help for delete
+      --id string              Username (email for password-based users, default: current user)
+
+

+ + + lakectl auth users credentials help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type credentials help [path to command] for full details.

+ +
lakectl auth users credentials help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth users credentials list + + +

+ + +

List user credentials

+ +
lakectl auth users credentials list [flags]
+
+

+ + + Options + + +

+ + +
      --id string      Username (email for password-based users, default: current user)
+      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth users delete + + +

+ + +

Delete a user

+ +
lakectl auth users delete [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for delete
+      --id string   Username (email for password-based users)
+
+

+ + + lakectl auth users groups + + +

+ + +

Manage user groups

+

+ + + Options + + +

+ + +
  -h, --help   help for groups
+
+

+ + + lakectl auth users groups help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type groups help [path to command] for full details.

+ +
lakectl auth users groups help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth users groups list + + +

+ + +

List groups for the given user

+ +
lakectl auth users groups list [flags]
+
+

+ + + Options + + +

+ + +
      --id string      Username (email for password-based users)
+      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth users help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type users help [path to command] for full details.

+ +
lakectl auth users help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth users list + + +

+ + +

List users

+ +
lakectl auth users list [flags]
+
+

+ + + Options + + +

+ + +
      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl auth users policies + + +

+ + +

Manage user policies

+

+ + + Synopsis + + +

+ + +

Manage user policies. Requires an external authorization server with matching support.

+

+ + + Options + + +

+ + +
  -h, --help   help for policies
+
+

+ + + lakectl auth users policies attach + + +

+ + +

Attach a policy to a user

+ +
lakectl auth users policies attach [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help            help for attach
+      --id string       Username (email for password-based users)
+      --policy string   Policy identifier
+
+

+ + + lakectl auth users policies detach + + +

+ + +

Detach a policy from a user

+ +
lakectl auth users policies detach [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help            help for detach
+      --id string       Username (email for password-based users)
+      --policy string   Policy identifier
+
+

+ + + lakectl auth users policies help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type policies help [path to command] for full details.

+ +
lakectl auth users policies help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl auth users policies list + + +

+ + +

List policies for the given user

+ +
lakectl auth users policies list [flags]
+
+

+ + + Options + + +

+ + +
      --effective      List all distinct policies attached to the user, including by group memberships
+      --id string      Username (email for password-based users)
+      --amount int     how many results to return (default 100)
+      --after string   show results after this value (used for pagination)
+  -h, --help           help for list
+
+

+ + + lakectl branch + + +

+ + +

Create and manage branches within a repository

+

+ + + Synopsis + + +

+ + +

Create delete and list branches within a lakeFS repository

+

+ + + Options + + +

+ + +
  -h, --help   help for branch
+
+

+ + + lakectl branch create + + +

+ + +

Create a new branch in a repository

+ +
lakectl branch create <branch URI> -s <source ref URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch create lakefs://example-repo/new-branch -s lakefs://example-repo/main
+
+

+ + + Options + + +

+ + +
  -h, --help            help for create
+  -s, --source string   source branch uri
+
+

+ + + lakectl branch delete + + +

+ + +

Delete a branch in a repository, along with its uncommitted changes (CAREFUL)

+ +
lakectl branch delete <branch URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch delete lakefs://my-repo/my-branch
+
+

+ + + Options + + +

+ + +
  -h, --help   help for delete
+  -y, --yes    Automatically say yes to all confirmations
+
+

+ + + lakectl branch help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type branch help [path to command] for full details.

+ +
lakectl branch help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl branch list + + +

+ + +

List branches in a repository

+ +
lakectl branch list <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch list lakefs://my-repo
+
+

+ + + Options + + +

+ + +
      --after string   show results after this value (used for pagination)
+      --amount int     number of results to return (default 100)
+  -h, --help           help for list
+
+

+ + + lakectl branch reset + + +

+ + +

Reset uncommitted changes - all of them, or by path

+

+ + + Synopsis + + +

+ + +

reset changes. There are four different ways to reset changes:

+
    +
  1. reset all uncommitted changes - reset lakefs://myrepo/main
  2. +
  3. reset uncommitted changes under specific path - reset lakefs://myrepo/main –prefix path
  4. +
  5. reset uncommitted changes for specific object - reset lakefs://myrepo/main –object path
  6. +
+ +
lakectl branch reset <branch URI> [--prefix|--object] [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch reset lakefs://my-repo/my-branch
+
+

+ + + Options + + +

+ + +
  -h, --help            help for reset
+      --object string   path to object to be reset
+      --prefix string   prefix of the objects to be reset
+  -y, --yes             Automatically say yes to all confirmations
+
+

+ + + lakectl branch revert + + +

+ + +

Given a commit, record a new commit to reverse the effect of this commit

+

+ + + Synopsis + + +

+ + +

The commits will be reverted in left-to-right order

+ +
lakectl branch revert <branch URI> <commit ref to revert> [<more commits>...] [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch revert lakefs://example-repo/example-branch commitA
+	          Revert the changes done by commitA in example-branch
+		      branch revert lakefs://example-repo/example-branch HEAD~1 HEAD~2 HEAD~3
+		      Revert the changes done by the second last commit to the fourth last commit in example-branch
+
+

+ + + Options + + +

+ + +
      --allow-empty-commit   allow empty commit (revert without changes)
+  -h, --help                 help for revert
+  -m, --parent-number int    the parent number (starting from 1) of the mainline. The revert will reverse the change relative to the specified parent.
+  -y, --yes                  Automatically say yes to all confirmations
+
+

+ + + lakectl branch show + + +

+ + +

Show branch latest commit reference

+ +
lakectl branch show <branch URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch show lakefs://my-repo/my-branch
+
+

+ + + Options + + +

+ + +
  -h, --help   help for show
+
+

+ + + lakectl branch-protect + + +

+ + +

Create and manage branch protection rules

+

+ + + Synopsis + + +

+ + +

Define branch protection rules to prevent direct changes. Changes to protected branches can only be done by merging from other branches.

+

+ + + Options + + +

+ + +
  -h, --help   help for branch-protect
+
+

+ + + lakectl branch-protect add + + +

+ + +

Add a branch protection rule

+

+ + + Synopsis + + +

+ + +

Add a branch protection rule for a given branch name pattern

+ +
lakectl branch-protect add <repository URI> <pattern> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch-protect add lakefs://my-repo 'stable_*'
+
+

+ + + Options + + +

+ + +
  -h, --help   help for add
+
+

+ + + lakectl branch-protect delete + + +

+ + +

Delete a branch protection rule

+

+ + + Synopsis + + +

+ + +

Delete a branch protection rule for a given branch name pattern

+ +
lakectl branch-protect delete <repository URI> <pattern> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch-protect delete lakefs://my-repo stable_*
+
+

+ + + Options + + +

+ + +
  -h, --help   help for delete
+
+

+ + + lakectl branch-protect help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type branch-protect help [path to command] for full details.

+ +
lakectl branch-protect help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl branch-protect list + + +

+ + +

List all branch protection rules

+ +
lakectl branch-protect list <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl branch-protect list lakefs://my-repo
+
+

+ + + Options + + +

+ + +
  -h, --help   help for list
+
+

+ + + lakectl cherry-pick + + +

+ + +

Apply the changes introduced by an existing commit

+

+ + + Synopsis + + +

+ + +

Apply the changes from the given commit to the tip of the branch. The changes will be added as a new commit.

+ +
lakectl cherry-pick <commit URI> <branch> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl cherry-pick lakefs://my-repo/600dc0ffee lakefs://my-repo/my-branch
+
+

+ + + Options + + +

+ + +
  -h, --help                help for cherry-pick
+  -m, --parent-number int   the parent number (starting from 1) of the cherry-picked commit. The cherry-pick will apply the change relative to the specified parent.
+
+

+ + + lakectl commit + + +

+ + +

Commit changes on a given branch

+ +
lakectl commit <branch URI> [flags]
+
+

+ + + Options + + +

+ + +
      --allow-empty-commit    allow a commit with no changes
+      --allow-empty-message   allow an empty commit message
+  -h, --help                  help for commit
+  -m, --message string        commit message
+      --meta strings          key value pair in the form of key=value
+
+

+ + + lakectl completion + + +

+ + +

Generate completion script

+

+ + + Synopsis + + +

+ + +

To load completions:

+ +

Bash:

+ +
$ source <(lakectl completion bash)
+
+ +

To load completions for each session, execute once: +Linux:

+ +
$ lakectl completion bash > /etc/bash_completion.d/lakectl
+
+ +

MacOS:

+ +
$ lakectl completion bash > /usr/local/etc/bash_completion.d/lakectl
+
+ +

Zsh:

+ +

If shell completion is not already enabled in your environment you will need +to enable it. You can execute the following once:

+ +
$ echo "autoload -U compinit; compinit" >> ~/.zshrc
+
+ +

To load completions for each session, execute once:

+
$ lakectl completion zsh > "${fpath[1]}/_lakectl"
+
+ +

You will need to start a new shell for this setup to take effect.

+ +

Fish:

+ +
$ lakectl completion fish | source
+
+ +

To load completions for each session, execute once:

+ +
$ lakectl completion fish > ~/.config/fish/completions/lakectl.fish
+
+ +
lakectl completion <bash|zsh|fish>
+
+

+ + + Options + + +

+ + +
  -h, --help   help for completion
+
+

+ + + lakectl config + + +

+ + +

Create/update local lakeFS configuration

+ +
lakectl config [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for config
+
+

+ + + lakectl diff + + +

+ + +

Show changes between two commits, or the currently uncommitted changes

+ +
lakectl diff <ref URI> [ref URI] [flags]
+
+

+ + + Examples + + +

+ + +

+	lakectl diff lakefs://example-repo/example-branch
+	Show uncommitted changes in example-branch.
+
+	lakectl diff lakefs://example-repo/main lakefs://example-repo/dev
+	This shows the differences between master and dev starting at the last common commit.
+	This is similar to the three-dot (...) syntax in git.
+	Uncommitted changes are not shown.
+
+	lakectl diff --two-way lakefs://example-repo/main lakefs://example-repo/dev
+	Show changes between the tips of the main and dev branches.
+	This is similar to the two-dot (..) syntax in git.
+	Uncommitted changes are not shown.
+
+	lakectl diff --two-way lakefs://example-repo/main lakefs://example-repo/dev$
+	Show changes between the tip of the main and the dev branch, including uncommitted changes on dev.
+	
+	lakectl diff --prefix some/path lakefs://example-repo/main lakefs://example-repo/dev
+	Show changes of objects prefixed with 'some/path' between the tips of the main and dev branches.
+
+

+ + + Options + + +

+ + +
  -h, --help            help for diff
+      --prefix string   Show only changes in the given prefix.
+      --two-way         Use two-way diff: show difference between the given refs, regardless of a common ancestor.
+
+

+ + + lakectl doctor + + +

+ + +

Run a basic diagnosis of the LakeFS configuration

+ +
lakectl doctor [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for doctor
+
+

+ + + lakectl fs + + +

+ + +

View and manipulate objects

+

+ + + Options + + +

+ + +
  -h, --help   help for fs
+
+

+ + + lakectl fs cat + + +

+ + +

Dump content of object to stdout

+ +
lakectl fs cat <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help       help for cat
+      --pre-sign   Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+
+

+ + + lakectl fs download + + +

+ + +

Download object(s) from a given repository path

+ +
lakectl fs download <path URI> [<destination path>] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help              help for download
+      --no-progress       Disable progress bar animation for IO operations
+  -p, --parallelism int   Max concurrent operations to perform (default 25)
+      --part-size int     part size in bytes for multipart download (default 8388608)
+      --pre-sign          Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+  -r, --recursive         recursively download all objects under path
+
+

+ + + lakectl fs help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type fs help [path to command] for full details.

+ +
lakectl fs help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl fs ls + + +

+ + +

List entries under a given tree

+ +
lakectl fs ls <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help        help for ls
+  -r, --recursive   list all objects under the specified path
+
+

+ + + lakectl fs presign + + +

+ + +

return a pre-signed URL for reading the specified object

+ +
lakectl fs presign <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for presign
+
+

+ + + lakectl fs rm + + +

+ + +

Delete object

+ +
lakectl fs rm <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -C, --concurrency int   max concurrent single delete operations to send to the lakeFS server (default 50)
+  -h, --help              help for rm
+  -r, --recursive         recursively delete all objects under the specified path
+
+

+ + + lakectl fs stat + + +

+ + +

View object metadata

+ +
lakectl fs stat <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help       help for stat
+      --pre-sign   Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+
+

+ + + lakectl fs upload + + +

+ + +

Upload a local file to the specified URI

+ +
lakectl fs upload <path URI> [flags]
+
+

+ + + Options + + +

+ + +
      --content-type string   MIME type of contents
+  -h, --help                  help for upload
+      --no-progress           Disable progress bar animation for IO operations
+  -p, --parallelism int       Max concurrent operations to perform (default 25)
+      --pre-sign              Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+  -r, --recursive             recursively copy all files under local source
+  -s, --source string         local file to upload, or "-" for stdin
+

+

+ + + lakectl fs stage + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Link an external object with a path in a repository

+

+ + + Synopsis + + +

+ + +

Link an external object with a path in a repository, creating an uncommitted change. +The object location must be outside the repository’s storage namespace

+ +
lakectl fs stage <path URI> [flags]
+
+

+ + + Options + + +

+ + +
      --checksum string       Object MD5 checksum as a hexadecimal string
+      --content-type string   MIME type of contents
+  -h, --help                  help for stage
+      --location string       fully qualified storage location (i.e. "s3://bucket/path/to/object")
+      --meta strings          key value pairs in the form of key=value
+      --mtime int             Object modified time (Unix Epoch in seconds). Defaults to current time
+      --size int              Object size in bytes
+
+

+ + + lakectl fs update-metadata + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Update user metadata on the specified URI

+ +
lakectl fs update-metadata <path URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help               help for update-metadata
+      --metadata strings   Metadata to set, in the form key1=value1,key2=value2
+
+

+ + + lakectl gc + + +

+ + +

Manage the garbage collection policy

+

+ + + Options + + +

+ + +
  -h, --help   help for gc
+
+

+ + + lakectl gc delete-config + + +

+ + +

Deletes the garbage collection policy for the repository

+ +
lakectl gc delete-config <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl gc delete-config lakefs://my-repo
+
+

+ + + Options + + +

+ + +
  -h, --help   help for delete-config
+
+

+ + + lakectl gc get-config + + +

+ + +

Show the garbage collection policy for this repository

+ +
lakectl gc get-config <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl gc get-config lakefs://my-repo
+
+

+ + + Options + + +

+ + +
  -h, --help   help for get-config
+  -p, --json   get rules as JSON
+
+

+ + + lakectl gc help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type gc help [path to command] for full details.

+ +
lakectl gc help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl gc set-config + + +

+ + +

Set garbage collection policy JSON

+

+ + + Synopsis + + +

+ + +

Sets the garbage collection policy JSON. +Example configuration file: +{ + “default_retention_days”: 21, + “branches”: [ + { + “branch_id”: “main”, + “retention_days”: 28 + }, + { + “branch_id”: “dev”, + “retention_days”: 14 + } + ] +}

+ +
lakectl gc set-config <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl gc set-config lakefs://my-repo -f config.json
+
+

+ + + Options + + +

+ + +
  -f, --filename string   file containing the GC policy as JSON
+  -h, --help              help for set-config
+
+

+ + + lakectl help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type lakectl help [path to command] for full details.

+ +
lakectl help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl identity + + +

+ + +

Show identity info

+

+ + + Synopsis + + +

+ + +

Show the info of the user cofigurated in lakectl

+ +
lakectl identity [flags]
+
+

+ + + Examples + + +

+ + +
lakectl identity
+
+

+ + + Options + + +

+ + +
  -h, --help   help for identity
+
+

+ + + lakectl import + + +

+ + +

Import data from external source to a destination branch

+ +
lakectl import --from <object store URI> --to <lakeFS path URI> [flags]
+
+

+ + + Options + + +

+ + +
      --allow-empty-message   allow an empty commit message (default true)
+      --from string           prefix to read from (e.g. "s3://bucket/sub/path/"). must not be in a storage namespace
+  -h, --help                  help for import
+  -m, --message string        commit message
+      --meta strings          key value pair in the form of key=value
+      --no-progress           switch off the progress output
+      --to string             lakeFS path to load objects into (e.g. "lakefs://repo/branch/sub/path/")
+
+

+ + + lakectl ingest + + +

+ + +

Ingest objects from an external source into a lakeFS branch (without actually copying them)

+ +
lakectl ingest --from <object store URI> --to <lakeFS path URI> [--dry-run] [flags]
+
+

+ + + Options + + +

+ + +
  -C, --concurrency int          max concurrent API calls to make to the lakeFS server (default 64)
+      --dry-run                  only print the paths to be ingested
+      --from string              prefix to read from (e.g. "s3://bucket/sub/path/"). must not be in a storage namespace
+  -h, --help                     help for ingest
+      --s3-endpoint-url string   URL to access S3 storage API (by default, use regular AWS S3 endpoint
+      --to string                lakeFS path to load objects into (e.g. "lakefs://repo/branch/sub/path/")
+  -v, --verbose                  print stats for each individual object staged
+
+

+ + + lakectl local + + +

+ + +

Sync local directories with lakeFS paths

+

+ + + Options + + +

+ + +
  -h, --help   help for local
+
+

+ + + lakectl local checkout + + +

+ + +

Sync local directory with the remote state.

+ +
lakectl local checkout [directory] [flags]
+
+

+ + + Options + + +

+ + +
      --all               Checkout given source branch or reference for all linked directories
+  -h, --help              help for checkout
+      --no-progress       Disable progress bar animation for IO operations
+  -p, --parallelism int   Max concurrent operations to perform (default 25)
+      --pre-sign          Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+  -r, --ref string        Checkout the given reference
+  -y, --yes               Automatically say yes to all confirmations
+
+

+ + + lakectl local clone + + +

+ + +

Clone a path from a lakeFS repository into a new directory.

+ +
lakectl local clone <path URI> [directory] [flags]
+
+

+ + + Options + + +

+ + +
      --gitignore         Update .gitignore file when working in a git repository context (default true)
+  -h, --help              help for clone
+      --no-progress       Disable progress bar animation for IO operations
+  -p, --parallelism int   Max concurrent operations to perform (default 25)
+      --pre-sign          Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+
+

+ + + lakectl local commit + + +

+ + +

Commit changes from local directory to the lakeFS branch it tracks.

+ +
lakectl local commit [directory] [flags]
+
+

+ + + Options + + +

+ + +
      --allow-empty-message   allow an empty commit message
+      --force                 Commit changes even if remote branch includes uncommitted changes external to the synced path
+  -h, --help                  help for commit
+  -m, --message string        commit message
+      --meta strings          key value pair in the form of key=value
+      --no-progress           Disable progress bar animation for IO operations
+  -p, --parallelism int       Max concurrent operations to perform (default 25)
+      --pre-sign              Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+
+

+ + + lakectl local help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type local help [path to command] for full details.

+ +
lakectl local help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl local init + + +

+ + +

set a local directory to sync with a lakeFS path.

+ +
lakectl local init <path URI> [directory] [flags]
+
+

+ + + Options + + +

+ + +
      --force       Overwrites if directory already linked to a lakeFS path
+      --gitignore   Update .gitignore file when working in a git repository context (default true)
+  -h, --help        help for init
+
+

+ + + lakectl local list + + +

+ + +

find and list directories that are synced with lakeFS.

+ +
lakectl local list [directory] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for list
+
+

+ + + lakectl local pull + + +

+ + +

Fetch latest changes from lakeFS.

+ +
lakectl local pull [directory] [flags]
+
+

+ + + Options + + +

+ + +
      --force             Reset any uncommitted local change
+  -h, --help              help for pull
+      --no-progress       Disable progress bar animation for IO operations
+  -p, --parallelism int   Max concurrent operations to perform (default 25)
+      --pre-sign          Use pre-signed URLs when downloading/uploading data (recommended) (default true)
+
+

+ + + lakectl local status + + +

+ + +

show modifications (both remote and local) to the directory and the remote location it tracks

+ +
lakectl local status [directory] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help    help for status
+  -l, --local   Don't compare against remote changes
+
+

+ + + lakectl log + + +

+ + +

Show log of commits

+

+ + + Synopsis + + +

+ + +

Show log of commits for a given branch

+ +
lakectl log <branch URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl log --dot lakefs://example-repository/main | dot -Tsvg > graph.svg
+
+

+ + + Options + + +

+ + +
      --after string         show results after this value (used for pagination)
+      --amount int           number of results to return. By default, all results are returned
+      --dot                  return results in a dotgraph format
+      --first-parent         follow only the first parent commit upon seeing a merge commit
+  -h, --help                 help for log
+      --limit                limit result just to amount. By default, returns whether more items are available.
+      --no-merges            skip merge commits
+      --objects strings      show results that contains changes to at least one path in that list of objects. Use comma separator to pass all objects together
+      --prefixes strings     show results that contains changes to at least one path in that list of prefixes. Use comma separator to pass all prefixes together
+      --show-meta-range-id   also show meta range ID
+      --since string         show results since this date-time (RFC3339 format)
+      --stop-at string       a Ref to stop at (included in results)
+
+

+ + + lakectl merge + + +

+ + +

Merge & commit changes from source branch into destination branch

+

+ + + Synopsis + + +

+ + +

Merge & commit changes from source branch into destination branch

+ +
lakectl merge <source ref> <destination ref> [flags]
+
+

+ + + Options + + +

+ + +
      --allow-empty           Allow merge when the branches have the same content
+      --allow-empty-message   allow an empty commit message (default true)
+      --force                 Allow merge into a read-only branch or into a branch with the same content
+  -h, --help                  help for merge
+  -m, --message string        commit message
+      --meta strings          key value pair in the form of key=value
+      --strategy string       In case of a merge conflict, this option will force the merge process to automatically favor changes from the dest branch ("dest-wins") or from the source branch("source-wins"). In case no selection is made, the merge process will fail in case of a conflict
+
+

+ + + lakectl metastore + + +

+ + +

Manage metastore commands

+

+ + + Options + + +

+ + +
  -h, --help   help for metastore
+
+

+ + + lakectl metastore copy + + +

+ + +

Copy or merge table

+

+ + + Synopsis + + +

+ + +

Copy or merge table. the destination table will point to the selected branch

+ +
lakectl metastore copy [flags]
+
+

+ + + Options + + +

+ + +
      --catalog-id string         Glue catalog ID
+      --dbfs-root dbfs:/          dbfs location root will replace dbfs:/ in the location before transforming
+      --from-client-type string   metastore type [hive, glue]
+      --from-schema string        source schema name
+      --from-table string         source table name
+  -h, --help                      help for copy
+      --metastore-uri string      Hive metastore URI
+  -p, --partition strings         partition to copy
+      --serde string              serde to set copy to  [default is  to-table]
+      --to-branch string          lakeFS branch name
+      --to-client-type string     metastore type [hive, glue]
+      --to-schema string          destination schema name [default is from-branch]
+      --to-table string           destination table name [default is  from-table] 
+
+

+ + + lakectl metastore copy-all + + +

+ + +

Copy from one metastore to another

+

+ + + Synopsis + + +

+ + +

copy or merge requested tables between hive metastores. the destination tables will point to the selected branch

+ +
lakectl metastore copy-all [flags]
+
+

+ + + Options + + +

+ + +
      --branch string             lakeFS branch name
+      --continue-on-error         prevent copy-all from failing when a single table fails
+      --dbfs-root dbfs:/          dbfs location root will replace dbfs:/ in the location before transforming
+      --from-address string       source metastore address
+      --from-client-type string   metastore type [hive, glue]
+  -h, --help                      help for copy-all
+      --schema-filter string      filter for schemas to copy in metastore pattern (default ".*")
+      --table-filter string       filter for tables to copy in metastore pattern (default ".*")
+      --to-address string         destination metastore address
+      --to-client-type string     metastore type [hive, glue]
+
+

+ + + lakectl metastore copy-schema + + +

+ + +

Copy schema

+

+ + + Synopsis + + +

+ + +

Copy schema (without tables). the destination schema will point to the selected branch

+ +
lakectl metastore copy-schema [flags]
+
+

+ + + Options + + +

+ + +
      --catalog-id string         Glue catalog ID
+      --dbfs-root dbfs:/          dbfs location root will replace dbfs:/ in the location before transforming
+      --from-client-type string   metastore type [hive, glue]
+      --from-schema string        source schema name
+  -h, --help                      help for copy-schema
+      --metastore-uri string      Hive metastore URI
+      --to-branch string          lakeFS branch name
+      --to-client-type string     metastore type [hive, glue]
+      --to-schema string          destination schema name [default is from-branch]
+
+ + + +

Create symlink table and data

+

+ + + Synopsis + + +

+ + +

create table with symlinks, and create the symlinks in s3 in order to access from external services that could only access s3 directly (e.g athena)

+ +
lakectl metastore create-symlink [flags]
+
+

+ + + Options + + +

+ + +
      --branch string             lakeFS branch name
+      --catalog-id string         Glue catalog ID
+      --from-client-type string   metastore type [hive, glue]
+      --from-schema string        source schema name
+      --from-table string         source table name
+  -h, --help                      help for create-symlink
+      --path string               path to table on lakeFS
+      --repo string               lakeFS repository name
+      --to-schema string          destination schema name
+      --to-table string           destination table name
+
+

+ + + lakectl metastore diff + + +

+ + +

Show column and partition differences between two tables

+ +
lakectl metastore diff [flags]
+
+

+ + + Options + + +

+ + +
      --catalog-id string         Glue catalog ID
+      --from-address string       source metastore address
+      --from-client-type string   metastore type [hive, glue]
+      --from-schema string        source schema name
+      --from-table string         source table name
+  -h, --help                      help for diff
+      --metastore-uri string      Hive metastore URI
+      --to-address string         destination metastore address
+      --to-client-type string     metastore type [hive, glue]
+      --to-schema string          destination schema name 
+      --to-table string           destination table name [default is from-table]
+
+

+ + + lakectl metastore help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type metastore help [path to command] for full details.

+ +
lakectl metastore help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl metastore import-all + + +

+ + +

Import from one metastore to another

+

+ + + Synopsis + + +

+ + +

import requested tables between hive metastores. the destination tables will point to the selected repository and branch +table with location s3://my-s3-bucket/path/to/table +will be transformed to location s3://repo-param/bucket-param/path/to/table

+ +
lakectl metastore import-all [flags]
+
+

+ + + Options + + +

+ + +
      --branch string             lakeFS branch name
+      --continue-on-error         prevent import-all from failing when a single table fails
+      --dbfs-root dbfs:/          dbfs location root will replace dbfs:/ in the location before transforming
+      --from-address string       source metastore address
+      --from-client-type string   metastore type [hive, glue]
+  -h, --help                      help for import-all
+      --repo string               lakeFS repo name
+      --schema-filter string      filter for schemas to copy in metastore pattern (default ".*")
+      --table-filter string       filter for tables to copy in metastore pattern (default ".*")
+      --to-address string         destination metastore address
+      --to-client-type string     metastore type [hive, glue]
+
+

+ + + lakectl repo + + +

+ + +

Manage and explore repos

+

+ + + Options + + +

+ + +
  -h, --help   help for repo
+
+

+ + + lakectl repo create + + +

+ + +

Create a new repository

+ +
lakectl repo create <repository URI> <storage namespace> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl repo create lakefs://my-repo s3://my-bucket
+
+

+ + + Options + + +

+ + +
  -d, --default-branch string   the default branch of this repository (default "main")
+  -h, --help                    help for create
+
+

+ + + lakectl repo delete + + +

+ + +

Delete existing repository

+ +
lakectl repo delete <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl repo delete lakefs://my-repo
+
+

+ + + Options + + +

+ + +
  -h, --help   help for delete
+  -y, --yes    Automatically say yes to all confirmations
+
+

+ + + lakectl repo help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type repo help [path to command] for full details.

+ +
lakectl repo help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl repo list + + +

+ + +

List repositories

+ +
lakectl repo list [flags]
+
+

+ + + Options + + +

+ + +
      --after string   show results after this value (used for pagination)
+      --amount int     number of results to return (default 100)
+  -h, --help           help for list
+

+

+ + + lakectl repo create-bare + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Create a new repository with no initial branch or commit

+ +
lakectl repo create-bare <repository URI> <storage namespace> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl create-bare lakefs://my-repo s3://my-bucket
+
+

+ + + Options + + +

+ + +
  -d, --default-branch string   the default branch name of this repository (will not be created) (default "main")
+  -h, --help                    help for create-bare
+
+

+ + + lakectl show + + +

+ + +

See detailed information about an entity

+

+ + + Options + + +

+ + +
  -h, --help   help for show
+
+

+ + + lakectl show commit + + +

+ + +

See detailed information about a commit

+ +
lakectl show commit <commit URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help                 help for commit
+      --show-meta-range-id   show meta range ID
+
+

+ + + lakectl show help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type show help [path to command] for full details.

+ +
lakectl show help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl tag + + +

+ + +

Create and manage tags within a repository

+

+ + + Synopsis + + +

+ + +

Create delete and list tags within a lakeFS repository

+

+ + + Options + + +

+ + +
  -h, --help   help for tag
+
+

+ + + lakectl tag create + + +

+ + +

Create a new tag in a repository

+ +
lakectl tag create <tag URI> <commit URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl tag create lakefs://example-repo/example-tag lakefs://example-repo/2397cc9a9d04c20a4e5739b42c1dd3d8ba655c0b3a3b974850895a13d8bf9917
+
+

+ + + Options + + +

+ + +
  -f, --force   override the tag if it exists
+  -h, --help    help for create
+
+

+ + + lakectl tag delete + + +

+ + +

Delete a tag from a repository

+ +
lakectl tag delete <tag URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for delete
+
+

+ + + lakectl tag help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type tag help [path to command] for full details.

+ +
lakectl tag help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl tag list + + +

+ + +

List tags in a repository

+ +
lakectl tag list <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl tag list lakefs://my-repo
+
+

+ + + Options + + +

+ + +
      --after string   show results after this value (used for pagination)
+      --amount int     number of results to return (default 100)
+  -h, --help           help for list
+
+

+ + + lakectl tag show + + +

+ + +

Show tag’s commit reference

+ +
lakectl tag show <tag URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for show
+

+

+ + + Undocumented commands + + +

+ + +

note: +⚠️ These commands are plumbing commands and for internal use only. +Avoid using them unless you’re really sure you know what you’re doing, or +have been in contact with lakeFS support!

+

+ + + lakectl abuse + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Abuse a running lakeFS instance. See sub commands for more info.

+

+ + + Options + + +

+ + +
  -h, --help   help for abuse
+
+

+ + + lakectl abuse commit + + +

+ + +

Commits to the source branch repeatedly

+ +
lakectl abuse commit <branch URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int     amount of commits to do (default 100)
+      --gap duration   duration to wait between commits (default 2s)
+  -h, --help           help for commit
+
+

+ + + lakectl abuse create-branches + + +

+ + +

Create a lot of branches very quickly.

+ +
lakectl abuse create-branches <source ref URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int             amount of things to do (default 1000000)
+      --branch-prefix string   prefix to create branches under (default "abuse-")
+      --clean-only             only clean up past runs
+  -h, --help                   help for create-branches
+      --parallelism int        amount of things to do in parallel (default 100)
+
+

+ + + lakectl abuse help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type abuse help [path to command] for full details.

+ +
lakectl abuse help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+ + + +

Link the same object in parallel.

+ +
lakectl abuse link-same-object <branch URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int        amount of link object to do (default 1000000)
+  -h, --help              help for link-same-object
+      --key string        key used for the test (default "linked-object")
+      --parallelism int   amount of link object to do in parallel (default 100)
+
+

+ + + lakectl abuse list + + +

+ + +

List from the source ref

+ +
lakectl abuse list <source ref URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int        amount of lists to do (default 1000000)
+  -h, --help              help for list
+      --parallelism int   amount of lists to do in parallel (default 100)
+      --prefix string     prefix to list under (default "abuse/")
+
+

+ + + lakectl abuse merge + + +

+ + +

Merge non-conflicting objects to the source branch in parallel

+ +
lakectl abuse merge <branch URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int        amount of merges to perform (default 1000)
+  -h, --help              help for merge
+      --parallelism int   number of merges to perform in parallel (default 100)
+
+

+ + + lakectl abuse random-delete + + +

+ + +

Delete keys from a file and generate random delete from the source ref for those keys.

+ +
lakectl abuse random-delete <source ref URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int         amount of reads to do (default 1000000)
+      --from-file string   read keys from this file ("-" for stdin)
+  -h, --help               help for random-delete
+      --parallelism int    amount of reads to do in parallel (default 100)
+
+

+ + + lakectl abuse random-read + + +

+ + +

Read keys from a file and generate random reads from the source ref for those keys.

+ +
lakectl abuse random-read <source ref URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int         amount of reads to do (default 1000000)
+      --from-file string   read keys from this file ("-" for stdin)
+  -h, --help               help for random-read
+      --parallelism int    amount of reads to do in parallel (default 100)
+
+

+ + + lakectl abuse random-write + + +

+ + +

Generate random writes to the source branch

+ +
lakectl abuse random-write <branch URI> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int        amount of writes to do (default 1000000)
+  -h, --help              help for random-write
+      --parallelism int   amount of writes to do in parallel (default 100)
+      --prefix string     prefix to create paths under (default "abuse/")
+
+

+ + + lakectl bisect + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Binary search to find the commit that introduced a bug

+

+ + + Options + + +

+ + +
  -h, --help   help for bisect
+
+

+ + + lakectl bisect bad + + +

+ + +

Set ‘bad’ commit that is known to contain the bug

+ +
lakectl bisect bad [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for bad
+
+

+ + + lakectl bisect good + + +

+ + +

Set current commit as ‘good’ commit that is known to be before the bug was introduced

+ +
lakectl bisect good [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for good
+
+

+ + + lakectl bisect help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type bisect help [path to command] for full details.

+ +
lakectl bisect help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+
+

+ + + lakectl bisect log + + +

+ + +

Print out the current bisect state

+ +
lakectl bisect log [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for log
+
+

+ + + lakectl bisect reset + + +

+ + +

Clean up the bisection state

+ +
lakectl bisect reset [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for reset
+
+

+ + + lakectl bisect run + + +

+ + +

Bisecting based on command status code

+ +
lakectl bisect run <command> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for run
+
+

+ + + lakectl bisect start + + +

+ + +

Start a bisect session

+ +
lakectl bisect start <bad ref URI> <good ref URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for start
+
+

+ + + lakectl bisect view + + +

+ + +

Current bisect commits

+ +
lakectl bisect view [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for view
+
+

+ + + lakectl cat-hook-output + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Cat actions hook output

+ +
lakectl cat-hook-output <repository URI> <run_id> <hook_id> [flags]
+
+

+ + + Examples + + +

+ + +
lakectl cat-hook-output lakefs://my-repo 20230719152411arS0z6I my_hook_name
+
+

+ + + Options + + +

+ + +
  -h, --help   help for cat-hook-output
+
+

+ + + lakectl cat-sst + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Explore lakeFS .sst files

+ +
lakectl cat-sst <sst-file> [flags]
+
+

+ + + Options + + +

+ + +
      --amount int    how many records to return, or -1 for all records (default -1)
+  -f, --file string   path to an sstable file, or "-" for stdin
+  -h, --help          help for cat-sst
+
+

+ + + lakectl docs + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +
lakectl docs [outfile] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for docs
+
+

+ + + lakectl find-merge-base + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Find the commits for the merge operation

+ +
lakectl find-merge-base <source ref URI> <destination ref URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for find-merge-base
+
+

+ + + lakectl refs-dump + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Dumps refs (branches, commits, tags) to the underlying object store

+ +
lakectl refs-dump <repository URI> [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help                     help for refs-dump
+  -o, --output string            output filename (default stdout)
+      --poll-interval duration   poll status check interval (default 3s)
+      --timeout duration         timeout for polling status checks (default 1h0m0s)
+
+

+ + + lakectl refs-restore + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Restores refs (branches, commits, tags) from the underlying object store to a bare repository

+

+ + + Synopsis + + +

+ + +

restores refs (branches, commits, tags) from the underlying object store to a bare repository.

+ +

This command is expected to run on a bare repository (i.e. one created with ‘lakectl repo create-bare’). +Since a bare repo is expected, in case of transient failure, delete the repository and recreate it as bare and retry.

+ +
lakectl refs-restore <repository URI> [flags]
+
+

+ + + Examples + + +

+ + +
aws s3 cp s3://bucket/_lakefs/refs_manifest.json - | lakectl refs-restore lakefs://my-bare-repository --manifest -
+
+

+ + + Options + + +

+ + +
  -h, --help                     help for refs-restore
+      --manifest refs-dump       path to a refs manifest json file (as generated by refs-dump). Alternatively, use "-" to read from stdin
+      --poll-interval duration   poll status check interval (default 3s)
+      --timeout duration         timeout for polling status checks (default 1h0m0s)
+
+

+ + + lakectl usage + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Usage reports from lakeFS

+

+ + + Options + + +

+ + +
  -h, --help   help for usage
+
+

+ + + lakectl usage help + + +

+ + +

Help about any command

+

+ + + Synopsis + + +

+ + +

Help provides help for any command in the application. +Simply type usage help [path to command] for full details.

+ +
lakectl usage help [command] [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for help
+

+

+ + + lakectl usage summary + + +

+ + +

note: +lakeFS plumbing command. Don’t use unless you’re really sure you know what you’re doing.

+ +

Summary reports from lakeFS

+ +
lakectl usage summary [flags]
+
+

+ + + Options + + +

+ + +
  -h, --help   help for summary
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/commands.html b/v1.46/reference/commands.html new file mode 100644 index 000000000..6c6c5a594 --- /dev/null +++ b/v1.46/reference/commands.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/configuration.html b/v1.46/reference/configuration.html new file mode 100644 index 000000000..08762d6ad --- /dev/null +++ b/v1.46/reference/configuration.html @@ -0,0 +1,1521 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +lakeFS Server Configuration | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Server Configuration + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Reference
  2. +
  3. Using Environment Variables
  4. +
  5. Example Configurations
  6. +
+ +
+ +

Configuring lakeFS is done using a YAML configuration file and/or environment variable. +The configuration file’s location can be set with the ‘–config’ flag. If not specified, the first file found in the following order will be used:

+
    +
  1. ./config.yaml
  2. +
  3. $HOME/lakefs/config.yaml
  4. +
  5. /etc/lakefs/config.yaml
  6. +
  7. $HOME/.lakefs.yaml
  8. +
+ +

Configuration items can each be controlled by an environment variable. The variable name will have a prefix of LAKEFS_, followed by the name of the configuration, replacing every ‘.’ with a ‘_’. +Example: LAKEFS_LOGGING_LEVEL controls logging.level.

+ +

This reference uses . to denote the nesting of values.

+

+ + + Reference + + +

+ + +
    +
  • listen_address (string : "0.0.0.0:8000") - A <host>:<port> structured string representing the address to listen on
  • +
+

+ + + logging + + +

+ + +
    +
  • logging.format (one of ["json", "text"] : "text") - Format to output log message in
  • +
  • logging.level (one of ["TRACE", "DEBUG", "INFO", "WARN", "ERROR", "NONE"] : "INFO") - Logging level to output
  • +
  • +

    logging.audit_log_level (one of ["TRACE", "DEBUG", "INFO", "WARN", "ERROR", "NONE"] : "DEBUG") - Audit logs level to output.

    + +

    Note: In case you configure this field to be lower than the main logger level, you won’t be able to get the audit logs

    +
  • +
  • logging.output (string : "-") - A path or paths to write logs to. A - means the standard output, = means the standard error.
  • +
  • logging.file_max_size_mb (int : 100) - Output file maximum size in megabytes.
  • +
  • logging.files_keep (int : 0) - Number of log files to keep, default is all.
  • +
+

+ + + actions + + +

+ + +
    +
  • actions.enabled (bool : true) - Setting this to false will block hooks from being executed.
  • +
  • actions.lua.net_http_enabled (bool : false) - Setting this to true will load the net/http package.
  • +
  • actions.env.enabled (bool : true) - Environment variables accessible by hooks, disabled values evaluated to empty strings
  • +
  • actions.env.prefix (string : "LAKEFSACTION_") - Access to environment variables is restricted to those with the prefix. When environment access is enabled and no prefix is provided, all variables are accessible.
  • +
+

+ + + database + + +

+ + +

Configuration section for the lakeFS key-value store database.

+
    +
  • database.type (string ["postgres"|"dynamodb"|"cosmosdb"|"local"] : ) - +lakeFS database type
  • +
+

+ + + database.postgres + + +

+ + +

Configuration section when using database.type="postgres"

+
    +
  • database.postgres.connection_string (string : "postgres://localhost:5432/postgres?sslmode=disable") - PostgreSQL connection string to use
  • +
  • database.postgres.max_open_connections (int : 25) - Maximum number of open connections to the database
  • +
  • database.postgres.max_idle_connections (int : 25) - Maximum number of connections in the idle connection pool
  • +
  • database.postgres.connection_max_lifetime (duration : 5m) - Sets the maximum amount of time a connection may be reused (valid units: ns|us|ms|s|m|h)
  • +
+

+ + + database.dynamodb + + +

+ + +

Configuration section when using database.type="dynamodb"

+
    +
  • database.dynamodb.table_name (string : "kvstore") - Table used to store the data
  • +
  • database.dynamodb.scan_limit (int : 1025) - Maximal number of items per page during scan operation + +
  • +
  • database.dynamodb.endpoint (string : ) - Endpoint URL for database instance
  • +
  • database.dynamodb.aws_region (string : ) - AWS Region of database instance
  • +
  • database.dynamodb.aws_profile (string : ) - AWS named profile to use
  • +
  • database.dynamodb.aws_access_key_id (string : ) - AWS access key ID
  • +
  • database.dynamodb.aws_secret_access_key (string : ) - AWS secret access key +
      +
    • Note: endpoint aws_region aws_access_key_id aws_secret_access_key are not required and used mainly for experimental purposes when working with DynamoDB with different AWS credentials.
    • +
    +
  • +
  • database.dynamodb.health_check_interval (duration : 0s) - Interval to run health check for the DynamoDB instance (won’t run if equal to 0).
  • +
  • database.dynamodb.max_attempts (int : 10) - The maximum number of attempts to perform on a DynamoDB request
  • +
  • database.dynamodb.max_connections (int : 0) - The maximum number of connections to DynamoDB. 0 means no limit.
  • +
+

+ + + database.cosmosdb + + +

+ + +

Configuration section when using database.type="cosmosdb"

+
    +
  • database.cosmosdb.key (string : "") - If specified, will +be used to authenticate to the CosmosDB account. Otherwise, Azure SDK +default authentication (with env vars) will be used.
  • +
  • database.cosmosdb.endpoint (string : "") - CosmosDB account endpoint, e.g. https://<account>.documents.azure.com/.
  • +
  • database.cosmosdb.database (string : "") - CosmosDB database name.
  • +
  • database.cosmosdb.container (string : "") - CosmosDB container name.
  • +
  • database.cosmosdb.throughput (int32 : ) - CosmosDB container’s RU/s. If not set - the default CosmosDB container throughput is used.
  • +
  • database.cosmosdb.autoscale (bool : false) - If set, CosmosDB container throughput is autoscaled (See CosmosDB docs for minimum throughput requirement). Otherwise, uses “Manual” mode (Docs).
  • +
+

+ + + database.local + + +

+ + +

Configuration section when using database.type="local"

+
    +
  • database.local.path (string : "~/lakefs/metadata") - Local path on the filesystem to store embedded KV metadata, like branches and uncommitted entries
  • +
  • database.local.sync_writes (bool: true) - Ensure each write is written to the disk. Disable to increase performance
  • +
  • database.local.prefetch_size (int: 256) - How many items to prefetch when iterating over embedded KV records
  • +
  • database.local.enable_logging (bool: false) - Enable trace logging for local driver
  • +
+

+ + + auth + + +

+ + +
    +
  • auth.login_duration (time duration : "168h") - The duration the login token is valid for
  • +
  • auth.login_max_duration (time duration : "168h") - The maximum duration user can ask for a login token
  • +
  • auth.cookie_domain (string : "") - Domain attribute to set the access_token cookie on (the default is an empty string which defaults to the same host that sets the cookie)
  • +
  • auth.ui_config.rbac (string: "none") - “none”, “simplified”, “external” or “internal” (enterprise feature). +If you have configured an external auth server you can set this to “external” to support the policy editor. +If you are using the enteprrise version of lakeFS, you can set this to “internal” to use the built-in policy editor.
  • +
+

+ + + auth.cache + + +

+ + +
    +
  • auth.cache.enabled (bool : true) - Whether to cache access credentials and user policies in-memory. Can greatly improve throughput when enabled.
  • +
  • auth.cache.size (int : 1024) - How many items to store in the auth cache. Systems with a very high user count should use a larger value at the expense of ~1kb of memory per cached user.
  • +
  • auth.cache.ttl (time duration : "20s") - How long to store an item in the auth cache. Using a higher value reduces load on the database, but will cause changes longer to take effect for cached users.
  • +
  • auth.cache.jitter (time duration : "3s") - A random amount of time between 0 and this value is added to each item’s TTL. This is done to avoid a large bulk of keys expiring at once and overwhelming the database.
  • +
  • auth.encrypt.secret_key (string : required) - A random (cryptographically safe) generated string that is used for encryption and HMAC signing +Note: It is best to keep this somewhere safe such as KMS or Hashicorp Vault, and provide it to the system at run time
  • +
+

+ + + auth.api + + +

+ + +
    +
  • auth.api.endpoint (string: https://external.service/api/v1) - URL to external Authorization Service described at authorization.yml;
  • +
  • auth.api.token (string: eyJhbGciOiJIUzI1NiIsInR5...) - API token used to authenticate requests to api endpoint
  • +
  • auth.api.health_check_timeout (time duration : "20s") - Timeout duration for external auth API health check
  • +
  • auth.api.skip_health_check (bool : false) - Skip external auth API health check
  • +
+

+ + + auth.authentication_api + + +

+ + +
    +
  • auth.authentication_api.endpoint (string: https://external.authentication-service/api/v1) - URL to external Authentication Service described at authentication.yml
  • +
  • auth.authentication_api.endpoint (string : "") - URL to external Authentication Service described at authentication.yml;
  • +
  • auth.authentication_api.external_principals_enabled (bool : false) - If true, external principals API will be enabled, e.g auth service and login api’s.
  • +
+

+ + + auth.remote_authenticator + + +

+ + +
    +
  • auth.remote_authenticator.enabled (bool : false) - If specified, also authenticate users via this Remote Authenticator server.
  • +
  • auth.remote_authenticator.endpoint (string : required) - Endpoint URL of the remote authentication service (e.g. https://my-auth.example.com/auth).
  • +
  • auth.remote_authenticator.default_user_group (string : Viewers) - Create users in this group (i.e Viewers, Developers, etc).
  • +
  • auth.remote_authenticator.request_timeout (duration : 10s) - If specified, timeout for remote authentication requests.
  • +
+

+ + + auth.cookie_auth_verification + + +

+ + +
    +
  • auth.cookie_auth_verification.validate_id_token_claims (map[string]string : ) - When a user tries to access lakeFS, validate that the ID token contains these claims with the corresponding values.
  • +
  • auth.cookie_auth_verification.default_initial_groups (string[] : [])` - By default, users will be assigned to these groups
  • +
  • auth.cookie_auth_verification.initial_groups_claim_name (string[] : []) - Use this claim from the ID token to provide the initial group for new users. This will take priority if auth.cookie_auth_verification.default_initial_groups is also set.
  • +
  • auth.cookie_auth_verification.friendly_name_claim_name (string[] : ) - If specified, the value from the claim with this name will be used as the user’s display name.
  • +
  • auth.cookie_auth_verification.persist_friendly_name (string : false) - If set to true, the friendly name is persisted to the KV store and can be displayed in the user list. This is meant to be used in conjunction with auth.cookie_auth_verification.friendly_name_claim_name.
  • +
  • auth.cookie_auth_verification.external_user_id_claim_name - (string : ) - If specified, the value from the claim with this name will be used as the user’s id name.
  • +
  • auth.cookie_auth_verification.auth_source - (string : ) - If specified, user will be labeled with this auth source.
  • +
+

+ + + auth.oidc + + +

+ + +
    +
  • auth.oidc.default_initial_groups (string[] : []) - By default, OIDC users will be assigned to these groups
  • +
  • auth.oidc.initial_groups_claim_name (string[] : []) - Use this claim from the ID token to provide the initial group for new users. This will take priority if auth.oidc.default_initial_groups is also set.
  • +
  • auth.oidc.friendly_name_claim_name (string[] : ) - If specified, the value from the claim with this name will be used as the user’s display name.
  • +
  • auth.oidc.persist_friendly_name (string : false) - If set to true, the friendly name is persisted to the KV store and can be displayed in the user list. This is meant to be used in conjunction with auth.oidc.friendly_name_claim_name.
  • +
  • auth.oidc.validate_id_token_claims (map[string]string : ) - When a user tries to access lakeFS, validate that the ID token contains these claims with the corresponding values.
  • +
+

+ + + blockstore + + +

+ + +
    +
  • blockstore.type (one of ["local", "s3", "gs", "azure", "mem"] : required). Block adapter to use. This controls where the underlying data will be stored
  • +
  • blockstore.default_namespace_prefix (string : ) - Use this to help your users choose a storage namespace for their repositories. + If specified, the storage namespace will be filled with this default value as a prefix when creating a repository from the UI. + The user may still change it to something else.
  • +
  • blockstore.signing.secret_key (string : required) - A random generated string that is used for HMAC signing when using get/link physical address
  • +
+

+ + + blockstore.local + + +

+ + +
    +
  • blockstore.local.path (string: "~/lakefs/data") - When using the local Block Adapter, which directory to store files in
  • +
  • blockstore.local.import_enabled (bool: false) - Enable import for local Block Adapter, relevant only if you are using shared location
  • +
  • blockstore.local.import_hidden (bool: false) - When enabled import will scan and import any file or folder that starts with a dot character.
  • +
  • blockstore.local.allowed_external_prefixes ([]string: []) - List of absolute path prefixes used to match any access for external location (ex: /var/data/). Empty list mean no access to external location.
  • +
+

+ + + blockstore.s3 + + +

+ + +
    +
  • blockstore.s3.region (string : "us-east-1") - Default region for lakeFS to use when interacting with S3.
  • +
  • blockstore.s3.profile (string : ) - If specified, will be used as a named credentials profile
  • +
  • blockstore.s3.credentials_file (string : ) - If specified, will be used as a credentials file
  • +
  • blockstore.s3.credentials.access_key_id (string : ) - If specified, will be used as a static set of credential
  • +
  • blockstore.s3.credentials.secret_access_key (string : ) - If specified, will be used as a static set of credential
  • +
  • blockstore.s3.credentials.session_token (string : ) - If specified, will be used as a static session token
  • +
  • blockstore.s3.endpoint (string : ) - If specified, custom endpoint for the AWS S3 API (https://s3_compatible_service_endpoint:port)
  • +
  • blockstore.s3.force_path_style (bool : false) - When true, use path-style S3 URLs (https:/// instead of https://.)
  • +
  • blockstore.s3.discover_bucket_region (bool : true) - (Can be turned off if the underlying S3 bucket doesn’t support the GetBucketRegion API).
  • +
  • blockstore.s3.skip_verify_certificate_test_only (bool : false) - Skip certificate verification while connecting to the storage endpoint. Should be used only for testing.
  • +
  • blockstore.s3.server_side_encryption (string : ) - Server side encryption format used (Example on AWS using SSE-KMS while passing “aws:kms”)
  • +
  • blockstore.s3.server_side_encryption_kms_key_id (string : ) - Server side encryption KMS key ID
  • +
  • blockstore.s3.pre_signed_expiry (time duration : "15m") - Expiry of pre-signed URL.
  • +
  • blockstore.s3.pre_signed_endpoint (string : ) - Custom endpoint for pre-signed URLs.
  • +
  • blockstore.s3.disable_pre_signed (bool : false) - Disable use of pre-signed URL.
  • +
  • blockstore.s3.disable_pre_signed_ui (bool : true) - Disable use of pre-signed URL in the UI.
  • +
  • blockstore.s3.disable_pre_signed_multipart (bool : ) - Disable use of pre-signed multipart upload experimental, enabled on s3 block adapter with presign support.
  • +
  • blockstore.s3.client_log_request (bool : false) - Set SDK logging bit to log requests
  • +
  • blockstore.s3.client_log_retries (bool : false) - Set SDK logging bit to log retries
  • +
+

+ + + blockstore.azure + + +

+ + +
    +
  • blockstore.azure.storage_account (string : ) - If specified, will be used as the Azure storage account
  • +
  • blockstore.azure.storage_access_key (string : ) - If specified, will be used as the Azure storage access key
  • +
  • blockstore.azure.pre_signed_expiry (time duration : "15m") - Expiry of pre-signed URL.
  • +
  • blockstore.azure.disable_pre_signed (bool : false) - Disable use of pre-signed URL.
  • +
  • blockstore.azure.disable_pre_signed_ui (bool : true) - Disable use of pre-signed URL in the UI.
  • +
  • blockstore.azure.china_cloud (bool : false) - Enable for using lakeFS on Azure China Cloud.
    +Note: Deprecated - In favor of blockstore.azure.domain
  • +
  • blockstore.azure.domain (string : blob.core.windows.net) - Enables support of different Azure cloud domains. Current supported domains (in Beta stage): [blob.core.chinacloudapi.cn, blob.core.usgovcloudapi.net]
  • +
+

+ + + blockstore.gs + + +

+ + +
    +
  • blockstore.gs.credentials_file (string : ) - If specified will be used as a file path of the JSON file that contains your Google service account key
  • +
  • blockstore.gs.credentials_json (string : ) - If specified will be used as JSON string that contains your Google service account key (when credentials_file is not set)
  • +
  • blockstore.gs.pre_signed_expiry (time duration : "15m") - Expiry of pre-signed URL.
  • +
  • blockstore.gs.disable_pre_signed (bool : false) - Disable use of pre-signed URL.
  • +
  • blockstore.gs.disable_pre_signed_ui (bool : true) - Disable use of pre-signed URL in the UI.
  • +
  • blockstore.gs.server_side_encryption_customer_supplied (string : ) - Server side encryption with AES key in hex format, exclusive with key ID below
  • +
  • blockstore.gs.server_side_encryption_kms_key_id (string : ) - Server side encryption KMS key ID, exclusive with above
  • +
+

+ + + graveler + + +

+ + +
    +
  • graveler.ensure_readable_root_namespace (bool: true) - When creating a new repository use this to verify that lakeFS has access to the root of the underlying storage namespace. Set false only if lakeFS should not have access (i.e pre-sign mode only).
  • +
  • graveler.max_batch_delay (duration : 3ms) - Controls the server batching period for references store operations.
  • +
  • graveler.background.rate_limit (int : 0) - Requests per seconds limit on background work performed (default: 0 - unlimited), like deleting committed staging tokens.
  • +
+

+ + + graveler.repository_cache + + +

+ + +
    +
  • graveler.reposiory_cache.size (int : 1000) - How many items to store in the repository cache.
  • +
  • graveler.reposiory_cache.ttl (time duration : "5s") - How long to store an item in the repository cache.
  • +
  • graveler.reposiory_cache.jitter (time duration : "2s") - A random amount of time between 0 and this value is added to each item’s TTL.
  • +
+

+ + + graveler.commit_cache + + +

+ + +
    +
  • graveler.commit_cache.size (int : 50000) - How many items to store in the commit cache.
  • +
  • graveler.commit_cache.ttl (time duration : "10m") - How long to store an item in the commit cache.
  • +
  • graveler.commit_cache.jitter (time duration : "2s") - A random amount of time between 0 and this value is added to each item’s TTL.
  • +
+

+ + + committed + + +

+ + +
    +
  • committed.block_storage_prefix (string : _lakefs) - Prefix for metadata file storage +in each repository’s storage namespace
  • +
  • committed.sstable.memory.cache_size_bytes (int : 200_000_000) - maximal size of +in-memory cache used for each SSTable reader.
  • +
+

+ + + committed.local_cache + + +

+ + +

An object describing the local (on-disk) cache of metadata from permanent storage.

+
    +
  • committed.local_cache.size_bytes (int : 1073741824) - bytes for local cache to use on disk. The cache may use more storage for short periods of time.
  • +
  • committed.local_cache.dir (string, ~/lakefs/local_tier) - directory to store local cache.
  • +
  • committed.local_cache.range_proportion (float : 0.9) - proportion of local cache to +use for storing ranges (leaves of committed metadata storage).
  • +
  • committed.local_cache.range.open_readers (int : 500) - maximal number of unused open +SSTable readers to keep for ranges.
  • +
  • committed.local_cache.range.num_shards (int : 30) - sharding factor for open SSTable +readers for ranges. Should be at least sqrt(committed.local_cache.range.open_readers).
  • +
  • committed.local_cache.metarange_proportion (float : 0.1) - proportion of local cache +to use for storing metaranges (roots of committed metadata storage).
  • +
  • committed.local_cache.metarange.open_readers (int : 50) - maximal number of unused open +SSTable readers to keep for metaranges.
  • +
  • committed.local_cache.metarange.num_shards (int : 10) - sharding factor for open +SSTable readers for metaranges. Should be at least +sqrt(committed.local_cache.metarange.open_readers).
  • +
+

+ + + committed.permanent + + +

+ + +
    +
  • committed.permanent.min_range_size_bytes (int : 0) - Smallest allowable range in +metadata. Increase to somewhat reduce random access time on committed metadata, at the cost +of increased committed metadata storage cost.
  • +
  • committed.permanent.max_range_size_bytes (int : 20971520) - Largest allowable range in +metadata. Should be close to the size at which fetching from remote storage becomes linear.
  • +
  • committed.permanent.range_raggedness_entries (int : 50_000) - Average number of object +pointers to store in each range (subject to min_range_size_bytes and +max_range_size_bytes).
  • +
+

+ + + email + + +

+ + +
    +
  • email.smtp_host (string) - A string representing the URL of the SMTP host.
  • +
  • email.smtp_port (int) - An integer representing the port of the SMTP service (465, 587, 993, 25 are some standard ports)
  • +
  • email.use_ssl (bool : false) - Use SSL connection with SMTP host.
  • +
  • email.username (string) - A string representing the username of the specific account at the SMTP. It’s recommended to provide this value at runtime from a secret vault of some sort.
  • +
  • email.password (string) - A string representing the password of the account. It’s recommended to provide this value at runtime from a secret vault of some sort.
  • +
  • email.local_name (string) - A string representing the hostname sent to the SMTP server with the HELO command. By default, “localhost” is sent.
  • +
  • email.sender (string) - A string representing the email account which is set as the sender.
  • +
  • email.limit_every_duration (duration : 1m) - The average time between sending emails. If zero is entered, there is no limit to the amount of emails that can be sent.
  • +
  • email.burst (int: 10) - Maximal burst of emails before applying limit_every_duration. The zero value means no burst and therefore no emails can be sent.
  • +
  • email.lakefs_base_url (string : "http://localhost:8000") - A string representing the base lakeFS endpoint to be directed to when emails are sent inviting users, reseting passwords etc.
  • +
+

+ + + gateways + + +

+ + +
    +
  • gateways.s3.domain_name (string : "s3.local.lakefs.io") - a FQDN +representing the S3 endpoint used by S3 clients to call this server +(*.s3.local.lakefs.io always resolves to 127.0.0.1, useful for +local development, if using virtual-host addressing.
  • +
  • gateways.s3.region (string : "us-east-1") - AWS region we’re pretending to be in, it should match the region configuration used in AWS SDK clients
  • +
  • gateways.s3.fallback_url (string) - If specified, requests with a non-existing repository will be forwarded to this URL. This can be useful for using lakeFS side-by-side with S3, with the URL pointing at an S3Proxy instance.
  • +
  • gateways.s3.verify_unsupported (bool : true) - The S3 gateway errors on unsupported requests, but when disabled, defers to target-based handlers.
  • +
+

+ + + tls + + +

+ + +
    +
  • tls.enabled (bool :false) - Enable TLS listening. The listen_address will be used to serve HTTPS requests. (mainly for local development)
  • +
  • tls.cert_file (string : ) - Server certificate file path used while serve HTTPS (.cert or .crt file - signed certificates).
  • +
  • tls.key_file (string : ) - Server secret key file path used whie serve HTTPS (.key file - private key).
  • +
+

+ + + stats + + +

+ + +
    +
  • stats.enabled (bool : true) - Whether to periodically collect anonymous usage statistics
  • +
  • stats.flush_interval (duration : 30s) - Interval used to post anonymous statistics collected
  • +
  • stats.flush_size (int : 100) - A size (in records) of anonymous statistics collected in which we post
  • +
+

+ + + installation + + +

+ + +
    +
  • installation.user_name (string : ) - When specified, an initial admin user will be created when the server is first run. Works only when database.type is set to local. Requires installation.access_key_id and installation.secret_access_key.
  • +
  • installation.access_key_id (string : ) - Admin’s initial access key id (used once in the initial setup process)
  • +
  • installation.secret_access_key (string : ) - Admin’s initial secret access key (used once in the initial setup process)
  • +
  • installation.allow_inter_region_storage (bool : true) - Allow storage in a different region than the one the server is running in.
  • +
+

+ + + usage_report + + +

+ + +
    +
  • usage_report.enabled (bool : true) - Store API and Gateway usage reports into key-value store.
  • +
  • usage_report.flush_interval (duration : 5m) - Sets interval for flushing in-memory usage data to key-value store.
  • +
+

+ + + ui + + +

+ + +
    +
  • ui.enabled (bool: true) - Whether to serve the embedded UI from the binary
  • +
+

+ + + security + + +

+ + +
    +
  • security.audit_check_interval (duration : 24h) - Duration in which we check for security audit.
  • +
+

+ + + garbage collection + + +

+ + +
    +
  • ugc.prepare_max_file_size (int: 125829120) - Uncommitted garbage collection prepare request, limit the produced file maximum size
  • +
  • ugc.prepare_interval (duraction: 1m) - Uncommitted garbage collection prepare request, limit produce time to interval
  • +
+

+ + + Using Environment Variables + + +

+ + +

All the configuration variables can be set or overridden using environment variables. +To set an environment variable, prepend LAKEFS_ to its name, convert it to upper case, and replace . with _:

+ +

For example, logging.format becomes LAKEFS_LOGGING_FORMAT, blockstore.s3.region becomes LAKEFS_BLOCKSTORE_S3_REGION, etc.

+

+ + + Example Configurations + + +

+ +

+ + + Local Development with PostgreSQL database + + +

+ + +
---
+listen_address: "0.0.0.0:8000"
+
+database:
+  type: "postgres"
+  postgres:
+    connection_string: "postgres://localhost:5432/postgres?sslmode=disable"
+
+logging:
+  format: text
+  level: DEBUG
+  output: "-"
+
+auth:
+  encrypt:
+    secret_key: "10a718b3f285d89c36e9864494cdd1507f3bc85b342df24736ea81f9a1134bcc"
+
+blockstore:
+  type: local
+  local:
+    path: "~/lakefs/dev/data"
+
+gateways:
+  s3:
+    region: us-east-1
+
+

+ + + AWS Deployment with DynamoDB database + + +

+ + +
---
+logging:
+  format: json
+  level: WARN
+  output: "-"
+
+database:
+  type: "dynamodb"
+  dynamodb:
+    table_name: "kvstore"
+
+auth:
+  encrypt:
+    secret_key: "10a718b3f285d89c36e9864494cdd1507f3bc85b342df24736ea81f9a1134bcc"
+
+blockstore:
+  type: s3
+  s3:
+    region: us-east-1 # optional, fallback in case discover from bucket is not supported
+    credentials_file: /secrets/aws/credentials
+    profile: default
+
+
+

+ + + Google Storage + + +

+ + +
---
+logging:
+  format: json
+  level: WARN
+  output: "-"
+
+database:
+  type: "postgres"
+  postgres:
+    connection_string: "postgres://user:pass@lakefs.rds.amazonaws.com:5432/postgres"
+
+auth:
+  encrypt:
+    secret_key: "10a718b3f285d89c36e9864494cdd1507f3bc85b342df24736ea81f9a1134bcc"
+
+blockstore:
+  type: gs
+  gs:
+    credentials_file: /secrets/lakefs-service-account.json
+
+
+

+ + + MinIO + + +

+ + +
---
+logging:
+  format: json
+  level: WARN
+  output: "-"
+
+database:
+  type: "postgres"
+  postgres:
+    connection_string: "postgres://user:pass@lakefs.rds.amazonaws.com:5432/postgres"
+
+auth:
+  encrypt:
+    secret_key: "10a718b3f285d89c36e9864494cdd1507f3bc85b342df24736ea81f9a1134bcc"
+
+blockstore:
+  type: s3
+  s3:
+    force_path_style: true
+    endpoint: http://localhost:9000
+    discover_bucket_region: false
+    credentials:
+      access_key_id: minioadmin
+      secret_access_key: minioadmin
+
+
+

+ + + Azure blob storage + + +

+ + +
---
+logging:
+  format: json
+  level: WARN
+  output: "-"
+
+database:
+  type: "cosmosdb"
+  cosmosdb:
+    key: "ExampleReadWriteKeyMD7nkPOWgV7d4BUjzLw=="
+    endpoint: "https://lakefs-account.documents.azure.com:443/"
+    database: "lakefs-db"
+    container: "lakefs-container"
+
+auth:
+  encrypt:
+    secret_key: "10a718b3f285d89c36e9864494cdd1507f3bc85b342df24736ea81f9a1134bcc"
+
+blockstore:
+  type: azure
+  azure:
+    storage_account: exampleStorageAcount
+    storage_access_key: ExampleAcessKeyMD7nkPOWgV7d4BUjzLw==
+
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/export.html b/v1.46/reference/export.html new file mode 100644 index 000000000..8f8def37e --- /dev/null +++ b/v1.46/reference/export.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/external-principals-aws.html b/v1.46/reference/external-principals-aws.html new file mode 100644 index 000000000..082f21417 --- /dev/null +++ b/v1.46/reference/external-principals-aws.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/garbage-collection.html b/v1.46/reference/garbage-collection.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/reference/garbage-collection.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/glossary.html b/v1.46/reference/glossary.html new file mode 100644 index 000000000..79940fc4e --- /dev/null +++ b/v1.46/reference/glossary.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/hooks.html b/v1.46/reference/hooks.html new file mode 100644 index 000000000..2ee933063 --- /dev/null +++ b/v1.46/reference/hooks.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/index.html b/v1.46/reference/index.html new file mode 100644 index 000000000..5755241d0 --- /dev/null +++ b/v1.46/reference/index.html @@ -0,0 +1,776 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Reference | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS Reference + + +

+ + +

lakeFS Docs

+

+ + + API + + +

+ + + +

+ + + Components + + +

+ + + +

+ + + Clients + + +

+ + + +

+ + + Security + + +

+ + + +

+ + + Other Reference Documentation + + +

+ + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/merge.html b/v1.46/reference/merge.html new file mode 100644 index 000000000..9a856dde2 --- /dev/null +++ b/v1.46/reference/merge.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/monitor.html b/v1.46/reference/monitor.html new file mode 100644 index 000000000..530262efd --- /dev/null +++ b/v1.46/reference/monitor.html @@ -0,0 +1,935 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Monitoring using Prometheus | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Monitoring using Prometheus + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Example prometheus.yml
  2. +
  3. Metrics exposed by lakeFS
  4. +
  5. Example queries
  6. +
+ +
+

+ + + Example prometheus.yml + + +

+ + +

lakeFS exposes metrics through the same port used by the lakeFS service, using the standard /metrics path. +An example prometheus.yml could look like this:

+ +
scrape_configs:
+- job_name: lakeFS
+  scrape_interval: 10s
+  metrics_path: /metrics
+  static_configs:
+  - targets:
+    - lakefs.example.com:8000
+
+

+ + + Metrics exposed by lakeFS + + +

+ + +

By default, Prometheus exports metrics with OS process information like memory and CPU. +It also includes Go-specific metrics such as details about GC and a number of goroutines. +You can learn about these default metrics in this post.

+ +

In addition, lakeFS exposes the following metrics to help monitor your deployment:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Name in PrometheusDescriptionLabels
api_requests_totallakeFS API requests (counter)code: http status
method: http method
api_request_duration_secondsDurations of lakeFS API requests (histogram)
operation: name of API operation
code: http status
gateway_request_duration_secondslakeFS S3-compatible endpoint request (histogram)
operation: name of gateway operation
code: http status
s3_operation_duration_secondsOutgoing S3 operations (histogram)
operation: operation name
error: “true” if error, “false” otherwise
gs_operation_duration_secondsOutgoing Google Storage operations (histogram)
operation: operation name
error: “true” if error, “false” otherwise
azure_operation_duration_secondsOutgoing Azure storage operations (histogram)
operation: operation name
error: “true” if error, “false” otherwise
kv_request_duration_secondsDurations of KV requests(histogram)
operation: name of KV operation
type: KV type(dynamodb, postgres, etc)
dynamo_request_duration_secondsTime spent doing DynamoDB requestsoperation: DynamoDB operation name
dynamo_consumed_capacity_totalThe capacity units consumed by operationoperation: DynamoDB operation name
dynamo_failures_totalThe total number of errors while working for kv storeoperation: DynamoDB operation name
pgxpool_acquire_countPostgreSQL cumulative count of successful acquires from the pooldb_name default to the kv table name (kv)
pgxpool_acquire_duration_nsPostgreSQL total duration of all successful acquires from the pool in nanosecondsdb_name default to the kv table name (kv)
pgxpool_acquired_connsPostgreSQL number of currently acquired connections in the pooldb_name default to the kv table name (kv)
pgxpool_canceled_acquire_countPostgreSQL cumulative count of acquires from the pool that were canceled by a contextdb_name default to the kv table name (kv)
pgxpool_constructing_connsPostgreSQL number of conns with construction in progress in the pooldb_name default to the kv table name (kv)
pgxpool_empty_acquirePostgreSQL cumulative count of successful acquires from the pool that waited for a resource to be released or constructed because the pool was emptydb_name default to the kv table name (kv)
pgxpool_idle_connsPostgreSQL number of currently idle conns in the pooldb_name default to the kv table name (kv)
pgxpool_max_connsPostgreSQL maximum size of the pooldb_name default to the kv table name (kv)
pgxpool_total_connsPostgreSQL total number of resources currently in the pooldb_name default to the kv table name (kv)
+

+ + + Example queries + + +

+ + +

Note: when using Prometheus functions like rate +or increase, results are extrapolated and may not be exact.

+

+ + + 99th percentile of API request latencies + + +

+ + +
sum by (operation)(histogram_quantile(0.99, rate(api_request_duration_seconds_bucket[1m])))
+
+

+ + + 50th percentile of S3-compatible API latencies + + +

+ + +
sum by (operation)(histogram_quantile(0.5, rate(gateway_request_duration_seconds_bucket[1m])))
+
+

+ + + Number of errors in outgoing S3 requests + + +

+ + +
sum by (operation) (increase(s3_operation_duration_seconds_count{error="true"}[1m]))
+
+

+ + + Number of open connections to the database + + +

+ + +
go_sql_stats_connections_open
+
+

+ + + Example Grafana dashboard + + +

+ + +

Grafana dashboard example

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/mount-csi-driver.html b/v1.46/reference/mount-csi-driver.html new file mode 100644 index 000000000..856709928 --- /dev/null +++ b/v1.46/reference/mount-csi-driver.html @@ -0,0 +1,1347 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Mount CSI Driver | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Mount CSI Driver (Everest on Kubernetes) + + +

+ + +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +

The lakeFS CSI (Container Storage Interface) Driver is an extension for Kubernetes that enables seamless access to data within a lakeFS repository, allowing Pods to interact with lakeFS data as if it were part of the local filesystem. This driver builds on the functionality of Everest, which provides a read-only view of lakeFS data by virtually mounting a repository.

+ +

Note +⚠️ The CSI Driver is in private preview. Please contact us to get access.

+ + +

+ + + How mount is executed on a Host + + +

+ + +
    +
  • While the csi driver is responsible for mounting and unmounting the volume on the host, it does not need permissions to execute the mount and umount commands directly.
  • +
  • The everest commands are executed by systemd service on the Host itself (i.e everest mount...).
  • +
  • The csi driver communicates with the systemd service via a unix socket to execute the mount and umount commands.
  • +
+

+ + + Status and Limitations + + +

+ + +
    +
  • Tested OS: BottleRocket-OS, Amazon Linux 2 and RHEL 8.
  • +
  • Minimal Kubernetes versions >=1.23.0.
  • +
  • Tested Cluster providers EKS, Openshift (Partially).
  • +
  • Static provisioning only explain below.
  • +
  • Setting Pods securityContext UID and GID (i.e runAsUser: 1000, runAsGroup: 2000) is very neuanced in nature and does not have wide coverage currently, not supported but might work.
  • +
  • Pod only supports access mode ReadOnlyMany
  • +
+ +

Static Provisioning only (Relevant for pods)

+ +

When requesting a mount from the CSI driver, the driver will create a PersistentVolume (PV) and PersistentVolumeClaim (PVC) for the Pod. +The driver only supports Static Provisioning as of today, and you need an existing lakeFS repository to use.

+ +

To use Static Provisioning, you should set storageClassName field of your PersistentVolume (PV) and PersistentVolumeClaim (PVC) to "" (empty string). Also, in order to make sure no other PVCs can claim your PV, you should define a one-to-one mapping using claimRef.

+

+ + + Requirements + + +

+ + +
    +
  1. For enterprise installations: lakeFS Version 1.25.0 or higher.
  2. +
  3. You have a Kubernetes cluster with version >=1.23.0 and Helm installed.
  4. +
  5. lakeFS Server that can be access from pods in the cluster.
  6. +
  7. Access to download treeverse/everest-lakefs-csi-driver from Docker Hub. Contact us to gain access to lakeFS Enterprise features.
  8. +
+

+ + + Deploy the CSI Driver + + +

+ + +

The CSI Driver is deployed to K8S cluster using a dedicated Helm chart everest-lakefs-csi-driver.

+

+ + + 1. Update your helm with the chart: + + +

+ + +

Add lakeFS Helm repository if not already added:

+ +
helm repo add lakefs https://charts.lakefs.io
+
+ +

Fetch the chart from lakeFS repository:

+ +
helm repo update lakefs
+
+ +

Verify the chart is available and updated:

+ +
helm show chart lakefs/everest-lakefs-csi-driver 
+
+ +

List all available chart versions:

+ +
helm search repo lakefs/everest-lakefs-csi-driver -l
+
+

+ + + 2. Configure the values for the CSI Driver in a values.yaml file + + +

+ + +

Helm Chart default values:

+ +
helm show values lakefs/everest-lakefs-csi-driver --version <version>
+
+ +

CSI driver config:

+ +

All the driver CLI flags can be configured via environment variables (prefiedx CSI_DRIVER_) and can be passed to the driver.

+ +

Example values.yaml file - Minimal required arguments not commented:

+ +
# image:  
+#   repository: treeverse/everest-lakefs-csi-driver
+# # Optional CSI Driver override version (default .Chart.AppVersion)
+#   tag: 1.2.3
+
+# Same as fluffy https://github.com/treeverse/fluffy?tab=readme-ov-file#1-dockerhub-token-for-fluffy
+imagePullSecret:
+  token: <dockerhub-token>
+  username: <dockerhub-user>
+
+# Credentials that will be used by everest as a default to access lakeFS mount paths
+lakeFSAccessSecret:
+  keyId: <lakefs-key-id>
+  accessKey: <lakefs-access-key>
+  endpoint: <lakefs-endpoint>
+
+node:
+  # verbosity level of the driver (normal values are 0-4, 5 would be most verbose)
+  logLevel: 4
+  # Only set if having issues with running or installing the everest binary 
+  # Path directory where the everest binary accessed by the underlying K8S Nodes (${everestInstallPath}/everest)
+  # The binary will copied from the CSI pod into that location by the init container job in the node.yaml
+  # This path will be a host path on the K8S Nodes
+  # depending on the underlying OS and the SELinux policy the binary will be executed by systemd on the Host.
+  # Known issue when using Bottlerocket OS https://github.com/bottlerocket-os/bottlerocket/pull/3779 
+  # everestInstallPath: /opt/everest-mount/bin/  # should end with "/"
+
+# Additional environment variables that will be passed to the driver can be used to configure the csi driver
+# extraEnvVars:
+#   - name: CSI_DRIVER_MOUNT_TIMEOUT
+#     value: "30s"
+#   - name: CSI_DRIVER_EVEREST_DEFAULT_CACHE_SIZE
+#     value: "10000000000"
+#   - name: VALUE_FROM_SECRET
+#     valueFrom:
+#       secretKeyRef:
+#         name: secret_name
+#         key: secret_key
+
+

+ + + 3. Install the Chart to K8S cluster + + +

+ + +

Install the chart with the values file:

+ +
helm install -f values.yaml lakefs lakefs/everest-lakefs-csi-driver --version <version>
+
+

+ + + Use in Pods + + +

+ + +

Once the CSI Driver is installed, we can start using it similarly to how all PersistentVolume (PV) and PersistentVolumeClaim (PVC) are used in Kubernetes.

+ +

The only required argument to set is lakeFSMountUri in the PV (See examples below).

+

+ + + Mount Options + + +

+ + +

Most of the options are optional and can be omitted, but each mount request can be configured with everest mount cli options, they are passed as mountOptions in the PVC spec.

+

+ + + Examples + + +

+ + +

The examples demonstrates different mount sncenarios with the CSI Driver. +All of them are essentially running ls <mount-dir> and tail -f /dev/null in a centos container. +If the mount succeeded you will see the contents of your mount directory.

+ +
    +
  1. Set lakeFSMountUri (i.e lakefs://<repo>/<repo>/[prefix/]) to the lakeFS mount URI you want to mount.
  2. +
  3. Run kubectl apply -f values.yaml
  4. +
  5. View the example pod logs to see the mount output kubectl logs -f <pod-name>
  6. +
+ +
+ + +
+ +

Configure lakeFSMountUri to the target URI.

+ +
apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: everest-pv
+spec:
+  capacity:
+    storage: 100Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  # everest mount options goes under mountOptions and forwarded to the everest mount command 
+  # mountOptions:
+    # set cache size in bytes 
+    # - cache-size 1000000000
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume
+    volumeAttributes:
+      # mount target, replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_MOUNT_URI>
+
+---
+apiVersion: v1
+kind: PersistentVolumeClaim
+metadata:
+  name: everest-claim
+spec:
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  storageClassName: "" # required for static provisioning
+  resources:
+    requests:
+      storage: 5Gi # ignored, required
+  volumeName: everest-pv
+---
+apiVersion: v1
+kind: Pod
+metadata:
+  name: everest-app
+spec:
+  containers:
+    - name: app
+      image: centos
+      command: ["/bin/sh"]
+      args: ["-c", "ls /data/; tail -f /dev/null"]
+      volumeMounts:
+        - name: persistent-storage-isan
+          mountPath: /data
+  volumes:
+    - name: persistent-storage-isan
+      persistentVolumeClaim:
+        claimName: everest-claim
+
+
+ +
+
+ +

Configure lakeFSMountUri to the target URI. +In this example a deployment is created with 3 replicas, all sharing a single PersistentVolume and PVC +Behind the scenes each pod get’s their own mount, even if on the same k8s node, each pod will get their own mount directory. +Unlike in StatefulSet, this can scale-up-down with no additional interference and deleted easily the same way.

+ +
apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: multiple-pods-one-pv
+spec:
+  capacity:
+    storage: 1200Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  # everest mount options goes under mountOptions and forwarded to the everest mount command 
+  # mountOptions:
+  #   - cache-size 1000000555
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume
+    volumeAttributes:
+      # mount target, replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_MOUNT_URI>
+---
+apiVersion: v1
+kind: PersistentVolumeClaim
+metadata:
+  name: multiple-pods-one-claim
+spec:
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  storageClassName: "" # required for static provisioning
+  resources:
+    requests:
+      storage: 1200Gi # ignored, required
+  volumeName: multiple-pods-one-pv
+---
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+  name: multiple-pods-one-pv-app
+  labels:
+    app: multiple-pods-one-pv-app
+spec:
+  replicas: 3
+  selector:
+    matchLabels:
+      app: multiple-pods-one-pv-app
+  template:
+    metadata:
+      labels:
+        app: multiple-pods-one-pv-app
+    spec:
+      containers:
+      - name: multiple-pods-one-pv-app
+        image: centos
+        command: ["/bin/sh"]
+        args: ["-c", "ls /data/; tail -f /dev/null"]
+        volumeMounts:
+        - name: persistent-storage
+          mountPath: /data
+        ports:
+        - containerPort: 80
+      volumes:
+      - name: persistent-storage
+        persistentVolumeClaim:
+          claimName: multiple-pods-one-claim
+
+
+
+
+ +

Deploy a pod with two mounts to different mount points. +Configure lakeFSMountUri for each PersistentVolume.

+ +
apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: multiple-mounts-one-pod-pv
+spec:
+  capacity:
+    storage: 1200Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  mountOptions:
+    - cache-size 1000000111
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume # must be unique
+    volumeAttributes:
+      # mount target local-lakefs dir, replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_URI_1>
+---
+apiVersion: v1
+kind: PersistentVolumeClaim
+metadata:
+  name: multple-mounts-one-pod-claim
+spec:
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  storageClassName: "" # required for static provisioning
+  resources:
+    requests:
+      storage: 1200Gi # ignored, required
+  volumeName: multiple-mounts-one-pod-pv
+---
+apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: multiple-mounts-one-pod-pv-2
+spec:
+  capacity:
+    storage: 1200Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # ReadOnlyMany
+  mountOptions:
+    - cache-size 1000000555
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume-2 # must be unique
+    volumeAttributes:
+      # mount target images dir, replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_URI_2>
+---
+apiVersion: v1
+kind: PersistentVolumeClaim
+metadata:
+  name: multple-mounts-one-pod-claim-2
+spec:
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  storageClassName: "" # required for static provisioning
+  resources:
+    requests:
+      storage: 1200Gi # ignored, required
+  volumeName: multiple-mounts-one-pod-pv-2
+---
+apiVersion: v1
+kind: Pod
+metadata:
+  name: everest-multi-mounts-one-pod
+spec:
+  containers:
+    - name: app
+      image: centos
+      command: ["/bin/sh"]
+      args: ["-c", "ls /data/; ls /data2/; tail -f /dev/null"]
+      volumeMounts:
+        - name: persistent-storage
+          mountPath: /data
+        - name: persistent-storage-2
+          mountPath: /data2
+  volumes:
+    - name: persistent-storage
+      persistentVolumeClaim:
+        claimName: multple-mounts-one-pod-claim
+    - name: persistent-storage-2
+      persistentVolumeClaim:
+        claimName: multple-mounts-one-pod-claim-2
+
+
+ +
+
+ +

Configure lakeFSMountUri to the target URI. +Because of the neuances described below, if not required it is best to avoid using a StatefulSet.

+ +

Deletion:

+ +

It’s intended behavior for StatefulSet in K8S that the PVC is not deleted automatically when the pod is deleted since the StatefulSet controller does not manage the PVC. +To completley delete use k delete with –force flag or first delete the PVC: ‘kubectl delete pvc -l app=sts-app-simple-everest’

+ +

Scale Down:

+ +

replicas: 0 can be set to scale down the StatefulSet and bring back up with replicas: 1.

+ +

Replicas > 1:

+ +

not supported in this example, since the driver only supports static provisoning. +to use Statefulset with replica > 1 we need to add PersistentVolume(s) manually.

+ +
apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: sts-simple-mount
+  labels:
+    app: sts-app-simple-everest
+spec:
+  capacity:
+    storage: 100Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  mountOptions:
+    # override default cache size for the mount (in bytes)
+    - cache-size 1000000555
+    - log-level debug
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume
+    volumeAttributes:
+      # mount target, replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_MOUNT_URI>
+---
+apiVersion: apps/v1
+kind: StatefulSet
+metadata:
+  name: sts-app-simple-everest
+spec:
+  replicas: 1
+  selector:
+    matchLabels:
+      app: sts-app-simple-everest
+  template:
+    metadata:
+      labels:
+        app: sts-app-simple-everest
+    spec:
+      containers:
+        - name: app
+          image: centos
+          command: ["/bin/sh"]
+          args: ["-c", "ls /data/; tail -f /dev/null"]
+          volumeMounts:
+            - name: sts-simple-mount
+              mountPath: /data
+  volumeClaimTemplates:
+  - metadata:
+      name: sts-simple-mount
+    spec:
+      selector:
+        matchLabels:
+          app: sts-app-simple-everest
+      storageClassName: "" # required for static provisioning
+      accessModes: [ "ReadOnlyMany" ]
+      resources:
+        requests:
+          storage: 5Gi # ignored, required
+
+
+
+ +
+ +

This demonstrates common flags and uncommon flags that can be used for a mount. +In general, the flags are set in mountOptions and are passed to the everest mount command.

+ +
apiVersion: v1
+kind: PersistentVolume
+metadata:
+  name: options-demo-pv
+spec:
+  capacity:
+    storage: 100Gi # ignored, required
+  accessModes:
+    - ReadOnlyMany # supported options: ReadOnlyMany
+  # everest mount options goes under mountOptions and forwarded to the everest mount command 
+  mountOptions:
+    # set cache size in bytes 
+    - cache-size 10000000
+    # set log level to debug when inspecting mount logs (very noisy!)
+    - log-level trace
+    # WARN: lakeFS credentials / endpoint should be managed securely by the CSI-driver, this is an advanced flag use-case
+    # override default lakeFS credentials (for use-cases where the default csi-driver credentials are not sufficient)
+    - lakectl-access-key-id <LAKEFS_ACCESS_KEY_ID>
+    - lakectl-secret-access-key <LAKEFS_SECRET_ACCESS_KEY>
+    - lakectl-server-url <LAKEFS_ENDPOINT>
+    # WARN: an advanced flag and rarelly needed if at all, performs mount directly using fuser relying on it to exist on the host server without using FUSE syscalls
+    # be default fuse-direct-mount is true
+    # - fuse-direct-mount false
+    # - mount-gid 2000
+    # - mount-uid 1000
+    # - presign false
+    # - log-enable-syslog false
+
+  csi:
+    driver: csi.everest.lakefs.io # required
+    volumeHandle: everest-csi-driver-volume
+    volumeAttributes:
+      # mount target, staging org (non default credentials on csi), replace with your lakeFS mount URI
+      lakeFSMountUri: <LAKEFS_MOUNT_URI>
+
+# REST OF THE RESOURCES
+# ... 
+
+
+ +
+

+ + + Troubleshooting + + +

+ + +
    +
  • Use kubectl and check the CSI driver pod and failed Pod for logs and events.
  • +
  • If a specific mount request failed, specifically inspect csi-node that the failed mount pod was deployed on.
  • +
  • Check the events and status of the PVC and PV of the failing pod kubectl get pv && kubectl get pvc
  • +
+ +

Advanced: SSH into the underlying K8S node:

+ +

Find the failed mount service systemctl list-units --type=service:

+ +
everest-lakefs-mount-0.0.8-everest-123.service loaded active running CSI driver FUSE daemon
+
+

Get systemd service status:

+ +
# service name example: everest-lakefs-mount-0.0.8-everest-123.service
+systemctl status <service>
+
+# output contains many things including the exec command to run, example:
+# ExecStart=/opt/bin/everest mount lakefs://test-mount/main/local-lakefs/ /var/lib/kubelet/pods/123/volumes/kubernetes.io~csi/everest-pv/mount --log-level=trace --cache-dir=/var/lib/kubelet/pods/123/volumes/kubernetes.io~csi/everest-pv/.everest --lakectl-config=/opt/mountpoint-s3-csi/bin/lakectl.yaml
+
+ +

See systemd logs of a service:

+ +
journalctl -f -u <service>
+
+# example:
+journalctl -f -u everest-lakefs-mount-0.0.8-everest-123.service
+
+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/mount.html b/v1.46/reference/mount.html new file mode 100644 index 000000000..e0b52c59c --- /dev/null +++ b/v1.46/reference/mount.html @@ -0,0 +1,1099 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Mount | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Mount (Everest) + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +

Everest is a complementary binary to lakeFS that allows users to virtually mount a remote lakeFS repository onto a local directory. +Once mounted, users can access the data as if it resides on their local filesystem, using any tool, library, or framework that reads from a local filesystem. +This functionality is currently in limited support and is a Read-Only file system and is pointing to a commit in lakeFS.

+ +

Note +⚠️ No installation is required. Please contact us to get access to the Everest binary.

+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Use Cases
  2. +
  3. Requirements
  4. +
  5. Authentication with lakeFS
  6. +
  7. Command Line Interface
  8. +
  9. Examples
  10. +
  11. FAQs
  12. +
+ +
+ + +

+ + + Use Cases + + +

+ + +
    +
  • Simplified Data Loading: With lakeFS Mount, there’s no need to write custom data loaders or use special SDKs. You can use your existing tools to read files directly from the filesystem.
  • +
  • Handle Large-scale Data Without changing Work Habits: Seamlessly scale from a few local files to millions without changing your tools or workflow. Use the same code from early experimentation all the way to production.
  • +
  • Enhanced Data Loading Efficiency: lakeFS Mount supports billions of files and offers fast data fetching, making it ideal for optimizing GPU utilization and other performance-sensitive tasks.
  • +
+

+ + + Requirements + + +

+ + +
    +
  • For enterprise installations: lakeFS Version 1.25.0 or higher.
  • +
+

+ + + OS and Protocol Support + + +

+ + +

Currently the implemented protocols are nfs and fuse.

+
    +
  • NFS V3 (Network File System) is supported on macOS.
  • +
  • FUSE is supported on Linux (no root required).
  • +
+

+ + + Authentication with lakeFS + + +

+ + +

The authentication with the target lakeFS server is equal to lakectl CLI. +Searching for lakeFS credentials and server endpoint in the following order:

+
    +
  • Command line flags --lakectl-access-key-id, --lakectl-secret-access-key and --lakectl-server-url
  • +
  • LAKECTL_* Environment variables
  • +
  • ~/.lakectl.yaml Configuration file or via --lakectl-config flag
  • +
+

+ + + Command Line Interface + + +

+ +

+ + + Mount Command + + +

+ + +

The mount command is used to mount a lakeFS repository to a local directory, it does it in 2 steps:

+
    +
  • Step 1: Starting a server that listens on a local address and serves the data from the remote lakeFS repository.
  • +
  • Step 2: Running the required mount command on the OS level to connect the server to the local directory.
  • +
+

+ + + Tips: + + +

+ + +
    +
  • Since the server runs in the background set --log-output /some/file to view the logs in a file.
  • +
  • Cache: Everest uses a local cache to store the data and metadata of the lakeFS repository. The optimal cache size is the size of the data you are going to read.
  • +
  • Reusing Cache: between restarts of the same mount endpoint, set --cache-dir to make sure the cache is reused.
  • +
  • Mounted data consistency: When providing lakeFS URI mount endpoint lakefs://<repo>/<ref>/<path> the <ref> should be a specific commit ID. If a branch/tag is provided, Everest will use the HEAD commit instead.
  • +
+

+ + + Usage + + +

+ +
everest mount <lakefs_uri> <mount_directory>
+
+Flags
+--presign: Use presign for downloading.
+--cache-dir: Directory to cache read files in.
+--cache-size: Size of the local cache in bytes.
+--cache-create-provided-dir: If cache-dir is explicitly provided and does not exist, create it.
+--listen: Address to listen on.
+--no-spawn: Do not spawn a new server, assume one is already running.
+--protocol: Protocol to use (default: nfs).
+--log-level: Set logging level.
+--log-format: Set logging output format.
+--log-output: Set logging output(s).
+
+

+ + + Umount Command + + +

+ +

The umount command is used to unmount a currently mounted lakeFS repository.

+ +
everest umount <data_directory>
+
+

+ + + mount-server Command (Advanced) + + +

+ + +

Note +⚠️ The mount-server command is for advanced use cases and will only spin the server without calling OS mount command.

+ +

The mount-server command starts a mount server manually. Generally, users would use the mount command which handles server operations automatically.

+ +
everest mount-server <remote_mount_uri>
+Flags
+--cache-dir: Directory to cache read files and metadata.
+--cache-create-provided-dir: Create the cache directory if it does not exist.
+--listen: Address to listen on.
+--protocol: Protocol to use (nfs | webdav).
+--callback-addr: Callback address to report back to.
+--log-level: Set logging level.
+--log-format: Set logging output format.
+--log-output: Set logging output(s).
+--cache-size: Size of the local cache in bytes.
+--parallelism: Number of parallel downloads for metadata.
+--presign: Use presign for downloading.
+
+

+ + + Examples + + +

+ + +

Note +⚠️ For simplicity, the examples show main as the ref, Everest will always mount a specific commit ID, given a ref it will use the HEAD (e.g most recent commit).

+

+ + + Data Exploration + + +

+ +

Mount the lakeFS repository and explore data as if it’s on the local filesystem.

+ +
everest mount "lakefs://image-repo/main/datasets/pets/" "./pets"
+ls -l "./pets/dogs/"
+find ./pets -name "*.small.jpg"
+open -a Preview "./pets/dogs/golden_retrievers/cute.jpg"
+everest umount "./pets"
+
+
+

+ + + Working with Data Locally + + +

+ +

Mount the remote lakeFS server and use all familiar tools without changing the workflow.

+
everest mount "lakefs://image-repo/main/datasets/pets/" "./pets"
+pytorch_train.py --input "./pets"
+duckdb "SELECT * FROM read_parquet('pets/labels.parquet')"
+everest umount "./pets"
+
+

+ + + FAQs + + +

+ + + +

+ + + How do I get started with lakeFS Mount (Everest)? + + +

+ + +

lakeFS Mount is avaialble for lakeFS Cloud and lakeFS Enterprise customers. Once your setup is complete, contact us to access the lakeFS Mounts binary and follow the provided docs.

+ +
    +
  • Want to try lakeFS Cloud? Signup for a 30-day free trial.
  • +
  • Interested in lakeFS Enterprise? Contact sales for a 30-day free license.
  • +
+

+ + + Can I write to lakeFS using lakeFS Mount? + + +

+ + +

Currently, lakeFS Mount supports read-only file system operations. Write support is on our roadmap and will be added in the future.

+

+ + + What operating systems are supported by lakeFS Mount? + + +

+ + +

lakeFS Mount supports Linux and MacOS. Windows support is on the roadmap.

+

+ + + How can I control access to my data when using lakeFS Mount? + + +

+ + +

You can use lakeFS’s existing Role-Based Access Control mechanism, which includes repository and path-level policies. lakeFS Mount translates filesystem operations into lakeFS API operations and authorizes them based on these policies.

+

+ + + Does data pass through the lakeFS server when using lakeFS Mount? + + +

+ + +

lakeFS Mount leverages pre-signed URLs to read data directly from the underlying object store, meaning data doesn’t pass through the lakeFS server. By default, presign is enabled. To disable it, use:

+
everest mount <lakefs_uri> <mount_directory> --presign=false
+
+

+ + + What happens if a lakeFS branch is updated after I mount it? + + +

+ + +

lakeFS Mount points to the commit that was the HEAD commit of the branch at the time of mounting. This means the local directory reflects the branch state at the time of mounting and does not update with subsequent branch changes.

+

+ + + When are files downloaded to my local environment? + + +

+ +

lakeFS Mount uses a lazy prefetch strategy. Files are not downloaded at mount time or during operations that only inspect file metadata (e.g., ls). Files are downloaded only when commands that require file access (e.g., cat) are used.

+ + + +

When using lakeFS Mount, the volume of data accessed by the local machine influences the scale limitations more than the total size of the dataset under the mounted prefix. This is because lakeFS Mount uses a lazy downloading approach, meaning it only downloads the accessed files. lakeFS Mount listing capability is limited to performing efficiently for prefixes containing fewer than 8000 objects, but we are working to increase this limit.

+ + + +

Ensure your cache size is large enough to accommodate the volume of files being accessed.

+

+ + + How does lakeFS Mount integrate with a Git repository? + + +

+ + +

It is perfectly safe to mount a lakeFS path within a Git repository. +lakeFS Mount prevents git from adding mounted objects to the git repository (i.e when running git add -A) by adding a virtual .gitignore file to the mounted directory.

+ +

The .gitignore file will also instruct Git to ignore all files except .everest/source and in its absence, it will try to find a .everest/source file in the destination folder, and read the lakeFS URI from there. +Since .everest/source is in source control, it will mount the same lakeFS commit every time!

+

+ + + I’m already using lakectl local for working with lakeFS data locally, why should I use lakeFS Mount? + + +

+ + +

While both lakectl local and lakeFS Mount enable working with lakeFS data locally, they serve different purposes:

+
+ + + Use lakectl local + + +
+ + + +
+ + + Use lakeFS Mount + + +
+ + +

For read-only local data access. lakeFS Mount offers several benefits over lakectl local:

+
    +
  • Optimized selective data access: The lazy prefetch strategy saves storage space and reduces latency by only fetching the required data.
  • +
  • Reduced initial latency: Start working on your data immediately without waiting for downloads.
  • +
+ +

Note +Note: Write support for lakeFS Mount is on our roadmap.

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/object-model.html b/v1.46/reference/object-model.html new file mode 100644 index 000000000..8e9879ba3 --- /dev/null +++ b/v1.46/reference/object-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/offboarding.html b/v1.46/reference/offboarding.html new file mode 100644 index 000000000..ad6a95fec --- /dev/null +++ b/v1.46/reference/offboarding.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/presigned-url.html b/v1.46/reference/presigned-url.html new file mode 100644 index 000000000..c58aa34d7 --- /dev/null +++ b/v1.46/reference/presigned-url.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/protected_branches.html b/v1.46/reference/protected_branches.html new file mode 100644 index 000000000..1c30c5849 --- /dev/null +++ b/v1.46/reference/protected_branches.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/pull_requests.html b/v1.46/reference/pull_requests.html new file mode 100644 index 000000000..480d2c990 --- /dev/null +++ b/v1.46/reference/pull_requests.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/rbac.html b/v1.46/reference/rbac.html new file mode 100644 index 000000000..3fdf4ca19 --- /dev/null +++ b/v1.46/reference/rbac.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/remote-authenticator.html b/v1.46/reference/remote-authenticator.html new file mode 100644 index 000000000..723069491 --- /dev/null +++ b/v1.46/reference/remote-authenticator.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/retention.html b/v1.46/reference/retention.html new file mode 100644 index 000000000..45b0557bc --- /dev/null +++ b/v1.46/reference/retention.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/reference/s3.html b/v1.46/reference/s3.html new file mode 100644 index 000000000..fee87778c --- /dev/null +++ b/v1.46/reference/s3.html @@ -0,0 +1,774 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +S3 Gateway API | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + S3-Supported API + + +

+ + +

The S3 Gateway emulates a subset of the API exposed by S3. +This subset includes all API endpoints relevant to data systems.

+ +

For more information, see architecture.

+ +

lakeFS supports the following API operations:

+ +
    +
  1. Identity and authorization +
      +
    1. SIGv2
    2. +
    3. SIGv4
    4. +
    +
  2. +
  3. Bucket operations: +
      +
    1. HEAD bucket
    2. +
    +
  4. +
  5. Object operations: +
      +
    1. DeleteObject
    2. +
    3. DeleteObjects
    4. +
    5. GetObject +
        +
      1. Support for caching headers, ETag
      2. +
      3. Support for range requests
      4. +
      5. No support for SSE
      6. +
      7. No support for SelectObject operations
      8. +
      +
    6. +
    7. HeadObject
    8. +
    9. PutObject +
        +
      1. Support multi-part uploads
      2. +
      3. No support for storage classes
      4. +
      5. No object level tagging
      6. +
      +
    10. +
    11. CopyObject
    12. +
    +
  6. +
  7. Object Listing: +
      +
    1. ListObjects
    2. +
    3. ListObjectsV2
    4. +
    5. Delimiter support (for "/" only)
    6. +
    +
  8. +
  9. Multipart Uploads: +
      +
    1. AbortMultipartUpload
    2. +
    3. CompleteMultipartUpload
    4. +
    5. CreateMultipartUpload
    6. +
    7. ListParts Currently supported only on AWS S3. Link to tracked issue
    8. +
    9. Upload Part
    10. +
    11. UploadPartCopy
    12. +
    +
  10. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/spark-client.html b/v1.46/reference/spark-client.html new file mode 100644 index 000000000..a21785164 --- /dev/null +++ b/v1.46/reference/spark-client.html @@ -0,0 +1,923 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Spark Client | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Spark Metadata Client + + +

+ + +

Utilize the power of Spark to interact with the metadata on lakeFS. Possible use cases include:

+ +
    +
  • Creating a DataFrame for listing the objects in a specific commit or branch.
  • +
  • Computing changes between two commits.
  • +
  • Exporting your data for consumption outside lakeFS.
  • +
  • Bulk operations on the underlying storage.
  • +
+

+ + + Getting Started + + +

+ + +

Please note that Spark 2 is no longer supported with the lakeFS metadata client.

+ +

The Spark metadata client is compiled for Spark 3.1.2 with Hadoop 3.2.1, but +can work for other Spark versions and higher Hadoop versions.

+ +
+ +
+

Start Spark Shell / PySpark with the --packages flag, for instance:

+ +
spark-shell --packages io.lakefs:lakefs-spark-client_2.12:0.14.1
+
+ +

Alternatively use the assembled jar (an “Überjar”) on S3, from +s3://treeverse-clients-us-east/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar +by passing its path to --jars. +The assembled jar is larger but shades several common libraries. Use it if Spark +complains about bad classes or missing methods.

+
+
+

Include this assembled jar (an “Überjar”) from S3, from +s3://treeverse-clients-us-east/lakefs-spark-client/0.14.1/lakefs-spark-client-assembly-0.14.1.jar.

+
+
+

+ + + Configuration + + +

+ + +
    +
  1. +

    To read metadata from lakeFS, the client should be configured with your lakeFS endpoint and credentials, using the following Hadoop configurations:

    + +
    + + + + + + + + + + + + + + + + + + + + +
    ConfigurationDescription
    spark.hadoop.lakefs.api.urllakeFS API endpoint, e.g: http://lakefs.example.com/api/v1
    spark.hadoop.lakefs.api.access_keyThe access key to use for fetching metadata from lakeFS
    spark.hadoop.lakefs.api.secret_keyCorresponding lakeFS secret key
    +
  2. +
  3. +

    The client will also directly interact with your storage using Hadoop FileSystem. +Therefore, your Spark session must be able to access the underlying storage of your lakeFS repository. +There are various ways to do this, +but for a non-production environment you can use the following Hadoop configurations:

    + +
    + + + + + + + + + + + + + + + + +
    ConfigurationDescription
    spark.hadoop.fs.s3a.access.keyAccess key to use for accessing underlying storage on S3
    spark.hadoop.fs.s3a.secret.keyCorresponding secret key to use with S3 access key
    +

    + + + Assuming role on S3 (Hadoop 3 only) + + +

    + + +

    The client includes support for assuming a separate role on S3 when +running on Hadoop 3. It uses the same configuration used by +S3AFileSystem to assume the role on S3A. Apache Hadoop AWS +documentation has details under “Working with IAM Assumed +Roles”. You will need to use the following Hadoop +configurations:

    + +
    + + + + + + + + + + + + + + + + +
    ConfigurationDescription
    fs.s3a.aws.credentials.providerSet to org.apache.hadoop.fs.s3a.auth.AssumedRoleCredentialProvider
    fs.s3a.assumed.role.arnSet to the ARN of the role to assume
    +
  4. +
+

+ + + Examples + + +

+ + +
    +
  1. +

    Get a DataFrame for listing all objects in a commit:

    + +
    import io.treeverse.clients.LakeFSContext
    +    
    +val commitID = "a1b2c3d4"
    +val df = LakeFSContext.newDF(spark, "example-repo", commitID)
    +df.show
    +/* output example:
    +   +------------+--------------------+--------------------+-------------------+----+
    +   |        key |             address|                etag|      last_modified|size|
    +   +------------+--------------------+--------------------+-------------------+----+
    +   |     file_1 |791457df80a0465a8...|7b90878a7c9be5a27...|2021-03-05 11:23:30|  36|
    +   |     file_2 |e15be8f6e2a74c329...|95bee987e9504e2c3...|2021-03-05 11:45:25|  36|
    +   |     file_3 |f6089c25029240578...|32e2f296cb3867d57...|2021-03-07 13:43:19|  36|
    +   |     file_4 |bef38ef97883445c8...|e920efe2bc220ffbb...|2021-03-07 13:43:11|  13|
    +   +------------+--------------------+--------------------+-------------------+----+
    + */
    +
    +
  2. +
  3. +

    Run SQL queries on your metadata:

    + +
    df.createOrReplaceTempView("files")
    +spark.sql("SELECT DATE(last_modified), COUNT(*) FROM files GROUP BY 1 ORDER BY 1")
    +/* output example:
    +   +----------+--------+
    +   |        dt|count(1)|
    +   +----------+--------+
    +   |2021-03-05|       2|
    +   |2021-03-07|       2|
    +   +----------+--------+
    + */
    +
    +
  4. +
  5. +

    Search by user metadata:

    + +
    import io.treeverse.clients.LakeFSContext
    +
    +val namespace = "s3://bucket/repo/path/"
    +val df = LakeFSContext.newDF(spark, namespace)
    +
    +val key = "SomeKey"
    +val searchedValue = "val3"
    +df.select("key", "user_metadata")
    +	.filter(_.getMap[String, String](1).toMap.get(s"X-Amz-Meta-${key}").getOrElse("") == searchedValue)
    +	.show()
    +/* output example:
    +   +---------+-----------------------------------------------------+
    +   |key      |user_metadata                                        |
    +   +---------+-----------------------------------------------------+
    +   |file1.txt|{X-Amz-Meta-SomeKey -> val3, X-Amz-Meta-Tag -> blue} |
    +   |file8.txt|{X-Amz-Meta-SomeKey -> val3, X-Amz-Meta-Tag -> green}|
    +   +---------+-----------------------------------------------------+
    + */
    +
    +
  6. +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/reference/upgrade.html b/v1.46/reference/upgrade.html new file mode 100644 index 000000000..63831114b --- /dev/null +++ b/v1.46/reference/upgrade.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/roadmap.html b/v1.46/roadmap.html new file mode 100644 index 000000000..41dce22e2 --- /dev/null +++ b/v1.46/roadmap.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/security/access-control-lists.html b/v1.46/security/access-control-lists.html new file mode 100644 index 000000000..9986eb150 --- /dev/null +++ b/v1.46/security/access-control-lists.html @@ -0,0 +1,822 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Access Control Lists (ACLs) -Deprecated- | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Access Control Lists (ACLs) + + +

+ + +
+

ACLs were removed from core lakeFS.

+ +

For a more robust authorization solution, please see Role-Based Access Control, available in lakeFS Cloud and lakeFS Enterprise.
+The following documentation is aimed for users with existing installations who wish to continue working with ACLs.

+
+ +
+

+ + + Table of contents + + +

+ + +
    +
  1. Basic Auth Functionality
  2. +
  3. ACLs
  4. +
+ +
+

+ + + Basic Auth Functionality + + +

+ + +

New lakeFS versions will provide basic auth functionality featuring a single Admin user with a single set of credentials. +Existing lakeFS installations that have a single user and a single set of credentials will migrate seamlessly to the new version.
+Installations that have more than one user / credentials will require to run a command and choose which set of user + credentials to migrate +(more details here)

+

+ + + ACLs + + +

+ + +

ACL server was moved out of core lakeFS and into a new package under contrib/auth/acl. +Though we decided to move ACLs out, we are committed to making sure existing users who still need the use of ACLs can continue using +this feature. +In order to do that, users will need to run the separate ACL server as part of their lakeFS deployment environment and configure lakeFS to work with it.

+

+ + + ACL server Configuration + + +

+ + +

Under the contrib/auth/acl you will be able to find an ACL server reference.

+ +
+

This implementation is a reference and is not fit for production use.

+ +

For a more robust authorization solution, please see Role-Based Access Control, available in lakeFS Cloud and lakeFS Enterprise.

+
+ +

The configuration of the ACL server is similar to lakeFS configuration, here’s an example of an .aclserver.yaml config file:

+
   ---
+   listen_address: "[ACL_SERVER_LISTEN_ADDRESS]"
+   database:
+     type: "postgres"
+     postgres:
+       connection_string: "[DATABASE_CONNECTION_STRING]"
+  
+   encrypt:
+     # This should be the same encryption key as in lakeFS
+     secret_key: "[ENCRYPTION_SECRET_KEY]"
+
+

It is possible to use environment variables to configure the server as in lakeFS. Use the ACLSERVER_ prefix to do so.
+For full configuration reference see: this

+

+ + + lakeFS Configuration + + +

+ + +

For the ACL server to work, configure the following values in lakeFS:
+auth.ui_config.rbac: simplified
+auth.api.endpoint: [ACL_SERVER_LISTEN_ADDRESS]

+

+ + + Migration of existing user + + +

+ + +

For installation with multiple users / credentials, upgrading to the new lakeFS version requires choosing which user + credentials will be used for the single user mode. +This is done via the lakefs superuser command. +For example, if you have a user with username <my-username> and credential key <my-access-key-id> use the following command to migrate that user:

+
lakefs superuser --user-name <my-username> --access-key-id <my-access-key-id>
+
+ +

After running the command you will be able to access the installation using the user’s access key id and its respective secret access key.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/authentication.html b/v1.46/security/authentication.html new file mode 100644 index 000000000..df9f89d8e --- /dev/null +++ b/v1.46/security/authentication.html @@ -0,0 +1,914 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Authentication | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Authentication + + +

+ + + +

+ + + Authentication + + +

+ +

+ + + User Authentication + + +

+ + +

lakeFS authenticates users from a built-in authentication database.

+

+ + + Built-in database + + +

+ + +

The built-in authentication database is always present and active. You can use the +Web UI at Administration / Users to create users. Users have an access key +AKIA... and an associated secret access key. These credentials are valid +for logging into the Web UI or authenticating programmatic requests to the API +Server or the S3 Gateway.

+

+ + + Remote Authenticator Service + + +

+ + +

lakeFS server supports external authentication, the feature can be configured by providing an HTTP endpoint to an external authentication service. This integration can be especially useful if you already have an existing authentication system in place, as it allows you to reuse that system instead of maintaining a new one. +To configure a Remote Authenticator see the configuration fields.

+

+ + + API Server Authentication + + +

+ + +

Authenticating against the API server is done using a key-pair, passed via Basic Access Authentication.

+ +

All HTTP requests must carry an Authorization header with the following structure:

+ +
Authorization: Basic <base64 encoded access_key_id:secret_access_key>
+
+ +

For example, assuming my access_key_id is my_access_key_id and my secret_access_key is my_secret_access_key, we’d send the following header with every request:

+ +
Authorization: Basic bXlfYWNjZXNzX2tleV9pZDpteV9hY2Nlc3Nfc2VjcmV0X2tleQ==
+
+

+ + + S3 Gateway Authentication + + +

+ + +

To provide API compatibility with Amazon S3, authentication with the S3 Gateway supports both SIGv2 and SIGv4. +Clients such as the AWS SDK that implement these authentication methods should work without modification.

+ +

See this example for authenticating with the AWS CLI.

+

+ + + OIDC support + + +

+ + +

Note +This feature is deprecated. For single sign-on with lakeFS, try lakeFS Cloud

+ +

OpenID Connect (OIDC) is a simple identity layer on top of the OAuth 2.0 protocol. +You can configure lakeFS to enable OIDC to manage your lakeFS users externally. +Essentially, once configured, this enables you the benefit of OpenID connect, such as a single sign-on (SSO), etc.

+

+ + + Configuring lakeFS server for OIDC + + +

+ + +

To support OIDC, add the following to your lakeFS configuration:

+ +
auth:
+  oidc:
+    enabled: true
+    client_id: example-client-id
+    client_secret: exampleSecretValue
+    callback_base_url: https://lakefs.example.com       # The scheme, domain (and port) of your lakeFS installation
+    url: https://my-account.oidc-provider-example.com
+    default_initial_groups: ["Developers"]
+    friendly_name_claim_name: name                      #  Optional: use the value from this claim as the user's display name
+    persist_friendly_name: true                         #  Optional: persist friendly name to KV store so it can be displayed in the user list
+
+ +

Your login page will now include a link to sign in using the +OIDC provider. When a user first logs in through the provider, a corresponding user is created in lakeFS.

+

+ + + Friendly Name Persistence + + +

+ + +

When the persist_friendly_name configuration property is set to true and friendly_name_claim_name is set to a valid claim name, which exists in the incoming id_token, the friendly name will be persisted to the KV store. This will allow users with access to the lakeFS administration section to see friendly names in the users list, when listing group members, and when adding/removing group members.
+The friendly name stored in KV is updated with each successful login, if the incoming value is different than the stored value. This means it will be kept up-to-date with changes to the user’s profile or if friendly_name_claim_name is re-configured.

+

+ + + Notes + + +

+ +
    +
  1. As always, you may choose to provide these configurations using environment variables.
  2. +
  3. You may already have other configuration values under the auth key, so make sure you combine them correctly.
  4. +
+

+ + + User permissions + + +

+ + +

Authorization is managed via lakeFS groups and policies.

+ +

By default, an externally managed user is assigned to the lakeFS groups configured in the default_initial_groups property above. +For a user to be assigned to other groups, add the initial_groups claim to their ID token claims. The claim should contain a +comma-separated list of group names.

+ +

Once the user has been created, you can manage their permissions from the Administration pages in the lakeFS UI or using lakectl.

+

+ + + Using a different claim name + + +

+ + +

To supply the initial groups using another claim from your ID token, you can use the auth.oidc.initial_groups_claim_name +lakeFS configuration. For example, to take the initial groups from the roles claim, add:

+ +
auth:
+  oidc:
+    # ... Other OIDC configurations
+    initial_groups_claim_name: roles
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/external-principals-aws.html b/v1.46/security/external-principals-aws.html new file mode 100644 index 000000000..8c9bb51a3 --- /dev/null +++ b/v1.46/security/external-principals-aws.html @@ -0,0 +1,939 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Login to lakeFS with AWS IAM Roles | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Authenticate to lakeFS with AWS IAM Roles + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

External principals API is available for lakeFS Enterprise. If you’re using the open-source version you can check the pluggable APIs.

+
+ + +

+ + + Overview + + +

+ + +

lakeFS supports authenticating users programmatically using AWS IAM roles instead of using static lakeFS access and secret keys. +The method enables you to bound IAM principal ARNs to lakeFS users. +A single lakeFS user may have many AWS’s principle ARNs attached to it. When a client is authenticating to a lakeFS server with an AWS’s session, the actions performed by the client are on behalf of the user attached to the ARN.

+

+ + + Using Session Names + + +

+ + +

The bound ARN can be attached to a single lakeFS user with or without SessionName, serving different users. +For example, consider the following mapping:

+ +
+ + + + + + + + + + + + + + + + +
Principal ARNlakeFS User
arn:aws:sts::123456:assumed-role/Devfoo
arn:aws:sts::123456:assumed-role/Dev/john@acme.comjohn
+ +

if the bound ARN were arn:aws:sts::123456:assumed-role/Dev/<SessionName> it would allow any principal assuming Dev role in AWS account 123456 to login to it. +If the SessionName is john@acme.com then lakeFS would return token for john user

+

+ + + How AWS authentication works + + +

+ + +

The AWS STS API includes a method, sts:GetCallerIdentity, which allows you to validate the identity of a client. The client signs a GetCallerIdentity query using the AWS Signature v4 algorithm and sends it to the lakeFS server.

+ +

The GetCallerIdentity query consists of four pieces of information: the request URL, the request body, the request headers and the request method. The AWS signature is computed over those fields. The lakeFS server reconstructs the query using this information and forwards it on to the AWS STS service. Depending on the response from the STS service, the server authenticates the client.

+ +

Notably, clients don’t need network-level access themselves to talk to the AWS STS API endpoint; they merely need access to the credentials to sign the request. However, it means that the lakeFS server does need network-level access to send requests to the STS endpoint.

+ +

Each signed AWS request includes the current timestamp to mitigate the risk of replay attacks. In addition, lakeFS allows you to require an additional header, X-LakeFS-Server-ID (added by default), to be present to mitigate against different types of replay attacks (such as a signed GetCallerIdentity request stolen from a dev lakeFS instance and used to authenticate to a prod lakeFS instance).

+ +

It’s also important to note that Amazon does NOT appear to include any sort of authorization around calls to GetCallerIdentity. For example, if you have an IAM policy on your credential that requires all access to be MFA authenticated, non-MFA authenticated credentials will still be able to authenticate to lakeFS using this method.

+

+ + + Server Configuration + + +

+ + +
+

Note: lakeFS Helm chart supports the configuration since version 1.2.11 - see usage values.yaml example.

+
+ +
    +
  • in lakeFS auth.authentication_api.external_principals_enabled must be set to true in the configuration file, other configuration (auth.authentication_api.*) can be found at configuration reference
  • +
+ +

For the full list of the Fluffy server configuration, see Fluffy Configuration under auth.external.aws_auth

+ +
+

By default, lakeFS clients will add the parameter X-LakeFS-Server-ID: <lakefs.ingress.domain> to the initial login request for STS.

+
+ +

Example configuration with required headers:

+ +

Configuration for lakefs.yaml:

+ +
auth:
+  authentication_api:
+    endpoint: http://<fluffy-sso>/api/v1
+    external_principals_enabled: true
+  api:
+    endpoint: http://<fluffy-rbac>/api/v1
+
+ +

Configuration for fluffy.yaml:

+ +
# fluffy address for lakefs auth.authentication_api.endpoint
+# used by lakeFS to log in and get the token
+listen_address: <fluffy-sso>
+auth:
+  # fluffy address for lakeFS auth.api.endpoint 
+  # used by lakeFS to manage the lifecycle attach/detach of the external principals
+  serve_listen_address: <fluffy-rbac>
+  external:
+    aws_auth:
+      enabled: true
+      # headers that must be present by the client when doing login request
+      required_headers:
+        # same host as the lakeFS server ingress
+        X-LakeFS-Server-ID: <lakefs.ingress.domain>
+
+

+ + + Administration of IAM Roles in lakeFS + + +

+ + +

Administration refers to the management of the IAM roles that are allowed to authenticate to lakeFS. +Operations such as attaching and detaching IAM roles to a user, listing the roles attached to a user, and listing the users attached to a role. +Currently, this is done through the lakeFS External Principals API and generated clients.

+ +

Example of attaching an IAM roles to a user:

+ +
import lakefs_sdk as lakefs  
+
+configuration = lakefs.Configuration(host = "...",username="...",password="...")
+username = "<lakefs-user>"
+api = lakefs.ApiClient(configuration)
+auth_api = lakefs.AuthApi(api)
+# attach the role(s)to a lakeFS user
+auth_api.create_user_external_principal(user_id=username, principal_id='arn:aws:sts::<id>:assumed-role/<role A>/<optional session name>')
+auth_api.create_user_external_principal(user_id=username, principal_id='arn:aws:sts::<id>:assumed-role/<role B>')
+# list the roles attached to the user
+resp = auth_api.list_user_external_principals(user_id=username)
+for p in resp.results:
+    # do something
+
+

+ + + Get lakeFS API Token + + +

+ + +

The login to lakeFS is done by calling the login API with the GetCallerIdentity request signed by the client. +Currently, the login operation is supported out of the box in:

+ + +

For other use cases authenticate to lakeFS via login endpoint, this will require building the request input.

+

+ + + Login with python + + +

+ +

+ + + prerequisites + + +

+ + +
    +
  1. lakeFS should be configured to allow external principals to authenticate and the used IAM role should be attached to the relevant lakeFS user
  2. +
  3. The Python SDK requires additional packages to be installed in order to generate a lakeFS client with the assumed role. +To install the required packages, run the following command:
  4. +
+ +
  pip install lakefs[aws-iam]
+
+ +

In order to generate a lakeFS client with the assumed role, initiate a boto3 session with the desired role and call lakefs.client.frow_aws_role:

+ +
import lakefs
+import boto3    
+session = boto3.Session()
+my_client = lakefs.client.from_aws_role(session=session, ttl_seconds=7200, host="<lakefs-host>")
+
+# list repositories
+repos = lakefs.repositories(client=my_client)
+for r in repos:
+    print(r)
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/index.html b/v1.46/security/index.html new file mode 100644 index 000000000..220017784 --- /dev/null +++ b/v1.46/security/index.html @@ -0,0 +1,753 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Security | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + lakeFS Security Reference + + +

+ + +

lakeFS Docs

+

+ + + Understanding Your Data Security + + +

+ + +

At lakeFS, we understand the critical nature of data security. Thousands of organizations worldwide rely on lakeFS to manage their data with confidence. Here’s a few concepts we follow to ensure your data remains secure:

+ +

Data Stays in Place: The data you version control remains within your existing object storage. lakeFS creates metadata for your data without moving it. New data is stored in the bucket you designate within your object storage.

+ +

lakeFS Servers Stores only Metadata: The lakeFS Server (also in the case of lakeFS Cloud) only stores the metadata used for version control operations (i.e. diff, merge). It does not store any of your actual data.

+ +

Minimal Permissions: lakeFS requires minimal permissions to manage your data. We can even version data we cannot directly access by utilizing presigned URLs.

+ +

Learn more about some of the featured that help keep lakeFS Secure:

+ +
    +
  • Authentication - An overview of the authentication and authorization mechanisms available in lakeFS, including built-in and external services, API and S3 Gateway authentication, and user permissions management.
  • +
  • Remote Authenticator - This feature allows organizations to leverage their existing identity infrastructure while using lakeFS, providing a flexible and secure authentication mechanism.
  • +
  • Role-Based Access Control (RBAC) - RBAC with lakeFS provides a flexible and granular approach to managing access and permissions, similar to other cloud-based systems like AWS IAM.
  • +
  • Presigned URL - This feature allows for more flexible and direct data access in lakeFS, particularly useful for scenarios where bypassing the lakeFS server for data retrieval or storage is beneficial.
  • +
  • Single Sign On (SSO) - lakeFS provides administrators with the necessary information to set up SSO for both lakeFS Cloud and Enterprise editions, covering various authentication protocols and identity providers.
  • +
  • Short Lived Token (STS like) - This feature allows lakeFS to leverage temporary credentials for secure and flexible authentication, integrating seamlessly with existing identity providers.
  • +
  • Login to lakeFS with AWS IAM - This feature enhances the integration between lakeFS and AWS by supporting authenticating users programmatically using AWS IAM roles instead of using static lakeFS access and secret keys.
  • +
+

+ + + SOC2 Compliance + + +

+ +

lakeFS Cloud is SOC2 compliant, demonstrating our commitment to stringent security standards.

+

+ + + More questions? Contact us + + +

+ +

For details on supported versions, security updates, and vulnerability reporting, please refer to our security policy on GitHub.

+ +

If you have additional questions regarding lakeFS Security, talk to an expert.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/presigned-url.html b/v1.46/security/presigned-url.html new file mode 100644 index 000000000..b3f7d1d4d --- /dev/null +++ b/v1.46/security/presigned-url.html @@ -0,0 +1,824 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Presigned URLs | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Configuring lakeFS to use presigned URLs + + +

+ + + + +

With lakeFS, you can access data directly from the storage and not through lakeFS using a presigned URL. +Based on the user’s access to an object in the object store, the presigned URL will get read or write access. +The presign support is enabled for block adapter that supports it (AWS, GCP, Azure), and can be disabled by the configuration (blockstore.<blockstore_type>.disable_pre_signed). Note that the UI support is disabled by default.

+ +
    +
  • It is possible to override the default pre-signed URL endpoint in AWS by setting the configuration (blockstore.s3.pre_signed_endpoint). +This is useful, for example, when you wish to define a VPC endpoint access for the pre-signed URL.
  • +
+

+ + + Using presigned URLs in the UI + + +

+ +

For using presigned URLs in the UI:

+
    +
  1. Enable the presigned URL support UI in the lakeFS configuration (blockstore.<blockstore_type>.disable_pre_signed_ui ).
  2. +
  3. Add CORS (Cross-Origin Resource Sharing) permissions to the bucket for the UI to fetch objects using a presigned URL (instead of through lakeFS).
  4. +
  5. The blockstore.<blockstore_type>.disable_pre_signed must be false to enable it in the UI.
  6. +
+ +

⚠️ Note Currently DuckDB fetching data from lakeFS does not support fetching data using presigned URL.

+

+ + + Example: AWS S3 + + +

+ + +
  [
+    {
+        "AllowedHeaders": [
+            "*"
+        ],
+        "AllowedMethods": [
+            "GET",
+            "PUT",
+            "HEAD"
+        ],
+        "AllowedOrigins": [
+            "lakefs.endpoint"
+        ],
+        "ExposeHeaders": [
+            "ETag"
+        ]
+    }
+  ]
+
+

+ + + Example: Google Storage + + +

+ + +
  [
+    {
+        "origin": ["lakefs.endpoint"],
+        "responseHeader": ["ETag"],
+        "method": ["PUT", "GET", "HEAD"],
+        "maxAgeSeconds": 3600
+    }
+   ]
+
+

+ + + Example: Azure blob storage + + +

+ + +
  <Cors>
+      <CorsRule>  
+          <AllowedOrigins>lakefs.endpoint</AllowedOrigins>  
+          <AllowedMethods>PUT,GET,HEAD</AllowedMethods>  
+          <AllowedHeaders>*</AllowedHeaders>  
+          <ExposedHeaders>ETag,x-ms-*</ExposedHeaders>  
+          <MaxAgeInSeconds>3600</MaxAgeInSeconds>  
+      </CorsRule>  
+  </Cors>
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/rbac.html b/v1.46/security/rbac.html new file mode 100644 index 000000000..e89a16b0c --- /dev/null +++ b/v1.46/security/rbac.html @@ -0,0 +1,1719 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Role-Based Access Control (RBAC) | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Role-Based Access Control (RBAC) + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

RBAC is available on lakeFS Cloud and lakeFS Enterprise.

+
+ + + + +

+ + + RBAC Model + + +

+ + +

Access to resources is managed very much like AWS IAM.

+ +

There are five basic components to the system:

+ +
    +
  1. +

    Users - Representing entities that access and use the system. A user is given one or more Access Credentials for authentication.

    +
  2. +
  3. +

    Actions - Representing a logical action within the system - reading a file, creating a repository, etc.

    +
  4. +
  5. +

    Resources - A unique identifier representing a specific resource in the system - a repository, an object, a user, etc.

    +
  6. +
  7. +

    Policies - Representing a set of Actions, a Resource and an effect: whether or not these actions are allowed or denied for the given resource(s).

    +
  8. +
  9. +

    Groups - A named collection of users. Users can belong to multiple groups.

    +
  10. +
+ +

Controlling access is done by attaching Policies, either directly to Users, or to Groups they belong to.

+

+ + + Authorization process + + +

+ + +

Every action in the system - be it an API request, UI interaction, S3 Gateway call, or CLI command - requires a set of actions to be allowed for one or more resources.

+ +

When a user makes a request to perform that action, the following process takes place:

+ +
    +
  1. Authentication - the credentials passed in the request are evaluated and the user’s identity is extracted.
  2. +
  3. Action permission resolution - lakeFS then calculates the set of allowed actions and resources that this request requires.
  4. +
  5. Effective policy resolution - the user’s policies (either attached directly or through group memberships) are calculated.
  6. +
  7. Policy/Permission evaluation - lakeFS will compare the given user policies with the request actions and determine whether or not the request is allowed to continue.
  8. +
+

+ + + Policy Precedence + + +

+ + +

Each policy attached to a user or a group has an Effect - either allow or deny. +During evaluation of a request, deny would take precedence over any other allow policy.

+ +

This helps us compose policies together. For example, we could attach a very permissive policy to a user and use deny rules to then selectively restrict what that user can do.

+

+ + + Resource naming - ARNs + + +

+ + +

lakeFS uses ARN identifier - very similar in structure to those used by AWS. +The resource segment of the ARN supports wildcards: use * to match 0 or more characters, or ? to match exactly one character.

+ +

Here are a some examples of valid ARNs within lakeFS and their meaning:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
ARNMeaning
arn:lakefs:auth:::user/jane.doeA specific user
arn:lakefs:auth:::user/*All users
arn:lakefs:fs:::repository/myrepo/*All resources under myrepo
arn:lakefs:fs:::repository/myrepo/object/foo/bar/bazA single object ARN
arn:lakefs:fs:::repository/myrepo/object/*All objects in myrepo
arn:lakefs:fs:::repository/*All repositories
arn:lakefs:fs:::*All resources under the fs ARN prefix
+ +

Additionally, the current user’s ID is interpolated in runtime into the ARN using the ${user} placeholder.

+ +

This allows us to create fine-grained policies affecting only a specific subset of resources.

+ +

See below for a full reference of ARNs and actions.

+

+ + + Actions and Permissions + + +

+ + +

For the full list of actions and their required permissions see the following table:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
Action namerequired actionResourceAPI endpointS3 gateway operation
List Repositoriesfs:ListRepositories*GET /repositoriesListBuckets
Get Repositoryfs:ReadRepositoryarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}HeadBucket
Get Commitfs:ReadCommitarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/commits/{commitId}-
Create Commitfs:CreateCommitarn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}POST /repositories/{repositoryId}/branches/{branchId}/commits-
Get Commit logfs:ReadBrancharn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}GET /repositories/{repositoryId}/branches/{branchId}/commits-
Create Repositoryfs:CreateRepositoryarn:lakefs:fs:::repository/{repositoryId}POST /repositories-
Namespace Attach to Repositoryfs:AttachStorageNamespacearn:lakefs:fs:::namespace/{storageNamespace}POST /repositories-
Import From Sourcefs:ImportFromStoragearn:lakefs:fs:::namespace/{storageNamespace}POST /repositories/{repositoryId}/branches/{branchId}/import-
Cancel Importfs:ImportCancelarn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}DELETE /repositories/{repositoryId}/branches/{branchId}/import-
Delete Repositoryfs:DeleteRepositoryarn:lakefs:fs:::repository/{repositoryId}DELETE /repositories/{repositoryId}-
List Branchesfs:ListBranchesarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/branchesListObjects/ListObjectsV2 (with delimiter = / and empty prefix)
Get Branchfs:ReadBrancharn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}GET /repositories/{repositoryId}/branches/{branchId}-
Create Branchfs:CreateBrancharn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}POST /repositories/{repositoryId}/branches-
Delete Branchfs:DeleteBrancharn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}DELETE /repositories/{repositoryId}/branches/{branchId}-
Merge branchesfs:CreateCommitarn:lakefs:fs:::repository/{repositoryId}/branch/{destinationBranchId}POST /repositories/{repositoryId}/refs/{sourceBranchId}/merge/{destinationBranchId}-
Diff branch uncommitted changesfs:ListObjectsarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/branches/{branchId}/diff-
Diff refsfs:ListObjectsarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/refs/{leftRef}/diff/{rightRef}-
Stat objectfs:ReadObjectarn:lakefs:fs:::repository/{repositoryId}/object/{objectKey}GET /repositories/{repositoryId}/refs/{ref}/objects/statHeadObject
Get Objectfs:ReadObjectarn:lakefs:fs:::repository/{repositoryId}/object/{objectKey}GET /repositories/{repositoryId}/refs/{ref}/objectsGetObject
List Objectsfs:ListObjectsarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/refs/{ref}/objects/lsListObjects, ListObjectsV2 (no delimiter, or “/” + non-empty prefix)
Upload Objectfs:WriteObjectarn:lakefs:fs:::repository/{repositoryId}/object/{objectKey}POST /repositories/{repositoryId}/branches/{branchId}/objectsPutObject, CreateMultipartUpload, UploadPart, CompleteMultipartUpload
Delete Objectfs:DeleteObjectarn:lakefs:fs:::repository/{repositoryId}/object/{objectKey}DELETE /repositories/{repositoryId}/branches/{branchId}/objectsDeleteObject, DeleteObjects, AbortMultipartUpload
Revert Branchfs:RevertBrancharn:lakefs:fs:::repository/{repositoryId}/branch/{branchId}PUT /repositories/{repositoryId}/branches/{branchId}-
Get Branch Protection Rulesbranches:GetBranchProtectionRulesarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/branch_protection-
Set Branch Protection Rulesbranches:SetBranchProtectionRulesarn:lakefs:fs:::repository/{repositoryId}POST /repositories/{repository}/branch_protection-
Delete Branch Protection Rulesbranches:SetBranchProtectionRulesarn:lakefs:fs:::repository/{repositoryId}DELETE /repositories/{repository}/branch_protection-
Create Userauth:CreateUserarn:lakefs:auth:::user/{userId}POST /auth/users-
List Usersauth:ListUsers*GET /auth/users-
Get Userauth:ReadUserarn:lakefs:auth:::user/{userId}GET /auth/users/{userId}-
Delete Userauth:DeleteUserarn:lakefs:auth:::user/{userId}DELETE /auth/users/{userId}-
Get Groupauth:ReadGrouparn:lakefs:auth:::group/{groupId}GET /auth/groups/{groupId}-
List Groupsauth:ListGroups*GET /auth/groups-
Create Groupauth:CreateGrouparn:lakefs:auth:::group/{groupId}POST /auth/groups-
Delete Groupauth:DeleteGrouparn:lakefs:auth:::group/{groupId}DELETE /auth/groups/{groupId}-
List Policiesauth:ListPolicies*GET /auth/policies-
Create Policyauth:CreatePolicyarn:lakefs:auth:::policy/{policyId}POST /auth/policies-
Update Policyauth:UpdatePolicyarn:lakefs:auth:::policy/{policyId}POST /auth/policies-
Delete Policyauth:DeletePolicyarn:lakefs:auth:::policy/{policyId}DELETE /auth/policies/{policyId}-
Get Policyauth:ReadPolicyarn:lakefs:auth:::policy/{policyId}GET /auth/policies/{policyId}-
List Group Membersauth:ReadGrouparn:lakefs:auth:::group/{groupId}GET /auth/groups/{groupId}/members-
Add Group Memberauth:AddGroupMemberarn:lakefs:auth:::group/{groupId}PUT /auth/groups/{groupId}/members/{userId}-
Remove Group Memberauth:RemoveGroupMemberarn:lakefs:auth:::group/{groupId}DELETE /auth/groups/{groupId}/members/{userId}-
List User Credentialsauth:ListCredentialsarn:lakefs:auth:::user/{userId}GET /auth/users/{userId}/credentials-
Create User Credentialsauth:CreateCredentialsarn:lakefs:auth:::user/{userId}POST /auth/users/{userId}/credentials-
Delete User Credentialsauth:DeleteCredentialsarn:lakefs:auth:::user/{userId}DELETE /auth/users/{userId}/credentials/{accessKeyId}-
Get User Credentialsauth:ReadCredentialsarn:lakefs:auth:::user/{userId}GET /auth/users/{userId}/credentials/{accessKeyId}-
List User Groupsauth:ReadUserarn:lakefs:auth:::user/{userId}GET /auth/users/{userId}/groups-
List User Policiesauth:ReadUserarn:lakefs:auth:::user/{userId}GET /auth/users/{userId}/policies-
Attach Policy To Userauth:AttachPolicyarn:lakefs:auth:::user/{userId}PUT /auth/users/{userId}/policies/{policyId}-
Detach Policy From Userauth:DetachPolicyarn:lakefs:auth:::user/{userId}DELETE /auth/users/{userId}/policies/{policyId}-
List Group Policiesauth:ReadGrouparn:lakefs:auth:::group/{groupId}GET /auth/groups/{groupId}/policies-
Attach Policy To Groupauth:AttachPolicyarn:lakefs:auth:::group/{groupId}PUT /auth/groups/{groupId}/policies/{policyId}-
Detach Policy From Groupauth:DetachPolicyarn:lakefs:auth:::group/{groupId}DELETE /auth/groups/{groupId}/policies/{policyId}-
Attach External Principal to a Userauth:CreateUserExternalPrincipalarn:lakefs:auth:::user/{userId}POST /auth/users/{userId}/external/principals-
Delete External Principal Attachment from a Userauth:DeleteUserExternalPrincipalarn:lakefs:auth:::user/{userId}DELETE /auth/users/{userId}/external/principals-
Get the User attached to an External Principalauth:ReadExternalPrincipalarn:lakefs:auth:::externalPrincipal/{principalId}GET /auth/external/principals-
Read Storage Configfs:ReadConfig*GET /config/storage-
Get Garbage Collection Rulesretention:GetGarbageCollectionRulesarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repositoryId}/gc/rules-
Set Garbage Collection Rulesretention:SetGarbageCollectionRulesarn:lakefs:fs:::repository/{repositoryId}POST /repositories/{repositoryId}/gc/rules-
Prepare Garbage Collection Commitsretention:PrepareGarbageCollectionCommitsarn:lakefs:fs:::repository/{repositoryId}POST /repositories/{repositoryId}/gc/prepare_commits-
List Repository Action Runsci:ReadActionarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/actions/runs-
Get Action Runci:ReadActionarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/actions/runs/{run_id}-
List Action Run Hooksci:ReadActionarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/actions/runs/{run_id}/hooks-
Get Action Run Hook Outputci:ReadActionarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/actions/runs/{run_id}/hooks/{hook_run_id}/output-
Get Pull Requestpr:ReadPullRequestarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/pulls/{pull_request}-
Create Pull Requestpr:WritePullRequestarn:lakefs:fs:::repository/{repositoryId}POST /repositories/{repository}/pulls-
Update Pull Requestpr:WritePullRequestarn:lakefs:fs:::repository/{repositoryId}PATCH /repositories/{repository}/pulls/{pull_request}-
Merge Pull Requestpr:WritePullRequest + Merge Branchesarn:lakefs:fs:::repository/{repositoryId}PUT /repositories/{repository}/pulls/{pull_request}/merge-
List Pull Requestspr:ListPullRequestsarn:lakefs:fs:::repository/{repositoryId}GET /repositories/{repository}/pulls-
+ +

Some APIs may require more than one action.For instance, in order to +create a repository (POST /repositories), you need permission to +fs:CreateRepository for the name of the repository and also +fs:AttachStorageNamespace for the storage namespace used.

+

+ + + Preconfigured Policies + + +

+ + +

The following Policies are created during initial setup:

+

+ + + FSFullAccess + + +

+ + +
{
+  "statement": [
+    {
+      "action": [
+        "fs:*"
+      ],
+      "effect": "allow",
+      "resource": "*"
+    }
+  ]
+}
+
+

+ + + FSReadAll + + +

+ + +
{
+  "statement": [
+    {
+      "action": [
+        "fs:List*",
+        "fs:Read*"
+      ],
+      "effect": "allow",
+      "resource": "*"
+    }
+  ]
+}
+
+

+ + + FSReadWriteAll + + +

+ + +
{
+    "statement": [
+        {
+            "action": [
+                "fs:Read*",
+                "fs:List*",
+                "fs:WriteObject",
+                "fs:DeleteObject",
+                "fs:RevertBranch",
+                "fs:CreateBranch",
+                "fs:CreateTag",
+                "fs:DeleteBranch",
+                "fs:DeleteTag",
+                "fs:CreateCommit",
+                "fs:CreateMetaRange"
+            ],
+            "effect": "allow",
+            "resource": "*"
+        }
+    ]
+}
+
+

+ + + AuthFullAccess + + +

+ + +
{
+  "statement": [
+    {
+      "action": [
+        "auth:*"
+      ],
+      "effect": "allow",
+      "resource": "*"
+    }
+  ]
+}
+
+

+ + + AuthManageOwnCredentials + + +

+ + +
{
+  "statement": [
+    {
+      "action": [
+        "auth:CreateCredentials",
+        "auth:DeleteCredentials",
+        "auth:ListCredentials",
+        "auth:ReadCredentials"
+      ],
+      "effect": "allow",
+      "resource": "arn:lakefs:auth:::user/${user}"
+    }
+  ]
+}
+
+

+ + + RepoManagementFullAccess + + +

+ + +
{
+    "statement": [
+        {
+            "action": [
+                "ci:*"
+            ],
+            "effect": "allow",
+            "resource": "*"
+        },
+        {
+            "action": [
+                "retention:*"
+            ],
+            "effect": "allow",
+            "resource": "*"
+        }
+    ]
+}
+
+

+ + + RepoManagementReadAll + + +

+ + +
{
+    "statement": [
+        {
+            "action": [
+                "ci:Read*"
+            ],
+            "effect": "allow",
+            "resource": "*"
+        },
+        {
+            "action": [
+                "retention:Get*"
+            ],
+            "effect": "allow",
+            "resource": "*"
+        }
+    ]
+}
+
+

+ + + Additional Policies + + +

+ + +

You can create additional policies to further limit user access. Use the web UI or the lakectl auth command to create policies. Here is an example to define read/write access for a specific repository:

+ +
{
+    "statement": [
+        {
+            "action": [
+                "fs:ReadRepository",
+                "fs:ReadCommit",
+                "fs:ListBranches",
+                "fs:ListTags",
+                "fs:ListObjects"
+            ],
+            "effect": "allow",
+            "resource": "arn:lakefs:fs:::repository/<repository-name>"
+        },
+        {
+            "action": [
+                "fs:RevertBranch",
+                "fs:ReadBranch",
+                "fs:CreateBranch",
+                "fs:DeleteBranch",
+                "fs:CreateCommit"
+            ],
+            "effect": "allow",
+            "resource": "arn:lakefs:fs:::repository/<repository-name>/branch/*"
+        },
+                {
+            "action": [
+                "fs:ListObjects",
+                "fs:ReadObject",
+                "fs:WriteObject",
+                "fs:DeleteObject"
+            ],
+            "effect": "allow",
+            "resource": "arn:lakefs:fs:::repository/<repository-name>/object/*"
+        },
+                {
+            "action": [
+                "fs:ReadTag",
+                "fs:CreateTag",
+                "fs:DeleteTag"
+            ],
+            "effect": "allow",
+            "resource": "arn:lakefs:fs:::repository/<repository-name>/tag/*"
+        },
+        {
+        	"action": ["fs:ReadConfig"],
+        	"effect": "allow",
+        	"resource": "*"
+        }
+    ]
+}
+
+

+ + + Preconfigured Groups + + +

+ + +

lakeFS has four preconfigured groups:

+ +
    +
  • Admins
  • +
  • SuperUsers
  • +
  • Developers
  • +
  • Viewers
  • +
+ +

They have the following policies granted to them:

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
PolicyAdminsSuperUsersDevelopersViewers
FSFullAccess  
AuthFullAccess   
RepoManagementFullAccess   
AuthManageOwnCredentials 
RepoManagementReadAll  
FSReadWriteAll   
FSReadAll   
+

+ + + Pluggable Authentication and Authorization + + +

+ + +

Authorization and authentication is pluggable in lakeFS. If lakeFS is attached to a remote authentication server (or you are using lakeFS Cloud) then the role-based access control user interface can be used. If you are using RBAC with your self-managed lakeFS then the lakeFS configuration element auth.ui_config.rbac should be set to external. An enterprise (paid) solution of lakeFS should set auth.ui_config.rbac as internal.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/remote-authenticator.html b/v1.46/security/remote-authenticator.html new file mode 100644 index 000000000..fafccfd09 --- /dev/null +++ b/v1.46/security/remote-authenticator.html @@ -0,0 +1,914 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Remote Authenticator | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Remote Authenticator + + +

+ + +

Remote Authenticator is a pluggable architecture for lakeFS which allows you to use existing organizational identity policies and infrastructure with the authentication mechanism of lakeFS. The Remote Authenticator’s job is to abstract away the complexities of existing infrastructure and implement a standard interface, which lakeFS can use to resolve user identity and manage access to lakeFS. This loose coupling allows you to implement federated identity without providing lakeFS with direct access to your identity infrastructure.

+

+ + + Architecture + + +

+ + +

Here’s the authentication flow that lakeFS uses when configured with a remote authenticator:

+ +
sequenceDiagram
+    participant A as lakeFS Client
+    participant B as lakeFS Server
+    participant C as Remote Authenticator
+    participant D as IdP
+    A->>B: Submit login form
+    B->>C: POST user credentials
+    C->>D: IdP request
+    D->>C: IdP response
+    C->>B: Auth response
+    B->>A: auth JWT
+
+

+ + + The Interface + + +

+ + +

To configure lakeFS to work with a Remote Authenticator add the following YAML to your lakeFS configuration:

+ +
auth:
+    remote_authenticator:
+        enabled: true
+        endpoint: <url-to-remote-authenticator-endpoint>
+        default_user_group: "Developers"
+    ui_config:
+        logout_url: /logout
+        login_cookie_names:
+            - internal_auth_session
+
+ +
    +
  • auth.remote_authenticator.enabled - set lakeFS to use the remote authenticator
  • +
  • auth.remote_authenticator.endpoint - an endpoint where the remote authenticator is able to receive a POST request from lakeFS
  • +
  • auth.remote_authenticator.default_user_group - the group assigned by default to new users
  • +
  • auth.ui_config.logout_url - the URL to redirect the browser when clicking the logout link in the user menu
  • +
  • auth.ui_config.login_cookie_names - the name of the cookie(s) lakeFS will set following a successful authentication. The value is the authenticated user’s JWT
  • +
+ +

A Remote Authenticator implementation should expose a single endpoint, which expects the following JSON request:

+ +
{
+    "username": "testy.mctestface@example.com",
+    "password": "Password1"
+}
+
+ +

and returns a JSON response like this:

+ +
{
+    "external_user_identifier": "TestyMcTestface"
+}
+
+

+ + + Example Request & Responses + + +

+ +

+ + + Request + + +

+ + +
POST https://remote-authenticator.example.com/auth
+Content-Type: application/json
+
+{
+  "username": "testy.mctestface@example.com",
+  "password": "Password1"
+}
+
+

+ + + Successful Response + + +

+ +
HTTP/1.1 200 OK
+Content-Type: application/json
+
+
+{
+  "external_user_identifier": "TestyMcTestface"
+}
+
+

+ + + Unauthorized Response + + +

+ +
HTTP/1.1 401 Unauthorized
+Content-Type: application/json
+
+{
+  "external_user_identifier": ""
+}
+
+ +

If the Remote Authenticator returns any HTTP status in the 2xx range, lakeFS considers this a successful authentication. Any HTTP status < 200 or > 300 is considered a failed authentication. If the Remote Authenticator returns a non-empty value for the external_user_identifier property along with a success HTTP status, lakeFS will show this identifier instead of an internal lakeFS user identifier in the UI.

+

+ + + Sample Implementation + + +

+ + +

Here is a sample Remote Authenticator implemented using node and express and written in TypeScript. This example implementation doesn’t integrate with any real IdP but illustrates the expected request/response patterns that you need to implement.

+ +
import dotenv from "dotenv";
+import express, { Express, Request, Response } from "express";
+import { StatusCodes } from "http-status-codes";
+
+type AuthRequestBody = {
+  username: string;
+  password: string;
+};
+
+type AuthResponseBody = {
+  external_user_identifier: string;
+};
+
+const DEFAULT_PORT = 80;
+
+dotenv.config();
+
+const port = process.env.PORT || DEFAULT_PORT;
+const app: Express = express();
+
+app.post(
+  "/auth",
+  (req: Request<AuthResponseBody, {}, AuthRequestBody>, res: Response) => {
+    const { username, password } = req.body;
+    if (!username?.length || !password?.length) {
+      return res.status(StatusCodes.BAD_REQUEST).json({
+        external_user_identifier: "",
+      });
+    }
+
+    // 👇🏻 This is where you would implement your own authentication logic
+    if (
+      username === "testy.mctestface@example.com" &&
+      password === "Password1"
+    ) {
+      return res.status(StatusCodes.OK).json({
+        external_user_identifier: "TestyMcTestface",
+      });
+    } else {
+      return res.status(StatusCodes.UNAUTHORIZED).json({
+        external_user_identifier: "",
+      });
+    }
+  }
+);
+
+app.listen(port, () => {
+  console.log(`Remote Authenticator listening on port ${port}`);
+});
+
+ +

To run this service on the sub-domain idp.example.com, use a lakeFS configuration that looks like this:

+ +
auth:
+    remote_authenticator:
+        enabled: true
+        endpoint: https://idp.example.com/auth
+        default_user_group: "Developers"
+    ui_config:
+        logout_url: /logout
+        login_cookie_names:
+            - internal_auth_session
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/sso.html b/v1.46/security/sso.html new file mode 100644 index 000000000..a5592a68a --- /dev/null +++ b/v1.46/security/sso.html @@ -0,0 +1,1248 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Single Sign On (SSO) | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Single Sign On (SSO) + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

SSO is available for lakeFS Cloud and lakeFS Enterprise. If you’re using the open-source version of lakeFS you can read more about the authentication options available.

+
+

+ + + SSO for lakeFS Cloud + + +

+ + +

lakeFS Cloud uses Auth0 for authentication and thus support the same identity providers as Auth0 including Active Directory/LDAP, ADFS, Azure Active Directory Native, Google Workspace, OpenID Connect, Okta, PingFederate, SAML, and Azure Active Directory.

+ +
+ +
+

+ + + Okta + + +

+ + +
+

This guide is based on Okta’s Create OIDC app integrations guide.

+
+ +

Steps:

+
    +
  1. Login to your Okta account
  2. +
  3. Select Applications > Applications, then Create App Integration.
  4. +
  5. Select Create New App and enter the following: +
      +
    1. For Sign-in method, choose OIDC.
    2. +
    3. Under Application type, choose Web app.
    4. +
    5. Select Next.
    6. +
    +
  6. +
  7. Under General Settings: +
      +
    1. App integration name, enter a name for your application. (i.e lakeFS Cloud)
    2. +
    +
  8. +
  9. In the Sign-in redirect URIs field, enter https://lakefs-cloud.us.auth0.com/login (United States) or https://lakefs-cloud.eu.auth0.com/login (Europe).
  10. +
  11. Under Sign-in redirect URIs, click Add URI, enter https://lakefs-cloud.us.auth0.com/login/callback (United States) or https://lakefs-cloud.eu.auth0.com/login/callback (Europe).
  12. +
  13. Under Assignments, choose the wanted Controlled access. (i.e Allow everyone in your organization to access)
  14. +
  15. Uncheck Enable immediate access with Federation Broker Mode.
  16. +
  17. Select Save.
  18. +
+ +

Once you finish registering your application with Okta, save the Client ID, Client Secret and your Okta Domain, send this to Treeverse’s team to finish the integration.

+
+
+

+ + + Active Directory Federation Services (AD FS) + + +

+ + +

Prerequisites:

+
    +
  • Client’s AD FS server should be exposed publicly or to Auth0’s IP ranges (either directly or using Web Application Proxy)
  • +
+ +

Steps:

+
    +
  1. Connect to the AD FS server
  2. +
  3. Open AD FS’ PowerShell CLI as Administrator through the server manager
  4. +
  5. +

    Execute the following:

    + +
     (new-object Net.WebClient -property @{Encoding = [Text.Encoding]::UTF8}).DownloadString("https://raw.github.com/auth0/adfs-auth0/master/adfs.ps1") | iex
    +
    + AddRelyingParty "urn:auth0:lakefs-cloud" "https://lakefs-cloud.us.auth0.com/login/callback"
    +
    + +

    Note: If your organization data is located in Europe, use lakefs-cloud.eu.auth0.com instead of lakefs-cloud.us.auth0.com.

    +
  6. +
+ +

Once you finish registering lakeFS Cloud with AD FS, save the AD FS URL and send this to Treeverse’s team to finish the integration.

+
+
+

+ + + Azure Active Directory (AD) + + +

+ + +

Prerequisites:

+
    +
  • Azure account with permissions to manage applications in Azure Active Directory
  • +
+ +

Note: If you’ve already set up lakeFS Cloud with your Azure account, you can skip the Register lakeFS Cloud with Azure and Add client secret and go directly to Add a redirect URI.

+

+ + + Register lakeFS Cloud with Azure + + +

+ + +

Steps:

+
    +
  1. Sign in to the Azure portal.
  2. +
  3. If you have access to multiple tenants, use the Directories + subscriptions filter in the top menu to switch to the tenant in which you want to register the application.
  4. +
  5. Search for and select Azure Active Directory.
  6. +
  7. Under Manage, select App registrations > New registration.
  8. +
  9. Enter a display Name for your application. Users of your application might see the display name when they use the app, for example during sign-in. You can change the display name at any time and multiple app registrations can share the same name. The app registration’s automatically generated Application (client) ID, not its display name, uniquely identifies your app within the identity platform.
  10. +
  11. +

    Specify who can use the application, sometimes called its sign-in audience.

    + +

    Note: don’t enter anything for Redirect URI (optional). You’ll configure a redirect URI in the next section.

    +
  12. +
  13. Select Register to complete the initial app registration.
  14. +
+ +

When registration finishes, the Azure portal displays the app registration’s Overview pane. You see the Application (client) ID. Also called the client ID, this value uniquely identifies your application in the Microsoft identity platform.

+ +

Important: new app registrations are hidden to users by default. When you are ready for users to see the app on their My Apps page you can enable it. To enable the app, in the Azure portal navigate to Azure Active Directory > Enterprise applications and select the app. Then on the Properties page toggle Visible to users? to Yes.

+

+ + + Add a secret + + +

+ +

Sometimes called an application password, a client secret is a string value your app can use in place of a certificate to identity itself.

+ +

Steps:

+
    +
  1. In the Azure portal, in App registrations, select your application.
  2. +
  3. Select Certificates & secrets > Client secrets > New client secret.
  4. +
  5. Add a description for your client secret.
  6. +
  7. Select an expiration for the secret or specify a custom lifetime. +
      +
    1. Client secret lifetime is limited to two years (24 months) or less. You can’t specify a custom lifetime longer than 24 months.
    2. +
    3. Microsoft recommends that you set an expiration value of less than 12 months.
    4. +
    +
  8. +
  9. Select Add.
  10. +
  11. Record the secret’s value for use in your client application code. This secret value is never displayed again after you leave this page.
  12. +
+

+ + + Add a redirect URI + + +

+ +

A redirect URI is the location where the Microsoft identity platform redirects a user’s client and sends security tokens after authentication.

+ +

You add and modify redirect URIs for your registered applications by configuring their platform settings.

+ +

Enter https://lakefs-cloud.us.auth0.com/login/callback as your redirect URI.

+ +

Settings for each application type, including redirect URIs, are configured in Platform configurations in the Azure portal. Some platforms, like Web and Single-page applications, require you to manually specify a redirect URI. For other platforms, like mobile and desktop, you can select from redirect URIs generated for you when you configure their other settings.

+ +

Steps:

+
    +
  1. In the Azure portal, in App registrations, select your application.
  2. +
  3. Under Manage, select Authentication.
  4. +
  5. Under Platform configurations, select Add a platform.
  6. +
  7. Under Configure platforms, select the web option.
  8. +
  9. Select Configure to complete the platform configuration.
  10. +
+ +

Once you finish registering lakeFS Cloud with Azure AD send the following items to the Treeverse’s team:

+
    +
  1. Client ID
  2. +
  3. Client Secret
  4. +
  5. Azure AD Domain
  6. +
  7. Identity API Version (v1 for Azure AD or v2 for Microsoft Identity Platform/Entra)
  8. +
+ +
+
+

+ + + SSO for lakeFS Enterprise + + +

+ + +

Authentication in lakeFS Enterprise is handled by a secondary service which runs side-by-side with lakeFS. With a nod to Hogwarts and their security system, we’ve named this service Fluffy. Details for configuring the supported identity providers with Fluffy are shown below. In addition, please review the necessary Helm configuration to configure Fluffy.

+ +
    +
  • Active Directory Federation Services (AD FS) (using SAML)
  • +
  • OpenID Connect
  • +
  • LDAP
  • +
+ +

If you’re using an authentication provider that is not listed please contact us for further assistance.

+ +
+ +
+

+ + + Active Directory Federation Services (AD FS) (using SAML) + + +

+ + +
+

AD FS integration uses certificates to sign & encrypt requests going out from Fluffy and decrypt incoming requests from AD FS server.

+
+ +

In order for Fluffy to work, the following values must be configured. Update (or override) the following attributes in the chart’s values.yaml file.

+
    +
  1. Replace fluffy.saml_rsa_public_cert and fluffy.saml_rsa_private_key with real certificate values
  2. +
  3. Replace fluffyConfig.auth.saml.idp_metadata_url with the metadata URL of the AD FS provider (e.g adfs-auth.company.com)
  4. +
  5. Replace fluffyConfig.auth.saml.external_user_id_claim_name with the claim name representing user id name in AD FS
  6. +
  7. Replace lakefs.company.com with your lakeFS server URL.
  8. +
+ +

If you’d like to generate the certificates using OpenSSL, you can take a look at the following example:

+ +
openssl req -x509 -newkey rsa:2048 -keyout myservice.key -out myservice.cert -days 365 -nodes -subj "/CN=lakefs.company.com" -
+
+ +

lakeFS Server Configuration (Update in helm’s values.yaml file):

+ +
auth:
+  cookie_auth_verification:
+    auth_source: saml
+    friendly_name_claim_name: displayName
+    persist_friendly_name: true
+    external_user_id_claim_name: samName
+    default_initial_groups:
+      - "Developers"
+  logout_redirect_url: "https://lakefs.company.com/logout-saml"
+  encrypt:
+    secret_key: shared-secrey-key
+  ui_config:
+    login_url: "https://lakefs.company.com/sso/login-saml"
+    logout_url: "https://lakefs.company.com/sso/logout-saml"
+    login_cookie_names:
+      - internal_auth_session
+      - saml_auth_session
+
+ +

Fluffy Configuration (Update in helm’s values.yaml file):

+ +
logging:
+  format: "json"
+  level: "INFO"
+  audit_log_level: "INFO"
+  output: "="
+auth:  
+  encrypt:
+    secret_key: shared-secrey-key    
+  logout_redirect_url: https://lakefs.company.com
+  post_login_redirect_url: https://lakefs.company.com
+  saml:
+    enabled: true 
+    sp_root_url: https://lakefs.company.com
+    sp_x509_key_path: '/etc/saml_certs/rsa_saml_private.cert'
+    sp_x509_cert_path: '/etc/saml_certs/rsa_saml_public.pem'
+    sp_sign_request: true
+    sp_signature_method: "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"
+    idp_metadata_url: "https://adfs-auth.company.com/federationmetadata/2007-06/federationmetadata.xml"
+    # idp_authn_name_id_format: "urn:oasis:names:tc:SAML:1.1:nameid-format:unspecified"
+    external_user_id_claim_name: samName
+    # idp_metadata_file_path: 
+    # idp_skip_verify_tls_cert: true
+
+
+
+

+ + + OpenID Connect + + +

+ + +

In order for Fluffy to work, the following values must be configured. Update (or override) the following attributes in the chart’s values.yaml file.

+
    +
  1. Replace lakefsConfig.friendly_name_claim_name with the right claim name.
  2. +
  3. Replace lakefsConfig.default_initial_groups with desired claim name (See pre-configured groups for enterprise)
  4. +
  5. Replace fluffyConfig.auth.logout_redirect_url with your full OIDC logout URL (e.g https://oidc-provider-url.com/logout/path)
  6. +
  7. Replace fluffyConfig.auth.oidc.url with your OIDC provider URL (e.g https://oidc-provider-url.com)
  8. +
  9. Replace fluffyConfig.auth.oidc.logout_endpoint_query_parameters with parameters you’d like to pass to the OIDC provider for logout.
  10. +
  11. Replace fluffyConfig.auth.oidc.client_id and fluffyConfig.auth.oidc.client_secret with the client ID & secret for OIDC.
  12. +
  13. Replace fluffyConfig.auth.oidc.logout_client_id_query_parameter with the query parameter that represent the client_id, note that it should match the the key/query param that represents the client id and required by the specific OIDC provider.
  14. +
  15. Replace lakefs.company.com with the lakeFS server URL.
  16. +
+ +

lakeFS Server Configuration (Update in helm’s values.yaml file):

+ +
  # Important: make sure to include the rest of your lakeFS Configuration here!
+auth:
+  encrypt:
+    secret_key: shared-secrey-key
+  oidc:
+    friendly_name_claim_name: "name"
+    persist_friendly_name: true
+    default_initial_groups: ["Developers"]
+  ui_config:
+    login_url: /oidc/login
+    logout_url: /oidc/logout
+    login_cookie_names:
+      - internal_auth_session
+      - oidc_auth_session
+
+ +

Fluffy Configuration (Update in helm’s values.yaml file):

+ +
logging:
+  format: "json"
+  level: "INFO"
+  audit_log_level: "INFO"
+  output: "="
+installation:
+  fixed_id: fluffy-authenticator
+auth:
+  post_login_redirect_url: /
+  logout_redirect_url: https://oidc-provider-url.com/logout/url
+  oidc:
+    enabled: true
+    url: https://oidc-provider-url.com/
+    client_id: <oidc-client-id>
+    client_secret: <oidc-client-secret>
+    callback_base_url: https://lakefs.company.com
+    is_default_login: true
+    logout_client_id_query_parameter: client_id
+    logout_endpoint_query_parameters:
+      - returnTo 
+      - https://lakefs.company.com/oidc/login
+  encrypt:
+    secret_key: shared-secrey-key
+
+
+
+

+ + + LDAP + + +

+ + +

Fluffy is incharge of providing LDAP authentication for lakeFS Enterprise. +The authentication works by querying the LDAP server for user information and authenticating the user based on the provided credentials.

+ +

Important: An administrative bind user must be configured. It should have search permissions for the LDAP server that will be used to query the LDAP server for user information.

+ +

For Helm: set the following attributes in the Helm chart values, for lakeFS lakefsConfig.* and fluffyConfig.* for fluffy.

+ +

No Helm: If not using Helm use the YAML below to directly update the configuration file for each service.

+ +

lakeFS Configuration:

+ +
    +
  1. Replace auth.remote_authenticator.enabled with true
  2. +
  3. Replace auth.remote_authenticator.endpoint with the fluffy authentication server URL combined with the api/v1/ldap/login suffix (e.g http://lakefs.company.com/api/v1/ldap/login)
  4. +
+ +

fluffy Configuration:

+ +

See Fluffy configuration reference.

+ +
    +
  1. Repalce auth.ldap.remote_authenticator.server_endpoint with your LDAP server endpoint (e.g ldaps://ldap.ldap-address.com:636)
  2. +
  3. Replace auth.ldap.remote_authenticator.bind_dn with the LDAP bind user/permissions to query your LDAP server.
  4. +
  5. Replace auth.ldap.remote_authenticator.user_base_dn with the user base to search users in.
  6. +
+ +

lakeFS Server Configuration file:

+ +

$lakefs run -c ./lakefs.yaml

+ +
# Important: make sure to include the rest of your lakeFS Configuration here!
+
+auth:
+  remote_authenticator:
+    enabled: true
+    endpoint: http://<Fluffy URL>:<Fluffy http port>/api/v1/ldap/login
+    default_user_group: "Developers" # Value needs to correspond with an existing group in lakeFS
+  ui_config:
+    logout_url: /logout
+    login_cookie_names:
+      - internal_auth_session
+
+ +

Fluffy Configuration file:

+ +

$fluffy run -c ./fluffy.yaml

+ +
logging:
+  format: "json"
+  level: "INFO"
+  audit_log_level: "INFO"
+  output: "="
+installation:
+  fixed_id: fluffy-authenticator
+auth:
+  post_login_redirect_url: /
+  ldap: 
+    server_endpoint: 'ldaps://ldap.company.com:636'
+    bind_dn: uid=<bind-user-name>,ou=<some-ou>,o=<org-id>,dc=<company>,dc=com
+    bind_password: '<ldap pwd>'
+    username_attribute: uid
+    user_base_dn: ou=<some-ou>,o=<org-id>,dc=<company>,dc=com
+    user_filter: (objectClass=inetOrgPerson)
+    connection_timeout_seconds: 15
+    request_timeout_seconds: 7
+
+

+ + + Troubleshooting LDAP issues + + +

+ +

+ + + Inspecting Logs + + +

+ + +

If you encounter LDAP connection errors, you should inspect the fluffy container logs to get more information.

+

+ + + Authentication issues + + +

+ + +

Auth issues (e.g. user not found, invalid credentials) can be debugged with the ldapwhoami CLI tool.

+ +

The Examples are based on the fluffy config above:

+ +

To verify that the main bind user can connect:

+ +
ldapwhoami -H ldap://ldap.company.com:636 -D "uid=<bind-user-name>,ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -x -W
+
+ +

To verify that a specific lakeFS user dev-user can connect:

+ +
ldapwhoami -H ldap://ldap.company.com:636 -D "uid=dev-user,ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -x -W
+
+

+ + + User not found issue + + +

+ + +

Upon a login request in fluffy, the bind user will search for the user in the LDAP server. If the user is not found it will be presented in the logs.

+ +

We can search the user using ldapsearch CLI tool.

+ +

Search ALL users in the base DN (no filters):

+ +

Note: -b is the user_base_dn, -D is bind_dn and -w is bind_password from the fluffy configuration.

+ +
ldapsearch -H ldap://ldap.company.com:636 -x -b "ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -D "uid=<bind-user-name>,ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -w '<bind_user_pwd>'
+
+ +

If the user is found, we should now use filters for the specific user the same way fluffy does it and expect to see the user.

+ +

For example, to repdocue the same search as fluffy does:

+
    +
  • user dev-user set from uid attribute in LDAP
  • +
  • Fluffy configuration values: user_filter: (objectClass=inetOrgPerson) and username_attribute: uid
  • +
+ +
ldapsearch -H ldap://ldap.company.com:636 -x -b "ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -D "uid=<bind-user-name>,ou=<some-ou>,o=<org-id>,dc=<company>,dc=com" -w '<bind_user_pwd>' "(&(uid=dev-user)(objectClass=inetOrgPerson))"
+
+ +
+
+

+ + + Helm + + +

+ + +

In order to use lakeFS Enterprise and Fluffy, we provided out of the box setup, see lakeFS Helm chart configuration.

+ +

Notes:

+
    +
  • Check the examples on GitHub we provide for each authentication method (oidc/adfs/ldap + rbac).
  • +
  • The examples are provisioned with a Postgres pod for quick-start, make sure to replace that to a stable database once ready.
  • +
  • The encrypt secret key secrets.authEncryptSecretKey is shared between fluffy and lakeFS for authentication.
  • +
  • The lakeFS image.tag must be >= 1.0.0
  • +
  • The fluffy image.tag must be >= 0.2.7
  • +
  • Change the ingress.hosts[0] from lakefs.company.com to a real host (usually same as lakeFS), also update additional references in the file (note: URL path after host if provided should stay unchanged).
  • +
  • Update the ingress configuration with other optional fields if used
  • +
  • Fluffy docker image: replace the fluffy.image.privateRegistry.secretToken with real token to dockerhub for the fluffy docker image.
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/security/sts-login.html b/v1.46/security/sts-login.html new file mode 100644 index 000000000..e218d5e66 --- /dev/null +++ b/v1.46/security/sts-login.html @@ -0,0 +1,882 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Short-lived token (STS like) Authentication for lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Short-lived token (STS like) + + +

+ +

lakeFS Cloud

+ +

lakeFS Enterprise

+ +
+

STS Login is available in lakeFS Cloud and lakeFS Enterprise. currently only tested on lakeFS Enterprise.

+
+ +

Secure Token Service (STS) authentication in lakeFS enables users to authenticate to lakeFS using temporary credentials obtained from an Identity Provider (IdP) via the OpenID Connect (OIDC) Authentication workflow. +This document outlines the process of setting up the STS authentication flow and using the temporary credentials to interact with lakeFS through the high-level Python SDK.

+

+ + + Login + + +

+ + +

Initiate a client session with temporary credentials using the high-level Python SDK:

+ +
import lakefs
+
+my_client = lakefs.client.from_web_identity(code = '<CODE_FROM_IDP>', state = '<STATE_FROM_IDP>' , redirect_uri = '<URI_USED_FOR_REDIRECT_FROM_IDP>', ttl_seconds = 7200)
+
+

+ + + Setup + + +

+ +

+ + + Prerequisites + + +

+ +

Ensure you have a way to generate the code, redirect_uri, and state values that are required to initiate a new client session using the STS login feature. +For a reference implementation, see the sample implementation section.

+

+ + + Configuration + + +

+ + +

To enable STS authentication, configure your lakeFS instance with the endpoint of the external Authentication Service.

+ +

+auth:
+    authentication_api:
+        endpoint: <url-to-remote-authenticator-endpoint>
+
+
+
+ +

The endpoint value should point to the external Authentication Service described at authentication.yml.
+Make sure to replace with the actual URL of your Authentication Service.

+

+ + + Sample implementation to generate the code, redirect_uri, and state + + +

+ +

The following code snippet demonstrates how to generate the values that are required to initiate a new client session using the STS login feature.

+ +
+

Replace <your-authorize-endpoint> with the path to your IdP authorize endpoint.
+Examples:
+Auth0: The authorize endpoint will be https://<your-auth0-domain>/authorize
+Entra ID: The authorize endpoint will be https://<your-entra-domain>/oauth2/v2.0/authorize

+
+ +
import crypto from 'crypto';
+
+import express from 'express';
+import axios from 'axios';
+import url from 'url';
+import jsonwebtoken from 'jsonwebtoken';
+
+const app = express();
+// the local script will will spin up the server and the IdP provider will return to this endpoint the response.
+const callback = "http://localhost:8080/oidc/callback"
+const authorizeEndpoint = "<your-authorize-endpoint>"
+
+// step 1 
+// Create a code_verifier, which is a cryptographically-random, Base64-encoded key that will eventually be sent to Auth0 to request tokens.
+function base64URLEncode(str) {
+    return str.toString('base64')
+        .replace(/\+/g, '-')
+        .replace(/\//g, '_')
+        .replace(/=/g, '');
+}
+var verifier = base64URLEncode(crypto.randomBytes(32));
+console.log(`verifier: ${verifier}`);
+
+// step 2 
+// Generate a code_challenge from the code_verifier that will be sent to Auth0 to request an authorization_code.
+function sha256(buffer) {
+    return crypto.createHash('sha256').update(buffer).digest();
+}
+
+var challenge = base64URLEncode(sha256(verifier));
+console.log(`challenge: ${challenge}`);
+
+
+const authorizeURL = `${authorizeEndpoint}?response_type=code&code_challenge=${challenge}&code_challenge_method=S256&client_id=${auth0ClientId}&redirect_uri=${callback}&scope=openid&state=${verifier}`
+
+console.log(`authorizeURL: ${authorizeURL}`)
+
+// Endpoint for OIDC callback
+app.get('/oidc/callback', async (req, res) => {
+    try {
+        const code = req.query.code;
+        const state = req.query.state;
+        console.log(`code: ${code}`);
+        console.log(`state: ${state}`);
+        // Return a success response
+        res.status(200).json({ code, state, redirect_uri: callback, python-cmd: `lakefs.client.from_web_identity(code = ${code} redirect_uri = ${callback} state = ${state}, ttl_seconds = 7200) ` });
+        return
+    } catch (err) {
+        console.error(err);
+        res.status(500).json({ message: 'Internal server error' });
+    }
+});
+
+// Start the server
+const PORT = 8080;
+app.listen(PORT, () => {
+    console.log(`Server is running on port ${PORT}`);
+});
+
+

+ + + Architecture + + +

+ + +

The STS authentication flow involves several components that facilitate secure communication between the lakeFS client, lakeFS server, the remote authenticator, and the IdP.

+ +
sequenceDiagram
+    participant A as lakeFS Client
+    participant B as lakeFS Server
+    participant C as Remote Authenticator
+    participant D as IdP
+    A->>B: Call STS login endpoint
+    B->>C: POST idp code state and redirect uri
+    C->>D: IdP request
+    D->>C: IdP response
+    C->>B: Auth response
+    B->>A: auth JWT
+
+
    +
  • lakeFS Client: Initiates the authentication process by providing IdP credentials.
  • +
  • lakeFS Server: Facilitates the authentication request between the client and the remote authenticator.
  • +
  • Remote Authenticator: Acts as a bridge between lakeFS and the IdP, handling credential validation.
  • +
  • IdP (Identity Provider): Validates the provided credentials and returns the authentication status.
  • +
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/setup/create-repo.html b/v1.46/setup/create-repo.html new file mode 100644 index 000000000..da83e1cb5 --- /dev/null +++ b/v1.46/setup/create-repo.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/hooks.html b/v1.46/setup/hooks.html new file mode 100644 index 000000000..2ee933063 --- /dev/null +++ b/v1.46/setup/hooks.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/import.html b/v1.46/setup/import.html new file mode 100644 index 000000000..4a3a4ef72 --- /dev/null +++ b/v1.46/setup/import.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/index.html b/v1.46/setup/index.html new file mode 100644 index 000000000..da83e1cb5 --- /dev/null +++ b/v1.46/setup/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/storage/blob.html b/v1.46/setup/storage/blob.html new file mode 100644 index 000000000..3ab302607 --- /dev/null +++ b/v1.46/setup/storage/blob.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/storage/gcs.html b/v1.46/setup/storage/gcs.html new file mode 100644 index 000000000..e0a39904c --- /dev/null +++ b/v1.46/setup/storage/gcs.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/storage/index.html b/v1.46/setup/storage/index.html new file mode 100644 index 000000000..da83e1cb5 --- /dev/null +++ b/v1.46/setup/storage/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/setup/storage/s3.html b/v1.46/setup/storage/s3.html new file mode 100644 index 000000000..fcfed0b37 --- /dev/null +++ b/v1.46/setup/storage/s3.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/sitemap.xml b/v1.46/sitemap.xml new file mode 100644 index 000000000..5817d8136 --- /dev/null +++ b/v1.46/sitemap.xml @@ -0,0 +1,375 @@ + + + +/v1.46/security/access-control-lists.html + + +/v1.46/quickstart/actions-and-hooks.html + + +/v1.46/integrations/airbyte.html + + +/v1.46/howto/hooks/airflow.html + + +/v1.46/integrations/airflow.html + + +/v1.46/reference/api.html + + +/v1.46/understand/architecture.html + + +/v1.46/enterprise/architecture.html + + +/v1.46/integrations/athena.html + + +/v1.46/reference/auditing.html + + +/v1.46/security/authentication.html + + +/v1.46/howto/deploy/aws.html + + +/v1.46/integrations/aws_cli.html + + +/v1.46/howto/deploy/azure.html + + +/v1.46/howto/backup-and-restore.html + + +/v1.46/quickstart/branch.html + + +/v1.46/project/docs/callouts.html + + +/v1.46/howto/catalog_exports.html + + +/v1.46/understand/data_lifecycle_management/ci.html + + +/v1.46/understand/use_cases/cicd_for_data.html + + +/v1.46/reference/cli.html + + +/v1.46/integrations/cloudera.html + + +/v1.46/project/code-migrate-1.0-sdk.html + + +/v1.46/quickstart/commit-and-merge.html + + +/v1.46/enterprise/configuration.html + + +/v1.46/reference/configuration.html + + +/v1.46/project/contributing.html + + +/v1.46/howto/copying.html + + +/v1.46/understand/data_lifecycle_management/data-devenv.html + + +/v1.46/understand/data-structure.html + + +/v1.46/integrations/databricks.html + + +/v1.46/integrations/delta.html + + +/v1.46/posts/deprecate-py-legacy.html + + +/v1.46/integrations/dremio.html + + +/v1.46/integrations/duckdb.html + + +/v1.46/understand/use_cases/etl_testing.html + + +/v1.46/howto/export.html + + +/v1.46/security/external-principals-aws.html + + +/v1.46/understand/faq.html + + +/v1.46/howto/garbage-collection/gc.html + + +/v1.46/howto/deploy/gcp.html + + +/v1.46/integrations/git.html + + +/v1.46/understand/glossary.html + + +/v1.46/integrations/glue_hive_metastore.html + + +/v1.46/integrations/glue_metastore.html + + +/v1.46/integrations/hive.html + + +/v1.46/integrations/huggingface_datasets.html + + +/v1.46/integrations/iceberg.html + + +/v1.46/howto/import.html + + +/v1.46/understand/how/ + + +/v1.46/understand/use_cases/ + + +/v1.46/understand/data_lifecycle_management/ + + +/v1.46/understand/ + + +/v1.46/cloud/ + + +/v1.46/howto/deploy/ + + +/v1.46/howto/garbage-collection/ + + +/v1.46/howto/hooks/ + + +/v1.46/howto/ + + +/v1.46/integrations/ + + +/v1.46/security/ + + +/v1.46/enterprise/getstarted/ + + +/v1.46/enterprise/ + + +/v1.46/posts/ + + +/v1.46/reference/ + + +/v1.46/quickstart/ + + +/v1.46/project/docs/ + + +/v1.46/project/ + + +/v1.46/ + + +/v1.46/enterprise/getstarted/install.html + + +/v1.46/integrations/kafka.html + + +/v1.46/integrations/kubeflow.html + + +/v1.46/understand/how/kv.html + + +/v1.46/experimental/lakectl_local_posix.html + + +/v1.46/quickstart/launch.html + + +/v1.46/quickstart/learning-more-lakefs.html + + +/v1.46/howto/local-checkouts.html + + +/v1.46/howto/hooks/lua.html + + +/v1.46/howto/garbage-collection/managed-gc.html + + +/v1.46/understand/how/merge.html + + +/v1.46/howto/migrate-away.html + + +/v1.46/enterprise/getstarted/migrate-from-oss.html + + +/v1.46/howto/mirroring.html + + +/v1.46/understand/model.html + + +/v1.46/reference/monitor.html + + +/v1.46/reference/mount-csi-driver.html + + +/v1.46/reference/mount.html + + +/v1.46/howto/deploy/onprem.html + + +/v1.46/understand/performance-best-practices.html + + +/v1.46/security/presigned-url.html + + +/v1.46/integrations/presto_trino.html + + +/v1.46/howto/private-link.html + + +/v1.46/understand/data_lifecycle_management/production.html + + +/v1.46/howto/protect-branches.html + + +/v1.46/howto/pull-requests.html + + +/v1.46/integrations/python.html + + +/v1.46/quickstart/query.html + + +/v1.46/enterprise/getstarted/quickstart.html + + +/v1.46/integrations/r.html + + +/v1.46/security/rbac.html + + +/v1.46/integrations/red_hat_openshift_ai.html + + +/v1.46/security/remote-authenticator.html + + +/v1.46/understand/use_cases/reproducibility.html + + +/v1.46/understand/use_cases/rollback.html + + +/v1.46/quickstart/rollback.html + + +/v1.46/reference/s3.html + + +/v1.46/integrations/sagemaker.html + + +/v1.46/howto/scim.html + + +/v1.46/posts/security_update.html + + +/v1.46/howto/sizing-guide.html + + +/v1.46/reference/spark-client.html + + +/v1.46/integrations/spark.html + + +/v1.46/security/sso.html + + +/v1.46/howto/garbage-collection/standalone-gc.html + + +/v1.46/security/sts-login.html + + +/v1.46/enterprise/troubleshooting.html + + +/v1.46/integrations/unity-catalog.html + + +/v1.46/howto/unity-delta-sharing.html + + +/v1.46/howto/deploy/upgrade.html + + +/v1.46/enterprise/upgrade.html + + +/v1.46/understand/how/versioning-internals.html + + +/v1.46/integrations/vertex_ai.html + + +/v1.46/howto/virtual-host-addressing.html + + +/v1.46/howto/hooks/webhooks.html + + +/v1.46/quickstart/work-with-data-locally.html + + diff --git a/v1.46/slack/index.html b/v1.46/slack/index.html new file mode 100644 index 000000000..860ea610f --- /dev/null +++ b/v1.46/slack/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/architecture.html b/v1.46/understand/architecture.html new file mode 100644 index 000000000..75566ed0a --- /dev/null +++ b/v1.46/understand/architecture.html @@ -0,0 +1,969 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Architecture | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Architecture + + +

+ + +

lakeFS is distributed as a single binary encapsulating several logical services.

+ +

The server itself is stateless, meaning you can easily add more instances to handle a bigger load.

+ +

Architecture

+ + +

+ + + Object Storage + + +

+ + +

lakeFS stores data in object stores. Those supported include:

+ +
    +
  • AWS S3
  • +
  • Google Cloud Storage
  • +
  • Azure Blob Storage
  • +
  • MinIO
  • +
  • NetApp StorageGRID
  • +
  • Ceph
  • +
+

+ + + Metadata Storage + + +

+ + +

In additional a Key Value storage is used for storing metadata, with supported databases including PostgreSQL, DynamoDB, and CosmosDB Instructions of how to deploy such database on AWS can be found here.

+ +

Additional information on the data format can be found in Versioning internals and Internal database structure

+

+ + + Load Balancing + + +

+ + +

Accessing lakeFS is done using HTTP. +lakeFS exposes a frontend UI, an OpenAPI server, as well as an S3-compatible service (see S3 Gateway below). +lakeFS uses a single port that serves all three endpoints, so for most use cases a single load balancer pointing +to lakeFS server(s) would do.

+ + +

+ + + lakeFS Components + + +

+ +

+ + + S3 Gateway + + +

+ + +

The S3 Gateway is the layer in lakeFS responsible for the compatibility with S3. It implements a compatible subset of the S3 API to ensure most data systems can use lakeFS as a drop-in replacement for S3.

+ +

See the S3 API Reference section for information on supported API operations.

+

+ + + OpenAPI Server + + +

+ + +

The Swagger (OpenAPI) server exposes the full set of lakeFS operations (see Reference). This includes basic CRUD operations against repositories and objects, as well as versioning related operations such as branching, merging, committing, and reverting changes to data.

+

+ + + Storage Adapter + + +

+ + +

The Storage Adapter is an abstraction layer for communicating with any underlying object store. +Its implementations allow compatibility with many types of underlying storage such as S3, GCS, Azure Blob Storage, or non-production usages such as the local storage adapter.

+ +

See the roadmap for information on the future plans for storage compatibility.

+

+ + + Graveler + + +

+ + +

The Graveler handles lakeFS versioning by translating lakeFS addresses to the actual stored objects. +To learn about the data model used to store lakeFS metadata, see the versioning internals page.

+

+ + + Authentication & Authorization Service + + +

+ + +

The Auth service handles the creation, management, and validation of user credentials and RBAC policies.

+ +

The credential scheme, along with the request signing logic, are compatible with AWS IAM (both SIGv2 and SIGv4).

+ +

Currently, the Auth service manages its own database of users and credentials and doesn’t use IAM in any way.

+

+ + + Hooks Engine + + +

+ + +

The Hooks Engine enables CI/CD for data by triggering user defined Actions that will run during commit/merge.

+

+ + + UI + + +

+ + +

The UI layer is a simple browser-based client that uses the OpenAPI server. It allows management, exploration, and data access to repositories, branches, commits and objects in the system.

+

+ + + Applications + + +

+ + +

As a rule of thumb, lakeFS supports any S3-compatible application. This means that many common data applications work with lakeFS out-of-the-box. +Check out our integrations to learn more.

+

+ + + lakeFS Clients + + +

+ + +

Some data applications benefit from deeper integrations with lakeFS to support different use cases or enhanced functionality provided by lakeFS clients.

+

+ + + OpenAPI Generated SDKs + + +

+ + +

OpenAPI specification can be used to generate lakeFS clients for many programming languages. +For example, the Python lakefs-client or the Java client are published with every new lakeFS release.

+

+ + + lakectl + + +

+ + +

lakectl is a CLI tool that enables lakeFS operations using the lakeFS API from your preferred terminal.

+

+ + + Spark Metadata Client + + +

+ + +

The lakeFS Spark Metadata Client makes it easy to perform +operations related to lakeFS metadata, at scale. Examples include garbage collection or exporting data from lakeFS.

+

+ + + lakeFS Hadoop FileSystem + + +

+ + +

Thanks to the S3 Gateway, it’s possible to interact with lakeFS using Hadoop’s S3AFIleSystem, +but due to limitations of the S3 API, doing so requires reading and writing data objects through the lakeFS server. +Using lakeFSFileSystem increases Spark ETL jobs performance by executing the metadata operations on the lakeFS server, +and all data operations directly through the same underlying object store that lakeFS uses.

+

+ + + How lakeFS Clients and Gateway Handle Metadata and Data Access + + +

+ + +

When using the Python client, lakeCTL, or the lakeFS Spark client, these clients communicate with the lakeFS server to retrieve metadata information. For example, they may query lakeFS to understand which version of a file is needed or to track changes in branches and commits. This communication does not include the actual data transfer, but instead involves passing only metadata about data locations and versions. +Once the client knows the exact data location from the lakeFS metadata, it directly accesses the data in the underlying object storage (potentially using presigned URLs) without routing through lakeFS. For instance, if data is stored in S3, the Spark client will retrieve the S3 paths from lakeFS, then directly read and write to those paths in S3 without involving lakeFS in the data transfer.

+ +

lakeFS Clients vs Gateway Data Flow

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/architecture/data-model.html b/v1.46/understand/architecture/data-model.html new file mode 100644 index 000000000..9c0ed11b5 --- /dev/null +++ b/v1.46/understand/architecture/data-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/branching-model.html b/v1.46/understand/branching-model.html new file mode 100644 index 000000000..8e9879ba3 --- /dev/null +++ b/v1.46/understand/branching-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/data-model.html b/v1.46/understand/data-model.html new file mode 100644 index 000000000..9c0ed11b5 --- /dev/null +++ b/v1.46/understand/data-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/data-structure.html b/v1.46/understand/data-structure.html new file mode 100644 index 000000000..de0c6c4d5 --- /dev/null +++ b/v1.46/understand/data-structure.html @@ -0,0 +1,782 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Data Structure | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + How Does lakeFS Store Your Data + + +

+ +

lakeFS being a data versioning engine, requires the ability to save multiple versions of the same object. As a result, lakeFS stores objects in the object store +in way that allows it to version the data in an efficient way. +This might cause confusion when trying to understand where our data is actually being stored. This page will try to shed a light on this subject.

+

+ + + lakeFS Repository Namespace Structure + + +

+ +

lakeFS stores repository data and metadata under the repository’s namespace. The lakeFS repository namespace is a dedicated path under the object store used by lakeFS to manage a repository. +Listing a repository storage namespace in the object store will provide the following output:

+ +
aws s3 ls s3://<storage_namespace>/
+                        PRE _lakefs/
+                        PRE data/
+
+ +

lakeFS stores the actual user data under the data/ prefix. The _lakefs/ prefix is used to store commit metadata which includes range and meta-range files and internal lakeFS data. +Since lakeFS manages immutable data, objects are not saved using their logical name - these might get overwritten, violating the immutability guarantee. This means that when you upload a csv file called allstar_games_stats.csv to branch main, lakeFS will generate a random physical +address under the data/ prefix and upload it to there.
+Mapping from a path to an object changes as you upload, commit, and merge on lakeFS. When updating an object, lakeFS will create a new physical address for that version preserving other versions of that object. +lakeFS will link between the object’s logical address and its physical address - and store that relation under the given commit metadata (range and meta-range)

+ +

lakeFS uses its object store immutably i.e. anything uploaded is never changed or overridden (Refer to GC for explanation on how and when lakeFS actually deletes data from the storage).
+To find data, lakeFS uses the logical address e.g. lakefs://my-repo/main/allstar_games_stats.csv, indicating a repository and branch. +Using the KV metadata store, lakeFS will first try to find any uncommitted version of the object in the given branch. If no uncommitted version exist, it will take the latest committed version from the branch head (which is the top commit of the branch)

+ +
    +
  1. In the KV metadata store under the current staging token of branch main. This will return any uncommitted changes for the given object
  2. +
  3. Read it from the branch’s head meta-range and range (which are saved under the _lakefs prefix in the object store. This will return the metadata for the object as it was stored in the latest commit for branch main.
    +The physical path returned will be in the form of s3://<storage_namespace>/data/gp0n1l7d77pn0cke6jjg/cg6p50nd77pn0cke6jk0. The same object in lakeFS might have several physical addresses, one for each version where it exists.
  4. +
+

+ + + Finding an object’s location on your object store + + +

+ +

One way to determine the physical location of an object is using the lakectl fs stat command:

+ +
lakectl fs stat --pre-sign=false lakefs://my-repo/main/allstar_games_stats.csv
+Path: allstar_games_stats.csv
+Modified Time: 2024-08-02 10:13:33 -0400 EDT
+Size: 0 bytes
+Human Size: 0 B
+Physical Address: s3://niro-test/repos/docs/data/data/geh1jurck6tfom0s1t8g/cqmej33ck6tfom0s1tvg
+Checksum: d41d8cd98f00b204e9800998ecf8427e
+Content-Type: application/octet-stream
+
+ +

lakeFS can show any version of an object. For example: to see an object’s physical location on branch dev from 3 versions ago, use reference dev~3:

+ +
lakectl fs stat lakefs://my-repo/dev~3/allstar_games_stats.csv
+Path: allstar_games_stats.csv
+Modified Time: 2024-08-02 10:11:49 -0400 EDT
+Size: 916393 bytes
+Human Size: 916.4 kB
+Physical Address: s3://<storage_namespace>/data/data/geh1jurck6tfom0s1t8g/cqmei9bck6tfom0s1tt0
+Checksum: 48e04a4c072acdcf932ee6c43f46ef14
+Content-Type: application/octet-stream
+
+ +

This can be done using any lakeFS reference type.

+ +

To learn more about the internals of lakeFS and how it stores your data, follow this blog post

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/data_lifecycle_management/ci.html b/v1.46/understand/data_lifecycle_management/ci.html new file mode 100644 index 000000000..cf1c436ee --- /dev/null +++ b/v1.46/understand/data_lifecycle_management/ci.html @@ -0,0 +1,765 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +During Deployment | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + During Deployment + + +

+ + +

Every day we introduce new data to the lake. And even if the code and infra doesn’t change, the data might, and those changes introduce potential quality issues. This is one of the complexities of a data product; the data we consume changes over the course of a month, a week, day, hour, or even minute-to-minute.

+ +

Examples of changes to data that may occur:

+
    +
  • A client-side bug in the data collection of website events
  • +
  • A new Android version that interferes with the collecting events from your App
  • +
  • COVID-19 abrupt impact on consumers’ behavior, and its effect on the accuracy of ML models.
  • +
  • During a change to Salesforce interface, the validation requirement from a certain field had been lost
  • +
+ +

lakeFS enable CI/CD-inspired workflows to help validate expectations and assumptions about the data before it goes live in production or lands in the data environment.

+

+ + + Example 1: Data update safety + + +

+ + +

Continuous deployment of existing data we expect to consume, flowing from ingest-pipelines into the lake. We merge data from an ingest branch (“events-data”), which allows us to create tests using data analysis tools or data quality services (e.g. Great Expectations, Monte Carlo) to ensure reliability of the data we merge to the main branch. Since merge is atomic, no performance issue will be introduced by using lakeFS, but your main branch will only include quality data.

+ +

branching_6

+ +

Each merge to the main branch creates a new commit on the main branch, which serves as a new version of the data. This allows us to easily revert to previous states of the data if a newer change introduces data issues.

+

+ + + Example 2: Test - Validate new data + + +

+ + +

Examples of common validation checks enforced in organizations:

+ +
    +
  • No user_* columns except under /private/…
  • +
  • Only (*.parquet | *.orc | _delta_log/*.json) files allowed
  • +
  • Under /production, only backward-compatible schema changes are allowed
  • +
  • New tables on main must be registered in our metadata repository first, with owner and SLA
  • +
+ +

lakeFS will assist in enforcing best practices by giving you a designated branch to ingest new data (“new-data-1” in the drawing). . You may run automated tests to validate predefined best practices as pre-merge hooks. If the validation passes, the new data will be automatically and atomically merged to the main branch. However, if the validation fails, you will be alerted and the new data will not be exposed to consumers.

+ +

By using this branching model and implementing best practices as pre merge hooks, you ensure the main lake is never compromised.

+ +

branching_4

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/data_lifecycle_management/data-devenv.html b/v1.46/understand/data_lifecycle_management/data-devenv.html new file mode 100644 index 000000000..53969c62f --- /dev/null +++ b/v1.46/understand/data_lifecycle_management/data-devenv.html @@ -0,0 +1,788 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +In Test | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + In Test + + +

+ + +

As part of our routine work with data we develop new code, improve and upgrade old code, upgrade infrastructures, and test new technologies. lakeFS enables a safe test environment on your data lake without the need to copy or mock data, work on the pipelines or involve DevOps.

+ +

Creating a branch provides you an isolated environment with a snapshot of your repository (any part of your data lake you chose to manage on lakeFS). While working on your own branch in isolation, all other data users will be looking at the repository’s main branch. They can’t see your changes, and you don’t see changes to main done after you created the branch.

+ +

No worries, no data duplication is done, it’s all metadata management behind the scenes. +Let’s look at 2 examples of a test environment and their branching models.

+

+ + + Example 1: Upgrading Spark and using Reset action + + +

+ + +

You installed the latest version of Apache Spark. As a first step you’ll test your Spark jobs to see that the upgrade doesn’t have any undesired side effects.

+ +

For this purpose, you may create a branch (testing-spark-3.0) which will only be used to test the Spark upgrade, and discarded later. Jobs may run smoothly (the theoretical possibility exists!), or they may fail halfway through, leaving you with some intermediate partitions, data and metadata. In this case, you can simply reset the branch to its original state, without worrying about the intermediate results of your last experiment, and perform another (hopefully successful) test in an isolated branch. Reset actions are atomic and immediate, so no manual cleanup is required.

+ +

Once testing is completed, and you have achieved the desired result, you can delete this experimental branch, and all data not used on any other branch will be deleted with it.

+ +

branching_1

+ +

Creating a testing branch:

+ +
   lakectl branch create \
+      lakefs://example-repo/testing-spark-3 \
+      --source lakefs://example-repo/main
+   # output:
+   # created branch 'testing-spark-3'
+
+ +

Resetting changes to a branch:

+ +
   lakectl branch reset lakefs://example-repo/testing-spark-3
+   # are you sure you want to reset all uncommitted changes?: y█
+
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and branch.

+

+ + + Example 2: Collaborate & Compare - Which option is better? + + +

+ + +

Easily compare by testing which one performs better on your data set. +Examples may be:

+
    +
  • Different computation tools, e.g Spark vs. Presto
  • +
  • Different compression algorithms
  • +
  • Different Spark configurations
  • +
  • Different code versions of an ETL
  • +
+ +

Run each experiment on its own independent branch, while the main remains untouched. Once both experiments are done, create a comparison query (using Hive or Presto or any other tool of your choice) to compare data characteristics, performance or any other metric you see fit.

+ +

With lakeFS you don’t need to worry about creating data paths for the experiments, copying data, and remembering to delete it. It’s substantially easier to avoid errors and maintain a clean lake after.

+ +

branching_2

+ +

Reading from and comparing branches using Spark:

+ +
   val dfExperiment1 = sc.read.parquet("s3a://example-repo/experiment-1/events/by-date")
+   val dfExperiment2 = sc.read.parquet("s3a://example-repo/experiment-2/events/by-date")
+
+   dfExperiment1.groupBy("...").count()
+   dfExperiment2.groupBy("...").count() // now we can compare the properties of the data itself
+
+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/data_lifecycle_management/index.html b/v1.46/understand/data_lifecycle_management/index.html new file mode 100644 index 000000000..27d1926c6 --- /dev/null +++ b/v1.46/understand/data_lifecycle_management/index.html @@ -0,0 +1,721 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Data Lifecycle Management | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Data Lifecycle Management in lakeFS + + +

+ + +

lakeFS provides full support for Data Lifecycle Management through all stages:

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/data_lifecycle_management/production.html b/v1.46/understand/data_lifecycle_management/production.html new file mode 100644 index 000000000..bdbb44778 --- /dev/null +++ b/v1.46/understand/data_lifecycle_management/production.html @@ -0,0 +1,796 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +In Production | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + In Production + + +

+ + +

Errors with data in production inevitably occur. When they do, they best thing we can do is remove the erroneous data, understand why the issue happened, and deploy changes that prevent it from occurring again.

+

+ + + Example 1: RollBack! - Data ingested from a Kafka stream + + +

+ + +

If you introduce a new code version to production and discover it has a critical bug, you can simply roll back to the previous version. +But you also need to roll back the results of running it. +Similar to Git, lakeFS allows you to revert your commits in case they introduced low-quality data. +Revert in lakeFS is an atomic action that prevents the data consumers from receiving low quality data until the issue is resolved.

+ +

As previously mentioned, with lakeFS the recommended branching schema is to ingest data to a dedicated branch. When streaming data, we can decide to merge the incoming data to main at a given time interval or checkpoint, depending on how we chose to write it from Kafka.

+ +

You can run quality tests for each merge (as discussed in the During Deployment section). Alas, tests are not perfect and we might still introduce low quality data to our main branch at some point. +In such a case, we can revert the bad commits from main to the last known high quality commit. This will record new commits reversing the effect of the bad commits.

+ +

branching_7

+ +

Reverting commits using the CLI

+ +
   lakectl branch revert lakefs://example-repo/main 20c30c96 ababea32
+
+ +

Note lakeFS version <= v0.33.1 uses ‘@’ (instead of ‘/’) as separator between repository and branch.

+

+ + + Example 2: Troubleshoot - Reproduce a bug in production + + +

+ + +

You upgraded spark and deployed changes in production. A few days or weeks later, you identify a data quality issue, a performance degradation, or an increase to your infra costs. Something that requires investigation and fixing (aka, a bug).

+ +

lakeFS allows you to open a branch of your lake from the specific merge/commit that introduced the changes to production. Using the metadata saved on the merge/commit you can reproduce all aspects of the environment, then reproduce the issue on the branch and debug it. Meanwhile, you can revert the main to a previous point in time, or keep it as is, depending on the use case

+ +

branching_3

+ +

Reading from a historic version (a previous commit) using Spark

+ +
   // represents the data as existed at commit "11eef40b":
+   spark.read.parquet("s3://example-repo/11eef40b/events/by-date")
+
+

+ + + Example 3: Cross collection consistency + + +

+ + +

We often need consistency between different data collections. A few examples may be:

+
    +
  • To join different collections in order to create a unified view of an account, a user or another entity we measure.
  • +
  • To introduce the same data in different formats
  • +
  • To introduce the same data with a different leading index or sorting due to performance considerations
  • +
+ +

lakeFS will help ensure you introduce only consistent data to your consumers by exposing the new collections and their join in one atomic action to main. Once you consumed the collections on a different branch, and only when both are synchronized, we calculated the join and merged to main.

+ +

In this example you can see two data sets (Sales data and Marketing data) consumed each to its own independent branch, and after the write of both data sets is completed, they are merged to a different branch (leads branch) where the join ETL runs and creates a joined collection by account. The joined table is then merged to main. +The same logic can apply if the data is ingested in streaming, using standard formats, or formats that allow upsert/delete such as Apache Hudi, Delta Lake or Iceberg.

+ +

branching_8

+

+ + + Case Study: Windward + + +

+ +

See how Windward is using lakeFS’ isolation and atomic commits to achieve consistency on top of S3.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/faq.html b/v1.46/understand/faq.html new file mode 100644 index 000000000..399ebf66d --- /dev/null +++ b/v1.46/understand/faq.html @@ -0,0 +1,782 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +FAQ | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Frequently Asked Questions (FAQ) + + +

+ +

+ + + 1. Is lakeFS open-source? + + +

+ +

lakeFS is completely free, open-source, and licensed under the Apache 2.0 License. We maintain a public product roadmap and a Slack channel for open discussions.

+

+ + + 2. How does lakeFS data versioning work? + + +

+ +

lakeFS uses zero-copy branching to avoid data duplication. That is, creating a new branch is a metadata-only operation: no objects are actually copied. Only when an object changes does lakeFS create another version of the data in the storage. For more information, see Versioning internals.

+

+ + + 3. How do I get support for my lakeFS installation? + + +

+ +

We are extremely responsive on our Slack channel, and we make sure to prioritize the most pressing issues for the community. For SLA-based support, please contact us at support@treeverse.io.

+

+ + + 4. Do you collect data from your active installations? + + +

+ +

We collect anonymous usage statistics to understand the patterns of use and to detect product gaps we may have so we can fix them. This is optional and may be turned off by setting stats.enabled to false. See the configuration reference for more details.

+ +

The data we gather is limited to the following:

+
    +
  1. A UUID which is generated when setting up lakeFS for the first time and contains no personal or otherwise identifiable information,
  2. +
  3. The lakeFS version currently running,
  4. +
  5. The OS and architecture lakeFS is running on,
  6. +
  7. Metadata regarding the database used (version, installed extensions and parameters such as DB Timezone and work memory),
  8. +
  9. Periodic aggregated action counters (e.g. how many “get_object” operations occurred).
  10. +
+

+ + + 5. How is lakeFS different from Delta Lake / Hudi / Iceberg? + + +

+ +

Delta Lake, Apache Hudi, and Apache Iceberg all define dedicated, structured data formats that allow deletes and upserts. lakeFS is format-agnostic and enables consistent cross-collection versioning of your data using Git-like operations. Read our comparison for a more detailed comparison.

+ + +

The Axolotl – a species of salamander, also known as the Mexican Lake Monster or the Peter Pan of the animal kingdom. It’s a magical creature, living in a lake - just like us! :)

+ +

Axolotl

+ +

+ copyright +

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/glossary.html b/v1.46/understand/glossary.html new file mode 100644 index 000000000..81f31e5a6 --- /dev/null +++ b/v1.46/understand/glossary.html @@ -0,0 +1,971 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Glossary | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Glossary + + +

+ +

This page has definition and explanations of all terms related to lakeFS technical internals and the architecture.

+ + +

+ + + Auditing + + +

+ +

Data auditing is data assessment to ensure its accuracy, security, and efficacy for specific usage. It also involves assessing data quality through its lifecycle and understanding the impact of poor quality data on the organization’s performance and revenue. Ensuring data reproducibility, auditability, and governance is one of the key concerns of data engineers today. lakeFS commit history helps the data teams to keep track of all changes to the data, supporting data auditing.

+

+ + + Branch + + +

+ + +

Branches in lakeFS allow users to create their own “isolated” view of the repository. Read more.

+

+ + + Collection + + +

+ +

A collection, roughly speaking, is a set of data. Collections may be structured or unstructured; a structured collection is often referred to as a table.

+

+ + + Commit + + +

+ + +

Using commits, you can view a repository at a certain point in its history and you’re guaranteed that the data you see is exactly as it was at the point of committing it. Read More.

+

+ + + Cross-Collection Consistency + + +

+ +

It is unfortunate that the word ‘consistency’ has multiple meanings, at least four of them according to Martin Kleppmann. Consistency in the context of lakeFS and data versioning is, the guarantee that operations in a transaction are performed accurately, correctly and most important, atomically.

+ +

A repository (and thus a branch) in lakeFS, can span multiple tables or collections. By providing branch, commit, merge and revert operations atomically on a branch, lakeFS achieves consistency guarantees across different logical collections. That is, data versioning is consistent across multiple collections within a repository.

+ +

It is sometimes referred as multi-table transactions. That is, lakeFS offers transactional guarantees across multiple tables.

+ + +

+ + + Data Lake Governance + + +

+ +

The goal of data lake governance is to apply policies, standards and processes on the data. This allows creating high-quality data and ensuring that it’s used appropriately across the organization. Data lake governance improves the data quality and increases data usage for business decision-making, leading to operational improvements, better-informed business strategies, and stronger financial performance. lakeFS Cloud offers advanced data lake management features such as: Role-Based Access Control, Branch Aware Managed Garbage Collection, Data Lineage and Audit log.

+

+ + + Data Lifecycle Management + + +

+ +

In data-intensive applications, data should be managed through its entire lifecycle similar to how teams manage code. By doing so, we could leverage the best practices and tools from application lifecycle management (like CI/CD operations) and apply them to data. lakeFS offers data lifecycle management via isolated data development environments instead of shared buckets.

+

+ + + Data Pipeline Reproducibility + + +

+ +

Reproducibility in data pipelines is the ability to repeat a process. An example of this is recreating an issue that occurred in the production pipeline. Reproducibility allows for the controlled manufacture of an error to debug and troubleshoot it at a later point in time. Reproducing a data pipeline issue is a challenge that most data engineers face on a daily basis. Learn more about how lakeFS supports data pipeline reproducibility. Other use cases include running ad-hoc queries (useful for data science), review, and backfill.

+

+ + + Data Quality Testing + + +

+ +

This term describes ways to test data for its accuracy, completeness, consistency, timeliness, validity, and integrity. lakeFS hooks can be used to implement and run data quality tests before promoting staging data into production.

+

+ + + Data Versioning + + +

+ +

To version data means creating a unique point-in-time reference for data that can be accessed later. This reference can take the form of a query, an ID, or also commonly, a DateTime identifier. Data versioning may also include saving an entire copy of the data under a new name or file path every time you want to create a version of it. More advanced versioning solutions like lakeFS perform versioning through zero-copy data operations. lakeFS also optimizes storage usage between versions and exposes special operations to manage them.

+

+ + + Git-like Operations + + +

+ +

lakeFS allows teams to treat their data lake as a Git repository. Git is used for code versioning, whereas lakeFS is used for data versioning. lakeFS provides Git-like operations such as branch, commit, merge and revert.

+

+ + + Graveler + + +

+ +

Graveler is the core versioning engine of lakeFS. It handles versioning by translating lakeFS addresses to the actual stored objects. See the versioning internals section to learn how lakeFS stores metadata.

+

+ + + Hooks + + +

+ +

lakeFS hooks allow you to automate and ensure that a given set of checks and validations happens before important lifecycle events. They are similar conceptually to Git Hooks, but in contrast, they run remotely on a server. Currently, lakeFS allows executing hooks when two types of events occur: pre-commit events that run before a commit is acknowledged and pre-merge events that trigger right before a merge operation.

+

+ + + Isolated Data Snapshot + + +

+ +

Creating a branch in lakeFS provides an isolated environment containing a snapshot of your repository. While working on your branch in isolation, all other data users will be looking at the repository’s main branch. So they won’t see your changes, and you also won’t see the changes applied to the main branch. All of this happens without any data duplication but metadata management.

+

+ + + Main Branch + + +

+ +

Every Git repository has the main branch (unless you take explicit steps to remove it) and it plays a key role in the software development process. In most projects, it represents the source of truth - all the code that works has been tested and is ready to be pushed to production. Similarly, main branch in lakeFS could be used as the single source of truth. For example, the live production data can be on the main branch.

+

+ + + Metadata Management + + +

+ +

Where there is data, there is also metadata. lakeFS uses metadata to define schema, data types, data versions, relations to other datasets, etc. This helps to improve discoverability and manageability. lakeFS performs data versioning through metadata operations.

+

+ + + Merge + + +

+ +

lakeFS merge command, similar to the Git merge functionality, allows you to merge data branches. Once you commit data, you can review it and then merge the committed data into the target branch. A merge generates a commit on the target branch with all your changes. lakeFS guarantees atomic merges that are fast, given they don’t involve copying data. Read More.

+

+ + + Repository + + +

+ + +

In lakeFS, a repository is a set of related objects (or collections of objects). Read More.

+

+ + + Rollback + + +

+ +

A rollback is an atomic operation reversing the effects of a previous commit. If a developer introduces a new code version to production and discovers that it has a critical bug, they can simply roll back to the previous version. In lakeFS, a rollback is an atomic action that prevents the data consumers from receiving low-quality data until the issue is resolved. Learn more about how lakeFS supports the rollback operation.

+

+ + + Storage Namespace + + +

+ +

The storage namespace is a location in the underlying storage dedicated to a specific repository. +lakeFS uses it to store the repository’s objects and some of its metadata.

+

+ + + Underlying Storage + + +

+ +

The underlying storage is a location in some object store where lakeFS keeps your objects and some metadata.

+

+ + + Tag + + +

+ + +

Tags are a way to give a meaningful name to a specific commit. Read More.

+

+ + + Fluffy + + +

+ + +

lakeFS Enterprise Single-Sign-On service, it’s delegated with lakeFS authentication requests and replies back to lakeFS with the authentication response.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/how/index.html b/v1.46/understand/how/index.html new file mode 100644 index 000000000..9c400bd80 --- /dev/null +++ b/v1.46/understand/how/index.html @@ -0,0 +1,723 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +How lakeFS Works | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + How lakeFS Works + + +

+ + +

The Architecture page includes a logical overview of lakeFS and its components.

+ +

For deep-dive content about lakeFS see:

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/how/kv.html b/v1.46/understand/how/kv.html new file mode 100644 index 000000000..16033f20f --- /dev/null +++ b/v1.46/understand/how/kv.html @@ -0,0 +1,808 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Internal database structure | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Internal database structure + + +

+ + + + +

Starting at version 0.80.2, lakeFS abandoned the tight coupling to PostgreSQL and moved all database operations to work over Key-Value Store

+ +

While SQL databases, and Postgres among them, have their obvious advantages, we felt that the tight coupling to Postgres is limiting our users and so, lakeFS with Key Value Store is introduced. +Our KV Store implements a generic interface, with methods for Get, Set, Compare-and-Set, Delete and Scan. Each entry is represented by a [partition, key, value] triplet. All these fields are generic byte-array, and the using module has maximal flexibility on the format to use for each field

+ +

Under the hood, our KV implementation relies on a backing DB, which persists the data. Theoretically, it could be any type of database and out of the box, we already implemented drivers for DynamoDB, for AWS users, and PostgreSQL, using its relational nature to store a KV Store. More databases will be supported in the future, and lakeFS users and contributors can develop their own driver to use their own favorite database. For experimenting purposes, an in-memory KV store can be used, though it obviously lack the persistency aspect

+ +

In order to store its metadata objects (that is Repositories, Branches, Commits, Tags, and Uncommitted Objects), lakeFS implements another layer over the generic KV Store, which supports serialization and deserialization of these objects as protobuf. As this layer relies on the generic interface of the KV Store layer, it is totally agnostic to whichever store implementation is in use, gaining our users the maximal flexibility

+ +

For further reading, please refer to our KV Design

+

+ + + Optimistic Locking with KV + + +

+ + +

One important key difference between SQL databases and Key Value Store is the ability to lock resources. While this is a common practice with relational databases, Key Value stores not always support this ability. When designing our KV Store, we tried to support the most simplistic straight-forward interface, with flexibility in backing DB selection, and so, we decided not to support locking. This decision brought some concurrency challenges we had to overcome. Let us take a look at a common lakeFS flow, Commit, during which several database operations are performed:

+
    +
  • All relevant (Branch correlated) uncommitted objects are collected and marked as committed
  • +
  • A new Commit object is created
  • +
  • The relevant Branch is updated to point to the new commit
  • +
+ +

The Commit flow includes multiple database accesses and modifications, and is very sensitive to concurrent executions: If 2 Commit flows run in parallel, we must guarantee correctness of the data. lakeFS with PostgreSQL simply locks the Branch for the entire Commit operation, preventing concurrent execution of such flows. +Now, with KV Store replacing the SQL database, this easy solution is no longer available. Instead, we implemented an Optimistic Locking algorithm, which leverages the KV Store Compare-And-Set (CAS) functionality to remember the Branch state at the beginning of the Commit flow, and updating the branch at the end, only if it remains unchanged, using CAS, with the former Branch state, used as a comparison criteria. If the sampled Branch state and the current state differ, it could only mean that another, later, Commit is in progress, causing the first Commit to fail, and give the later Commit a chance to complete. +Here’s a running example:

+
    +
  • Commit A sets the StagingToken to tokenA and samples the Branch,
  • +
  • Commit B sets the StagingToken to tokenB and samples the Branch,
  • +
  • Commit A finishes, tries to update the Branch and fails due to the recent modification by Commit B - the StagingToken is set to tokenB and not tokenA as expected by Commit A,
  • +
  • Commit B finishes and updates the branch, as tokenB is set as StagingToken and it matches the flow expectation
  • +
+ +

An important detail to note, is that as a Commit starts, and the StagingToken is set a new value, the former value is added to a list of ‘still valid’ StagingToken_s - _SealedToken - on the Branch, which makes sure no StagingToken and no object are lost due to a failed Commit

+ +

You can read more on the Commit Flow in the dedicated section in the KV Design

+

+ + + DB Transactions and Atomic Updates + + +

+ + +

Another notable difference is the existence of DB transactions with PostgreSQL, ability that our KV Store lacks. This ability was leveraged by lakeFS to construct several DB updates, into one “atomic” operation - each failure, in each step, rolled back the entire operation, keeping the DB consistent and clean. +With KV Store, this ability is gone, and we had to come up with various solutions. As a starting point, the DB consistency is, obviously, not anything we can risk. On the other hand, maintaining the DB clean, and as a result smaller, is something that can be sacrificed, at least as a first step. Let us take a look at a relatively simple flow of a new Repository creation: +A brand new Repository has 3 objects: The Repository object itself, an initial Branch object and an initial Commit, which the Branch points to. With SQL DB, it was as simple as creating all 3 objects in the DB under one transaction (at this order). Any failure resulted in a rollback and no redundant leftovers in our DB. +With no transaction in KV Store, if for example the Branch creation fails, it will leave the Repository without an initial Branch (or a Branch at all), yet the Repository will be accessible. Trying to delete the Repository as a response to Branch creation failure is ony a partial solution as this operation can fail as well. +To mitigate this we introduced a per-_Repository_-partition, which holds all repository related objects (the Branch and Commit in this scenario). The partition key can only be derived from the specific Repository instance itself. In addition we first create the Repository objects, the Commit and the Branch, under the Repository’s partition key, and then the Repository is created. The Repository and its objects will be accessible only after a successful creation of all 3 entities. A failure in this flow might leave some dangling objects, but consistency is maintained. +The number of such dangling objects is not expected to be significant, and we plan to implement a cleaning algorithm to keep our KV Store neat and clean

+

+ + + So, Which Approach is Better? + + +

+ + +

This documents provides a peek into our new database approach - Key Value Store instead of a Relational SQL. It discusses the challenges we faced, and the solutions we provided to overcome these challenges. Considering the fact that lakeFS over with relational database did work, you might ask yourself why did we bother to develop another solution. The simple answer, is that while PostgreSQL was not a bad option, it was the only option, and any drawback of PostgreSQL, reflected on our users:

+
    +
  • PostgreSQL can only scale vertically and that is a limitation. At some point this might not hold.
  • +
  • PostgreSQL is not a managed solution, meaning that users had to take care of all maintenance tasks, including the above mentioned scale (when needed)
  • +
  • As an unmanaged database, scaling means downtime - is that acceptable?
  • +
  • It might even get to the point that your organization is not willing to work with PostgreSQL due to various business considerations
  • +
+ +

If none of the above apply, and you have no seemingly reason to switch from PostgreSQL, it can definitely still be used as an excellent option for the backing database for the lakeFS KV Store. If you do need another solution, you have DynamoDB support, out of the box. DynamoDB, as a fully managed solution, with horizontal scalability support and optimized partitions support, answers all the pain-points specified above. It is definitely an option to consider, if you need to overcome these +And, of course, you can always decide to implement your own KV Store driver to use your database of choice - we would love to add your contribution to lakeFS

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/how/merge.html b/v1.46/understand/how/merge.html new file mode 100644 index 000000000..2ac2405a1 --- /dev/null +++ b/v1.46/understand/how/merge.html @@ -0,0 +1,874 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Merge | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Merges in lakeFS + + +

+ + +

The merge operation in lakeFS is similar to Git. It incorporates changes from a merge source (a commit/reference) into a merge destination (a branch).

+

+ + + How does it work? + + +

+ + +

lakeFS first finds the merge base: the nearest common ancestor of the two commits. +It can now perform a three-way merge, by examining the presence and identity of files in each commit. In the table +below, “A”, “B” and “C” are possible file contents, “X” is a missing file, and “conflict” +(which only appears as a result) is a merge failure.

+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
In baseIn sourceIn destinationResultComment
AAAAUnchanged file
ABBBFiles changed on both sides in same way
ABCconflictFiles changed on both sides differently
AABBFile changed only on one branch
ABABFile changed only on one branch
AXXXFiles deleted on both sides
ABXconflictFile changed on one side, deleted on the other
AXBconflictFile changed on one side, deleted on the other
AAXXFile deleted on one side
AXAXFile deleted on one side
+

+ + + Merge Strategies + + +

+ + +

The API and lakectl allow passing an optional strategy flag with the following values:

+

+ + + source-wins + + +

+ + +

In case of a conflict, merge will pick the source objects.

+

+ + + Example + + +

+ + +
lakectl merge lakefs://example-repo/validated-data lakefs://example-repo/production --strategy source-wins
+
+

When a merge conflict arises, the conflicting objects in the validated-data branch will be chosen to end up in production.

+

+ + + dest-wins + + +

+ + +

In case of a conflict, merge will pick the destination objects.

+

+ + + Example + + +

+ + +
lakectl merge lakefs://example-repo/validated-data lakefs://example-repo/production --strategy dest-wins
+
+

When a merge conflict arises, the conflicting objects in the production branch will be chosen to end up in validated-data. The production branch will not be affected by object changes from validated-data conflicting objects.

+ +

The strategy will affect all conflicting objects in the merge if it is set. Currently it is not possible to treat conflicts individually.

+ +

As a format-agnostic system, lakeFS currently merges by complete files. Format-specific and +other user-defined merge strategies for handling conflicts are on the roadmap.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/how/versioning-internals.html b/v1.46/understand/how/versioning-internals.html new file mode 100644 index 000000000..896e4679f --- /dev/null +++ b/v1.46/understand/how/versioning-internals.html @@ -0,0 +1,848 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Versioning Internals | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Versioning Internals + + +

+ + + +

+ + + Overview + + +

+ + +

Since commits in lakeFS are immutable, they are easy to store on an immutable object store.

+ +

Older commits are rarely accessed, while newer commits are accessed very frequently, a tiered storage approach can work very well - the object store is the source of truth, while local disk and even RAM can be used to cache the more frequently accessed ones.

+ +

Since they are immutable - once cached, you only need to evict them when space is running out. There’s no complex invalidation that needs to happen.

+ +

In terms of storage format, commits are be stored as SSTables, compatible with RocksDB.

+ +

SSTables were chosen as a storage format for 3 major reasons:

+ +
    +
  1. Extremely high read throughput on modern hardware: using commits representing a 200m object repository (modeled after the S3 inventory of one of our design partners), we were able to achieve close to 500k random GetObject calls / second. This provides a very high throughput/cost ratio, probably as high as can be achieved on public clouds.
  2. +
  3. Being a known storage format means it’s relatively easy to generate and consume. Storing it in the object store makes it accessible to data engineering tools for analysis and distributed computation, effectively reducing the silo effect of storing it in an operational database.
  4. +
  5. The SSTable format supports delta encoding for keys which makes them very space efficient for data lakes where many keys share the same common prefixes.
  6. +
+ +

Each lakeFS commit is represented as a set of contiguous, non-overlapping SSTables that make up the entire keyspace of a repository at that commit.

+

+ + + SSTable File Format (“Graveler File”) + + +

+ + +

lakeFS metadata is encoded into a format called “Graveler” - a standardized way to encode content-addressable key value pairs. This is what a Graveler file looks like:

+ +

Graveler File

+ +

Each Key/Value pair (“ValueRecord”) is constructed of a key, identity, and value.

+ +

A simple identity could be, for example, a sha256 hash of the value’s bytes. It could be any sequence of bytes that uniquely identifies the value. As far as the Graveler is concerned, two ValueRecords are considered identical if their key and identity fields are equal.

+ +

A Graveler file itself is content-addressable, i.e., similarly to Git, the name of the file is its identity. +File identity is calculated based on the identity of the ValueRecords the file contains:

+ +

valueRecordID = h(h(valueRecord.key) || h(valueRecord.Identity))
+fileID = h(valueRecordID1 + … + valueRecordIDN)

+

+ + + Constructing a consistent view of the keyspace (i.e., a commit) + + +

+ + +

We have two additional requirements for the storage format:

+ +
    +
  1. Be space and time efficient when creating a commit - assuming a commit changes a single object out of a billion, we don’t want to write a full snapshot of the entire repository. Ideally, we’ll be able to reuse some data files that haven’t changed to make the commit operations (in both space and time) proportional to the size of the difference as opposed to the total size of the repository.
  2. +
  3. Allow an efficient diff between commits which runs in time proportional to the size of their difference and not their absolute sizes.
  4. +
+ +

To support these requirements, we decided to essentially build a 2-layer Merkle tree composed of a set of leaf nodes (“Range”) addressed by their content address, and a “Meta Range”, which is a special range containing all ranges, thus representing an entire consistent view of the keyspace:

+ +

Metarange to ranges relationship

+ +

Assuming commit B is derived from commit A, and only changed files in range e-f, it can reuse all ranges except for SSTable #N (the one containing the modified range of keys), which will be recreated with a new hash representing the state as exists after applying commit B’s changes. This will, in turn, also create a new Metarange since its hash is now changed as well (as it is derived from the hash of all contained ranges).

+ +

Assuming most commits usually change related objects (i.e., that are likely to share some common prefix), the reuse ratio could be very high. We tested this assumption using S3 inventory from 2 design partners - we partitioned the keyspace to an arbitrary number of simulated blocks and measured their change over time. We saw a daily change rate of about 5-20%.

+ +

Given the size of the repositories, it’s safe to assume that a single day would translate into multiple commits. At a modest 20 commits per day, a commit is expected to reuse >= 99% of the previous commit blocks, so acceptable in terms of write amplification generated on commit.

+ +

On the object store, ranges are stored in the following hierarchy:

+ +
<lakefs root>
+    _lakefs/
+        <range hash1>
+        <range hash2>
+        <range hashN>
+        ...
+        <metarange hash1>
+        <metarange hash2>
+        <metarange hashN>
+        ...
+    <data object hash1>
+    <data object hash2>
+    <data object hashN>
+    ...
+
+ +

Note: This relatively flat structure could be modified in the future. Looking at the diagram above, it imposes no real limitations on the depth of the tree. A tree could easily be made recursive by having Meta Ranges point to other Meta Ranges - and still provide all the same characteristics. For simplicity, we decided to start with a fixed 2-level hierarchy.

+

+ + + Representing references and uncommitted metadata + + +

+ + +

lakeFS always stores the object data in the storage namespace in the user’s object store, committed and uncommitted data alike.

+ +

However, the lakeFS object metadata might be stored in either the object store or a key-value store.

+ +

Unlike committed metadata which is immutable, uncommitted (or “staged”) metadata experiences frequent random writes and is very mutable in nature. This is also true for “refs” - in particular, branches, which are simply pointers to an underlying commit, are modified frequently: on every commit or merge operation.

+ +

Both these types of metadata are not only mutable, but also require strong consistency guarantees while also being fault tolerant. If we can’t access the current pointer of the main branch, a big portion of the system is essentially down.

+ +

Luckily, this is also much smaller set of metadata compared to the committed metadata.

+ +

References and uncommitted metadata are currently stored on a key-value store (See supported databases) for consistency guarantees.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/images/1.0-is-coming.jpeg b/v1.46/understand/images/1.0-is-coming.jpeg new file mode 100644 index 000000000..4f7e56f3b Binary files /dev/null and b/v1.46/understand/images/1.0-is-coming.jpeg differ diff --git a/v1.46/understand/images/1.0-is-coming.jpg b/v1.46/understand/images/1.0-is-coming.jpg new file mode 100644 index 000000000..18e32ba60 Binary files /dev/null and b/v1.46/understand/images/1.0-is-coming.jpg differ diff --git a/v1.46/understand/index.html b/v1.46/understand/index.html new file mode 100644 index 000000000..02d176cbb --- /dev/null +++ b/v1.46/understand/index.html @@ -0,0 +1,778 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Understanding lakeFS | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + +
+ +

+ + + Understanding lakeFS + + +

+ + +

lakeFS Docs

+

+ + + Architecture and Internals + + +

+ + +

The Architecture page includes a logical overview of lakeFS and its components.

+ +

For deep-dive content about lakeFS see:

+ + +

+ + + lakeFS Use Cases + + +

+ + +

lakeFS has many uses in the data world, including

+ + + +

One of the important things that lakeFS provides is full support for Data Lifecycle Management through all stages:

+ + +

+ + + lakeFS Concepts and Model + + +

+ + +

lakeFS adopts many of the terms and concepts from git. This page goes into details on the similarities and differences, and provides a good background to the concepts used in lakeFS.

+

+ + + Performance + + +

+ + +

Check out the Performance best practices guide for useful hints and tips on ensuring high performance from lakeFS.

+

+ + + FAQ and Glossary + + +

+ + +

The FAQ covers many common questions around lakeFS, and the glossary provides a useful reference for the terms used in lakeFS.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/kv-in-a-nutshell.html b/v1.46/understand/kv-in-a-nutshell.html new file mode 100644 index 000000000..9f3bb5688 --- /dev/null +++ b/v1.46/understand/kv-in-a-nutshell.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/licensing.html b/v1.46/understand/licensing.html new file mode 100644 index 000000000..41dce22e2 --- /dev/null +++ b/v1.46/understand/licensing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/merge.html b/v1.46/understand/merge.html new file mode 100644 index 000000000..9a856dde2 --- /dev/null +++ b/v1.46/understand/merge.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/model.html b/v1.46/understand/model.html new file mode 100644 index 000000000..aaec0a4c1 --- /dev/null +++ b/v1.46/understand/model.html @@ -0,0 +1,940 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Concepts and Model | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Concepts and Model + + +

+ + +
+

+ + + Table of contents + + +

+ + +
    +
  1. Objects
  2. +
  3. Version Control
      +
    1. Repository
    2. +
    3. Commits
    4. +
    5. Branches
    6. +
    7. Tags
    8. +
    9. History
    10. +
    11. Merge
    12. +
    13. Ref expressions
    14. +
    +
  4. +
  5. Concepts unique to lakeFS
      +
    1. lakefs protocol URIs
    2. +
    +
  6. +
+ +
+ +

lakeFS blends concepts from object stores such as S3 with concepts from Git. This reference +defines the common concepts of lakeFS.

+

+ + + Objects + + +

+ + +

lakeFS is an interface to manage objects in an object store.

+ +
+

The actual data itself is not stored inside lakeFS directly but in an underlying object store. +lakeFS manages pointers and additional metadata about these objects.

+
+

+ + + Version Control + + +

+ + +

lakeFS is spearheading version control semantics for data. Most of these concepts will be familiar to Git users:

+

+ + + Repository + + +

+ + +

In lakeFS, a repository is a set of related objects (or collections of objects). In many cases, these represent tables of various formats for tabular data, semi-structured data such as JSON or log files - or a set of unstructured objects such as images, videos, sensor data, etc.

+ +

lakeFS represents repositories as a logical namespace used to group together objects, branches, and commits - analogous to a repository in Git.

+ +

lakeFS repository naming requirements are as follows:

+ +
    +
  • Start with a lower case letter or number
  • +
  • Contain only lower case letters, numbers and hyphens
  • +
  • Be between 3 and 63 characters long
  • +
+

+ + + Commits + + +

+ + +

Using commits, you can view a repository at a certain point in its history and you’re guaranteed that the data you see is exactly as it was at the point of committing it.

+ +

These commits are immutable “checkpoints” containing all contents of a repository at a given point in the repository’s history.

+ +

Each commit contains metadata - the committer, timestamp, a commit message, as well as arbitrary key/value pairs you can choose to add.

+ +

Identifying Commits

+ A commit is identified by its commit ID, a digest of all contents of the commit.
+ Commit IDs are by nature long, so you may use a unique prefix to abbreviate them. A commit may also be identified by using a textual definition, called a ref.

+ Examples of refs include tags, branch names, and expressions.

+

+ + + Branches + + +

+ + +

Branches in lakeFS allow users to create their own “isolated” view of the repository.

+ +

Changes on one branch do not appear on other branches. Users can take changes from one branch and apply it to another by merging them.

+

+ + + Zero-copy branching + + +

+ + +

Under the hood, branches are simply a pointer to a commit along with a set of uncommitted changes. +Creating a branch is a zero-copy operation; instead of duplicating data, it involves creating a pointer to the source commit for the branch.

+

+ + + Tags + + +

+ + +

Tags are a way to give a meaningful name to a specific commit. +Using tags allow users to reference specific releases, experiments, or versions by using a human friendly name.

+ +

Example tags:

+ +
    +
  • v2.3 to mark a release.
  • +
  • dev-jane-before-v2.3-merge to mark Jane’s private temporary point.
  • +
+ +

Tag names adhere to the same rules as git ref names.

+

+ + + History + + +

+ + +

The history of the branch is the list of commits from the branch tip through the first +parent of each commit. Histories go back in time.

+

+ + + Merge + + +

+ + +

Merging is the way to integrate changes from a branch into another branch. +The result of a merge is a new commit, with the destination as the first parent and the source as the second.

+ +

To learn more about how merging works in lakeFS, see the merge reference

+

+ + + Ref expressions + + +

+ + +

lakeFS also supports expressions for creating a ref. These are similar to revisions in +Git; indeed all ~ and ^ +examples at the end of that section will work unchanged in lakeFS.

+ +
    +
  • A branch or a tag are ref expressions.
  • +
  • If <ref> is a ref expression, then: +
      +
    • <ref>^ is a ref expression referring to its first parent.
    • +
    • <ref>^N is a ref expression referring to its N’th parent; in particular <ref>^1 is the +same as <ref>^.
    • +
    • <ref>~ is a ref expression referring to its first parent; in particular <ref>~ is the +same as <ref>^ and <ref>~.
    • +
    • <ref>~N is a ref expression referring to its N’th parent, always traversing to the first +parent. So <ref>~N is the same as <ref>^^...^ with N consecutive carets ^.
    • +
    +
  • +
+

+ + + Concepts unique to lakeFS + + +

+ + +

The underlying storage is a location in an object store where lakeFS keeps your objects and some immutable metadata.

+ +

When creating a lakeFS repository, you assign it with a storage namespace. The repository’s +storage namespace is a location in the underlying storage where data for this repository +will be stored.

+ +

We sometimes refer to underlying storage as physical. The path used to store the contents of an object is then termed a physical path. +Once lakeFS saves an object in the underlying storage it is never modified, except to remove it +entirely during some cleanups.

+ +

A lot of what lakeFS does is to manage how lakeFS paths translate to physical paths on the +object store. This mapping is generally not straightforward. Importantly (and contrary to +many object stores), lakeFS may map multiple paths to the same object on backing storage, and +always does this for objects that are unchanged across versions.

+

+ + + lakefs protocol URIs + + +

+ + +

lakeFS uses a specific format for path URIs. The URI lakefs://<REPO>/<REF>/<KEY> is a path +to objects in the given repo and ref expression under key. This is used both for path +prefixes and for full paths. In similar fashion, lakefs://<REPO>/<REF> identifies the +repository at a ref expression, and lakefs://<REPO> identifes a repo.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/object-model.html b/v1.46/understand/object-model.html new file mode 100644 index 000000000..8e9879ba3 --- /dev/null +++ b/v1.46/understand/object-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/performance-best-practices.html b/v1.46/understand/performance-best-practices.html new file mode 100644 index 000000000..67542d6ca --- /dev/null +++ b/v1.46/understand/performance-best-practices.html @@ -0,0 +1,819 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Performance Best Practices | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Performance Best Practices + + +

+ + + +

+ + + Overview + + +

+ +

Use this guide to achieve the best performance with lakeFS.

+

+ + + Avoid concurrent commits/merges + + +

+ +

Just like in Git, branch history is composed by commits and is linear by nature. +Concurrent commits/merges on the same branch result in a race. The first operation will finish successfully while the rest will retry.

+

+ + + Perform meaningful commits + + +

+ +

It’s a good idea to perform commits that are meaningful in the senese that they represent a logical point in your data’s lifecycle. While lakeFS supports arbirartily large commits, avoiding commits with a huge number of objects will result in a more comprehensible commit history.

+

+ + + Use zero-copy import + + +

+ +

To import object into lakeFS, either a single time or regularly, lakeFS offers a zero-copy import feature. +Use this feature to import a large number of objects to lakeFS, instead of simply copying them into your repository. +This feature will create a reference to the existing objects on your bucket and avoids the copy.

+

+ + + Read data using the commit ID + + +

+ +

In cases where you are only interested in reading committed data:

+
    +
  • Use a commit ID (or a tag ID) in your path (e.g: lakefs://repo/a1b2c3).
  • +
  • Add @ before the path lakefs://repo/main@/path.
  • +
+ +

When accessing data using the branch name (e.g. lakefs://repo/main/path) lakeFS will also try to fetch uncommitted data, which may result in reduced performance. +For more information, see how uncommitted data is managed in lakeFS.

+

+ + + Operate directly on the storage + + +

+ +

Sometimes, storage operations can become a bottleneck. For example, when your data pipelines upload many big objects. +In such cases, it can be beneficial to perform only versioning operations on lakeFS, while performing storage reads/writes directly on the object store. +lakeFS offers multiple ways to do that:

+ + +

Accessing the object store directly is a faster way to interact with your data.

+

+ + + Zero-copy + + +

+ +

lakeFS provides a zero-copy mechanism to data. Instead of copying the data, we can check out to a new branch. +Creating a new branch will take constant time as the new branch points to the same data as its parent. +It will also lower the storage cost.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/roadmap.html b/v1.46/understand/roadmap.html new file mode 100644 index 000000000..41dce22e2 --- /dev/null +++ b/v1.46/understand/roadmap.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/sizing-guide.html b/v1.46/understand/sizing-guide.html new file mode 100644 index 000000000..252d1aa93 --- /dev/null +++ b/v1.46/understand/sizing-guide.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/understand/data-model.html b/v1.46/understand/understand/data-model.html new file mode 100644 index 000000000..9c0ed11b5 --- /dev/null +++ b/v1.46/understand/understand/data-model.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/understand/use_cases/cicd_for_data.html b/v1.46/understand/use_cases/cicd_for_data.html new file mode 100644 index 000000000..c0d4cd92a --- /dev/null +++ b/v1.46/understand/use_cases/cicd_for_data.html @@ -0,0 +1,875 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +CI/CD for Data Lakes | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + CI/CD for Data + + +

+ + + +

+ + + Why do I need CI/CD? + + +

+ + +

Data pipelines feed processed data from data lakes to downstream consumers like business dashboards and machine learning models. As more and more organizations rely on data to enable business critical decisions, data reliability and trust are of paramount concern. Thus, it’s important to ensure that production data adheres to the data governance policies of businesses. These data governance requirements can be as simple as a file format validation, schema check, or an exhaustive PII(Personally Identifiable Information) data removal from all of organization’s data.

+ +

Thus, to ensure the quality and reliability at each stage of the data lifecycle, data quality gates need to be implemented. That is, we need to run Continuous Integration(CI) tests on the data, and only if data governance requirements are met can the data can be promoted to production for business use.

+ +

Everytime there is an update to production data, the best practice would be to run CI tests and then promote(deploy) the data to production.

+

+ + + How do I implement CI/CD for data with lakeFS? + + +

+ + +

lakeFS makes implementing CI/CD pipelines for data simpler. lakeFS provides a feature called hooks that allow automation of checks and validations of data on lakeFS branches. These checks can be triggered by certain data operations like committing, merging, etc.

+ +

Functionally, lakeFS hooks are similar to Git Hooks. lakeFS hooks are run remotely on a server, and they are guaranteed to run when the appropriate event is triggered.

+ +

Here are some examples of the hooks lakeFS supports:

+
    +
  • pre-merge
  • +
  • pre-commit
  • +
  • post-merge
  • +
  • post-commit
  • +
  • pre-create-branch
  • +
  • post-create-branch
  • +
+ +

and so on.

+ +

By leveraging the pre-commit and pre-merge hooks with lakeFS, you can implement CI/CD pipelines on your data lakes.

+ +

Specific trigger rules, quality checks and the branch on which the rules are to be applied are declared in actions.yaml file. When a specific event (say, pre-merge) occurs, lakeFS runs all the validations declared in actions.yaml file. If validations error out, the merge event is blocked.

+ +

Here is a sample actions.yaml file that has pre-merge hook configured to allow only parquet and delta lake file formats on main branch.

+ +
name: ParquetOnlyInProduction
+description: This webhook ensures that only parquet files are written under production/
+on:
+  pre-merge:
+    branches:
+      - main
+hooks:
+  - id: production_format_validator
+    type: webhook
+    description: Validate file formats
+    properties:
+      url: "http://lakefs-hooks:5001/webhooks/format"
+      query_params:
+        allow: ["parquet", "delta_lake"]
+        prefix: analytics/
+
+

+ + + Using hooks as data quality gates + + +

+ + +

Hooks are run on a remote server that can serve http requests from lakeFS server. lakeFS supports two types of hooks.

+
    +
  1. webhooks (run remotely on a web server. e.g.: flask server in python)
  2. +
  3. airflow hooks (a dag of complex data quality checks/tasks that can be run on airflow server)
  4. +
+ +

In this tutorial, we will show how to use webhooks (python flask webserver) to implement quality gates on your data branches. Specifically, how to configure hooks to allow only parquet and delta lake format files in the main branch.

+ +

The tutorial provides a lakeFS environment, python flask server, a Jupyter notebook and sample data sets to demonstrate the integration of lakeFS hooks with Apache Spark and Python. It runs on Docker Compose.

+ +

To understand how hooks work and how to configure hooks in your production system, refer to the documentation: Hooks.

+ +

lakeFS hooks - Promotion workflow

+ +

Follow the steps below to try out CI/CD for data lakes.

+

+ + + Implementing CI/CD pipeline with lakeFS - Demo + + +

+ + +

The sample below provides a lakeFS environment, a Jupyter notebook, and a server on which for the lakeFS webhooks to run.

+

+ + + Prerequisites & Setup + + +

+ + +

Before we get started, make sure Docker is installed on your machine.

+ +
    +
  • +

    Start by cloning the lakeFS samples Git repository:

    + +
    git clone https://github.com/treeverse/lakeFS-samples.git
    +
    + +
    cd lakeFS-samples
    +
    +
  • +
  • +

    Run following commands to start the components:

    + +
    git submodule init
    +git submodule update
    +docker compose up
    +
    +
  • +
+ +

Open the local Jupyter Notebook and go to the hooks-demo.ipynb notebook.

+

+ + + Resources + + +

+ + +

To explore different checks and validations on your data, refer to pre-built hooks config by the lakeFS team.

+ +

To understand the comprehensive list of hooks supported by lakeFS, refer to the documentation.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/use_cases/etl_testing.html b/v1.46/understand/use_cases/etl_testing.html new file mode 100644 index 000000000..3aae942ef --- /dev/null +++ b/v1.46/understand/use_cases/etl_testing.html @@ -0,0 +1,998 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +ETL Testing Environment | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + ETL Testing with Isolated Dev/Test Environments + + +

+ + + +

+ + + Why are multiple environments so important? + + +

+ + +

When working with a data lake, it’s useful to have replicas of your production environment. These replicas allow you to test these ETLs and understand changes to your data without impacting the consumers of the production data.

+ +

Running ETL and transformation jobs directly in production without proper ETL testing presents a huge risk of having data issues flow into dashboards, ML models, and other consumers sooner or later.

+ +

The most common approach to avoid making changes directly in production is to create and maintain multiple data environments and perform ETL testing on them. Dev environments give you a space in which to develop the data pipelines and test environment where pipeline changes are tested before pushing it to production.

+ +

Without lakeFS, the challenge with this approach is that it can be time-consuming and costly to maintain these separate dev/test environments to enable thorough effective ETL testing. And for larger teams it forces multiple people to share these environments, requiring significant coordination. Depending on the size of the data involved there can also be high costs due to the duplication of data.

+

+ + + How does lakeFS help with Dev/Test environments? + + +

+ + +

lakeFS makes creating isolated dev/test environments for ETL testing quick and cheap. lakeFS uses zero-copy branching which means that there is no duplication of data when you create a new environment. This frees you from spending time on environment maintenance and makes it possible to create as many environments as needed.

+ +

In a lakeFS repository, data is always located on a branch. You can think of each branch in lakeFS as its own environment. This is because branches are isolated, meaning changes on one branch have no effect other branches.

+ +

Objects that remain unchanged between two branches are not copied, but rather shared to both branches via metadata pointers that lakeFS manages. If you make a change on one branch and want it reflected on another, you can perform a merge operation to update one branch with the changes from another.

+

+ + + Using branches as development and testing environments + + +

+ + +

The key difference when using lakeFS for isolated data environments is that you can create them immediately before testing a change. And once new data is merged into production, you can delete the branch - effectively deleting the old environment.

+ +

This is different from creating a long-living test environment used as a staging area to test all the updates. With lakeFS, we create a new branch for each change to production that we want to make. One benefit of this is the ability to test multiple changes at one time.

+ +

dev/test branches as environments

+

+ + + Try it out! Creating Dev/Test Environments with lakeFS for ETL Testing + + +

+ + +

lakeFS supports UI, CLI (lakectl commandline utility) and several clients for the API to run the Git-like operations. Let us explore how to create dev/test environments using each of these options below.

+ +

There are two ways that you can try out lakeFS:

+ +
    +
  • The lakeFS Playground on lakeFS Cloud - fully managed lakeFS with a 30-day free trial
  • +
  • Local Docker-based quickstart and samples
  • +
+ +

You can also deploy lakeFS locally or self-managed on your cloud of choice.

+

+ + + Using lakeFS Playground on lakeFS Cloud + + +

+ + +

In this tutorial, we will use a lakeFS playground environment to create dev/test data environments for ETL testing. This allows you to spin up a lakeFS instance in a click, create different data environments by simply branching out of your data repository and develop & test data pipelines in these isolated branches.

+ +

First, let us spin up a playground instance. Once you have a live environment, login to your instance with access and secret keys. Then, you can work with the sample data repository my-repo that is created for you.

+ +

sample repository

+ +

Click on my-repo and notice that by default, the repo has a main branch created and sample_data preloaded to work with.

+ +

main branch

+ +

You can create a new branch (say, test-env) by going to the Branches tab and clicking Create Branch. Once it is successful, you will see two branches under the repo: main and test-env.

+ +

test-env branch

+ +

Now you can add, modify or delete objects under the test-env branch without affecting the data in the main branch.

+

+ + + Trying out lakeFS with Docker and Jupyter Notebooks + + +

+ + +

This use case shows how to create dev/test data environments for ETL testing using lakeFS branches. The following tutorial provides a lakeFS environment, a Jupyter notebook, and Python lakefs_client API to demonstrate integration of lakeFS with Spark. You can run this tutorial on your local machine.

+ +

Follow the tutorial video below to get started with the playground and Jupyter notebook, or follow the instructions on this page.

+ +

+ + + Prerequisites + + +

+ + +

Before getting started, you will need Docker installed on your machine.

+

+ + + Running lakeFS and Jupyter Notebooks + + +

+ + +

Follow along the steps below to create dev/test environment with lakeFS.

+ +
    +
  • +

    Start by cloning the lakeFS samples Git repository:

    + +
    git clone https://github.com/treeverse/lakeFS-samples.git
    +
    + +
    cd lakeFS-samples
    +
    +
  • +
  • +

    Run following commands to download and run Docker container which includes Python, Spark, Jupyter Notebook, JDK, Hadoop binaries, lakeFS Python client and Airflow (Docker image size is around 4.5GB):

    + +
    git submodule init
    +git submodule update
    +docker compose up
    +
    +
  • +
+ +

Open the local Jupyter Notebook and go to the spark-demo.ipynb notebook.

+

+ + + Configuring lakeFS Python Client + + +

+ + +

Setup lakeFS access credentials for the lakeFS instance running. The defaults for these that the samples repo Docker Compose uses are shown here:

+ +
lakefsAccessKey = 'AKIAIOSFODNN7EXAMPLE'
+lakefsSecretKey = 'wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY'
+lakefsEndPoint = 'http://lakefs:8000'
+
+ +

Next, setup the storage namespace to a location in the bucket you have configured. The storage namespace is a location in the underlying storage where data for this repository will be stored.

+ +
storageNamespace = 's3://example/' 
+
+ +

You can use lakeFS through the UI, API or lakectl commandline. For this use-case, we use python lakefs_client to run lakeFS core operations.

+ +
import lakefs_client
+from lakefs_client import models
+from lakefs_client.client import LakeFSClient
+
+# lakeFS credentials and endpoint
+configuration = lakefs_client.Configuration()
+configuration.username = lakefsAccessKey
+configuration.password = lakefsSecretKey
+configuration.host = lakefsEndPoint
+
+client = LakeFSClient(configuration)
+
+ +

lakeFS can be configured to work with Spark in two ways:

+ +

+ + + Upload the Sample Data to Main Branch + + +

+ + +

To upload an object to the my-repo, use the following command.

+ +
import os
+contentToUpload = open('/data/lakefs_test.csv', 'rb')
+client.objects.upload_object(
+    repository="my-repo",
+    branch="main",
+    path=fileName, content=contentToUpload)
+
+ +

Once uploaded, commit the changes to the main branch and attach some metadata to the commit as well.

+
client.commits.commit(
+    repository="my-repo",
+    branch="main",
+    commit_creation=models.CommitCreation(
+        message='Added my first object!',
+        metadata={'using': 'python_api'}))
+
+ +

In this example, we use lakeFS S3A gateway to read data from the storage bucket.

+
dataPath = f"s3a://my-repo/main/lakefs_test.csv"
+df = spark.read.csv(dataPath)
+df.show()
+
+

+ + + Create a Test Branch + + +

+ + +

Let us start by creating a new branch test-env on the example repo my-repo.

+ +
client.branches.create_branch(
+    repository="my-repo",
+    branch_creation=models.BranchCreation(
+        name="test-env",
+        source="main"))
+
+ +

Now we can use Spark to write the csv file from main branch as a Parquet file to the test-env of our lakeFS repo. Suppose we accidentally write the dataframe back to “test-env” branch again, this time in append mode.

+ +
df.write.mode('overwrite').parquet('s3a://my-repo/test-env/')
+df.write.mode('append').parquet('s3a://my-repo/test-env/')
+
+

What happens if we re-read in the data on both branches and perform a count on the resulting DataFrames? +There will be twice as many rows in test-env branch. That is, we accidentally duplicated our data! Oh no!

+ +

Data duplication introduce errors into our data analytics, BI and machine learning efforts; hence we would like to avoid duplicating our data.

+ +

On the main branch however, there is still just the original data - untouched by our Spark code. This shows the utility of branch-based isolated environments with lakeFS.

+ +

You can safely continue working with the data from main which is unharmed due to lakeFS isolation capabilities.

+

+ + + Further Reading + + +

+ + + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/use_cases/index.html b/v1.46/understand/use_cases/index.html new file mode 100644 index 000000000..f215577a1 --- /dev/null +++ b/v1.46/understand/use_cases/index.html @@ -0,0 +1,730 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Use Cases | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + lakeFS Use Cases + + +

+ + +

lakeFS has many uses in the data world, including

+ + + +

One of the important things that lakeFS provides is full support for Data Lifecycle Management through all stages:

+ + + + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/use_cases/reproducibility.html b/v1.46/understand/use_cases/reproducibility.html new file mode 100644 index 000000000..d5e04c5d1 --- /dev/null +++ b/v1.46/understand/use_cases/reproducibility.html @@ -0,0 +1,756 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Reproducibility | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + The Benefits of Reproducible Data + + +

+ + +

Data changes frequently. This makes the task of keeping track of its exact state over time difficult. Oftentimes, people maintain only one state of their data––its current state.

+ +

This has a negative impact on the work, as it becomes hard to:

+
    +
  • Debug a data issue.
  • +
  • Validate machine learning training accuracy (re-running a model over different data gives different results).
  • +
  • Comply with data audits.
  • +
+ +

In comparison, lakeFS exposes a Git-like interface to data that allows keeping track of more than just the current state of data. This makes reproducing its state at any point in time straightforward.

+

+ + + Achieving Reproducibility with lakeFS + + +

+ + +

To make data reproducible, we recommend taking a new commit of your lakeFS repository every time the data in it changes. As long as there’s a commit taken, the process to reproduce a given state is as simple as reading the data from a path that includes the unique commit_id generated for each commit.

+ +

To read data at it’s current state, we can use a static path containing the repository and branch names. To give an example, if you have a repository named example with a branch named main, reading the latest state of this data into a Spark Dataframe is always:

+ +
df = spark.read.parquet(‘s3://example/main/”)
+
+

Note: The code above assumes that all objects in the repository under this path are stored in parquet format. If a different format is used, the applicable Spark read method should be used.

+ +

In a lakeFS repository, we are capable of taking many commits over the data, making many points in time reproducible.

+ +

Commit History

+ +

In the repository above, a new commit is taken each time a model training script is run, and the commit message includes the specific run number.

+ +

If we wanted to re-run the model training script and reproduce the exact same results for a historical run, say run #435, we could copy the commit ID associated with the run and read the data into a dataframe like so:

+ +
df = spark.read.parquet("s3://example/296e54fbee5e176f3f4f4aeb7e087f9d57515750e8c3d033b8b841778613cb23/training_dataset/”)
+
+ +

The ability to reference a specific commit_id in code simplifies reproducing the specific state a data collection or even multiple collections. This has many applications that are common in data development, such as historical debugging, identifying deltas in a data collection, audit compliance, and more.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/use_cases/rollback.html b/v1.46/understand/use_cases/rollback.html new file mode 100644 index 000000000..87b2e07f2 --- /dev/null +++ b/v1.46/understand/use_cases/rollback.html @@ -0,0 +1,789 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Rollback | lakeFS Documentation + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Link + + + + + + + Search + + + + + + + Menu + + + + + + + + + + + + Expand + + + + + + Document + + + + + + +
+ +
+ +
+ + + +
+
+ + + + + +
+ +

+ + + Rollbacks + + +

+ +

+ + + What Is a Rollback? + + +

+ + +

A rollback operation is used to to fix critical data errors immediately.

+ +

What is a critical data error? Think of a situation where erroneous or misformatted data causes a significant issue with an important service or function. In such situations, the first thing to do is stop the bleeding.

+ +

Rolling back returns data to a state in the past, before the error was present. You might not be showing all the latest data after a rollback, but at least you aren’t showing incorrect data or raising errors.

+

+ + + Why Rollbacks Are Useful + + +

+ + +

A Rollback is used as a stopgap measure to “put out the fire” as quickly as possible while RCA (root cause analysis) is performed to understand 1) exactly how the error happened, and 2) what can be done to prevent it from happening again.

+ +

It can be a pressured, stressful situation to deal with a critical data error. Having the ability to employ a rollback relieves some of the pressure and makes it more likely you can figure out what happened without creating additional issues.

+ +

As a real world example, the 14-day outage some Atlassian users experienced in May 2022 could have been an uninteresting minor incident had rolling back the deleted customer data been an option.

+

+ + + Performing Rollbacks with lakeFS + + +

+ + +

lakeFS lets you develop in your data lake in such a way that rollbacks are simple to perform. This starts by taking a commit over your lakeFS repository whenever a change to its state occurs.

+ +

Using the lakeFS UI or CLI, you can set the current state, or HEAD, of a branch to any historical commit in seconds, effectively performing a rollback.

+ +

To demonstrate how this works, let’s take the example of a lakeFS repo with the following commit history:

+ +

Commit History

+ +

As can be inferred from the history, this repo is updated every minute with a data sync from some data source. An example data sync is a typical ETL job that replicates data from an internal database or any other data source. After each sync, a commit is taken in lakeFS to save a snapshot of data at that point in time.

+

+ + + How to Rollback From a Bad Data Sync? + + +

+ + +

Say a situation occurs where one of the syncs had bad data and is causing downstream dashboards to fail to load. Since we took a commit of the repo right after the sync ran, we can use a revert operation to undo the changes introduced in that sync.

+ +

Copy Commit ID

+ +

Step 1: Copy the commit_id associated with the commit we want to revert. As the screenshot above shows, you can use the Copy ID to Clipboard button to do this.

+ +

Step 2: Run the revert command using lakectl, the lakeFS CLI. In this example, the command will be as follows:

+ +
lakectl branch revert lakefs://example/main 9666d7c9daf37b3ba6964e733d08596ace2ec2c7bc3a4023ad8e80737a6c3e9d
+
+ +

This will undo the changes introduced by this commit, completing the rollback!

+ +

Rollback Commit

+ +

The rollback operation is that simple, even if many changes were introduced in a commit, spanning acrossmultiple data collections.

+ +

In lakeFS, rolling back data is always a one-liner.

+ + + + + + + + +
+
+ + + + +
+ +
+
+ + + + + + + + diff --git a/v1.46/understand/versioning-internals.html b/v1.46/understand/versioning-internals.html new file mode 100644 index 000000000..9c0ed11b5 --- /dev/null +++ b/v1.46/understand/versioning-internals.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/cicd_for_data.html b/v1.46/use_cases/cicd_for_data.html new file mode 100644 index 000000000..c3fe42ec1 --- /dev/null +++ b/v1.46/use_cases/cicd_for_data.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/etl_testing.html b/v1.46/use_cases/etl_testing.html new file mode 100644 index 000000000..81a672a11 --- /dev/null +++ b/v1.46/use_cases/etl_testing.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/index.html b/v1.46/use_cases/index.html new file mode 100644 index 000000000..3e823de3e --- /dev/null +++ b/v1.46/use_cases/index.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/iso_env.html b/v1.46/use_cases/iso_env.html new file mode 100644 index 000000000..81a672a11 --- /dev/null +++ b/v1.46/use_cases/iso_env.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/reproducibility.html b/v1.46/use_cases/reproducibility.html new file mode 100644 index 000000000..57e7bbfdf --- /dev/null +++ b/v1.46/use_cases/reproducibility.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/use_cases/rollback.html b/v1.46/use_cases/rollback.html new file mode 100644 index 000000000..c64ea4f76 --- /dev/null +++ b/v1.46/use_cases/rollback.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/usecases/cd.html b/v1.46/usecases/cd.html new file mode 100644 index 000000000..c3fe42ec1 --- /dev/null +++ b/v1.46/usecases/cd.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/usecases/ci.html b/v1.46/usecases/ci.html new file mode 100644 index 000000000..c3fe42ec1 --- /dev/null +++ b/v1.46/usecases/ci.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/usecases/data-devenv.html b/v1.46/usecases/data-devenv.html new file mode 100644 index 000000000..66f7a21a2 --- /dev/null +++ b/v1.46/usecases/data-devenv.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/usecases/production.html b/v1.46/usecases/production.html new file mode 100644 index 000000000..8896605e8 --- /dev/null +++ b/v1.46/usecases/production.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using.html b/v1.46/using.html new file mode 100644 index 000000000..71d0a685d --- /dev/null +++ b/v1.46/using.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/airflow.html b/v1.46/using/airflow.html new file mode 100644 index 000000000..917dde3ee --- /dev/null +++ b/v1.46/using/airflow.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/athena.html b/v1.46/using/athena.html new file mode 100644 index 000000000..1ce8d031a --- /dev/null +++ b/v1.46/using/athena.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/aws_cli.html b/v1.46/using/aws_cli.html new file mode 100644 index 000000000..a63fea316 --- /dev/null +++ b/v1.46/using/aws_cli.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/boto.html b/v1.46/using/boto.html new file mode 100644 index 000000000..7899a7d60 --- /dev/null +++ b/v1.46/using/boto.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/databricks.html b/v1.46/using/databricks.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/using/databricks.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/dremio.html b/v1.46/using/dremio.html new file mode 100644 index 000000000..95c9c6d91 --- /dev/null +++ b/v1.46/using/dremio.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/emr.html b/v1.46/using/emr.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/using/emr.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/glue_etl.html b/v1.46/using/glue_etl.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/using/glue_etl.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/glue_hive_metastore.html b/v1.46/using/glue_hive_metastore.html new file mode 100644 index 000000000..1279ee1f4 --- /dev/null +++ b/v1.46/using/glue_hive_metastore.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/glue_metastore.html b/v1.46/using/glue_metastore.html new file mode 100644 index 000000000..8d36f97e0 --- /dev/null +++ b/v1.46/using/glue_metastore.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/hive.html b/v1.46/using/hive.html new file mode 100644 index 000000000..e1c2655f3 --- /dev/null +++ b/v1.46/using/hive.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/kakfa.html b/v1.46/using/kakfa.html new file mode 100644 index 000000000..7e1bfa8ee --- /dev/null +++ b/v1.46/using/kakfa.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/minio.html b/v1.46/using/minio.html new file mode 100644 index 000000000..4af5b9be6 --- /dev/null +++ b/v1.46/using/minio.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/presto.html b/v1.46/using/presto.html new file mode 100644 index 000000000..10389a5cd --- /dev/null +++ b/v1.46/using/presto.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/python.html b/v1.46/using/python.html new file mode 100644 index 000000000..7899a7d60 --- /dev/null +++ b/v1.46/using/python.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/sagemaker.html b/v1.46/using/sagemaker.html new file mode 100644 index 000000000..2f7018de9 --- /dev/null +++ b/v1.46/using/sagemaker.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using/spark.html b/v1.46/using/spark.html new file mode 100644 index 000000000..35bd6c453 --- /dev/null +++ b/v1.46/using/spark.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/v1.46/using_lakefs.html b/v1.46/using_lakefs.html new file mode 100644 index 000000000..2045b63eb --- /dev/null +++ b/v1.46/using_lakefs.html @@ -0,0 +1,11 @@ + + + + Redirecting… + + + + +

Redirecting…

+ Click here if you are not redirected. + diff --git a/versions.json b/versions.json index bee557c95..6efee606c 100644 --- a/versions.json +++ b/versions.json @@ -99,5 +99,6 @@ "v1.42": "v1.42", "v1.43": "v1.43", "v1.44": "v1.44", - "v1.45": "v1.45" + "v1.45": "v1.45", + "v1.46": "v1.46" }