diff --git a/404.html b/404.html index 3ac38d05..94ac5cdb 100644 --- a/404.html +++ b/404.html @@ -10,14 +10,14 @@ - - - + + + -
Skip to main content
404

Page Not Found

The page you're looking for doesn't exist or has been moved. Try navigating back to the docs or homepage.

+
Skip to main content
404

Page Not Found

The page you're looking for doesn't exist or has been moved. Try navigating back to the docs or homepage.

\ No newline at end of file diff --git a/assets/css/styles.3df0f3f7.css b/assets/css/styles.4bd6bc2b.css similarity index 65% rename from assets/css/styles.3df0f3f7.css rename to assets/css/styles.4bd6bc2b.css index 16dfab56..316646cf 100644 --- a/assets/css/styles.3df0f3f7.css +++ b/assets/css/styles.4bd6bc2b.css @@ -1 +1 @@ -@import url(https://fonts.googleapis.com/css2?family=Racing+Sans+One&family=Poppins:wght@300&family=Roboto+Mono:wght@500;300&display=swap);.col,.container{padding:0 var(--ifm-spacing-horizontal);width:100%}.markdown>h2,.markdown>h3,.markdown>h4,.markdown>h5,.markdown>h6{margin-bottom:calc(var(--ifm-heading-vertical-rhythm-bottom)*var(--ifm-leading))}.markdown li,body{word-wrap:break-word}body,ol ol,ol ul,ul ol,ul ul{margin:0}pre,table{overflow:auto}blockquote,pre{margin:0 0 var(--ifm-spacing-vertical)}.breadcrumbs__link,.button{transition-timing-function:var(--ifm-transition-timing-default)}.button,code{vertical-align:middle}.button--outline.button--active,.button--outline:active,.button--outline:hover,:root{--ifm-button-color:var(--ifm-font-color-base-inverse)}.menu__link:hover,a{transition:color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.navbar--dark,:root{--ifm-navbar-link-hover-color:var(--ifm-color-primary)}.menu,.navbar-sidebar{overflow-x:hidden}:root,html[data-theme=dark]{--ifm-color-emphasis-500:var(--ifm-color-gray-500)}#carbonads .carbon-poweredby,#carbonads a,#carbonads a:hover{color:var(--carbon-text-color)}.toggleButton_gllP,html{-webkit-tap-highlight-color:transparent}*,.loadingRing_RJI3 div{box-sizing:border-box}.close,.home-page-features-wrapper .feature div.read-the-docs{float:right}.commitMsg_XeKn,.name_mmMI,.text--truncate{text-overflow:ellipsis}.clean-list,.containsTaskList_mC6p,.details_lb9f>summary,.dropdown__menu,.grid_qTnk,.menu__list{list-style:none}:root{--ifm-color-scheme:light;--ifm-dark-value:10%;--ifm-darker-value:15%;--ifm-darkest-value:30%;--ifm-light-value:15%;--ifm-lighter-value:30%;--ifm-lightest-value:50%;--ifm-contrast-background-value:90%;--ifm-contrast-foreground-value:70%;--ifm-contrast-background-dark-value:70%;--ifm-contrast-foreground-dark-value:90%;--ifm-color-primary:#3578e5;--ifm-color-secondary:#ebedf0;--ifm-color-success:#00a400;--ifm-color-info:#54c7ec;--ifm-color-warning:#ffba00;--ifm-color-danger:#fa383e;--ifm-color-primary-dark:#306cce;--ifm-color-primary-darker:#2d66c3;--ifm-color-primary-darkest:#2554a0;--ifm-color-primary-light:#538ce9;--ifm-color-primary-lighter:#72a1ed;--ifm-color-primary-lightest:#9abcf2;--ifm-color-primary-contrast-background:#ebf2fc;--ifm-color-primary-contrast-foreground:#102445;--ifm-color-secondary-dark:#d4d5d8;--ifm-color-secondary-darker:#c8c9cc;--ifm-color-secondary-darkest:#a4a6a8;--ifm-color-secondary-light:#eef0f2;--ifm-color-secondary-lighter:#f1f2f5;--ifm-color-secondary-lightest:#f5f6f8;--ifm-color-secondary-contrast-background:#fdfdfe;--ifm-color-secondary-contrast-foreground:#474748;--ifm-color-success-dark:#009400;--ifm-color-success-darker:#008b00;--ifm-color-success-darkest:#007300;--ifm-color-success-light:#26b226;--ifm-color-success-lighter:#4dbf4d;--ifm-color-success-lightest:#80d280;--ifm-color-success-contrast-background:#e6f6e6;--ifm-color-success-contrast-foreground:#003100;--ifm-color-info-dark:#4cb3d4;--ifm-color-info-darker:#47a9c9;--ifm-color-info-darkest:#3b8ba5;--ifm-color-info-light:#6ecfef;--ifm-color-info-lighter:#87d8f2;--ifm-color-info-lightest:#aae3f6;--ifm-color-info-contrast-background:#eef9fd;--ifm-color-info-contrast-foreground:#193c47;--ifm-color-warning-dark:#e6a700;--ifm-color-warning-darker:#d99e00;--ifm-color-warning-darkest:#b38200;--ifm-color-warning-light:#ffc426;--ifm-color-warning-lighter:#ffcf4d;--ifm-color-warning-lightest:#ffdd80;--ifm-color-warning-contrast-background:#fff8e6;--ifm-color-warning-contrast-foreground:#4d3800;--ifm-color-danger-dark:#e13238;--ifm-color-danger-darker:#d53035;--ifm-color-danger-darkest:#af272b;--ifm-color-danger-light:#fb565b;--ifm-color-danger-lighter:#fb7478;--ifm-color-danger-lightest:#fd9c9f;--ifm-color-danger-contrast-background:#ffebec;--ifm-color-danger-contrast-foreground:#4b1113;--ifm-color-white:#fff;--ifm-color-black:#000;--ifm-color-gray-0:var(--ifm-color-white);--ifm-color-gray-100:#f5f6f7;--ifm-color-gray-200:#ebedf0;--ifm-color-gray-300:#dadde1;--ifm-color-gray-400:#ccd0d5;--ifm-color-gray-500:#bec3c9;--ifm-color-gray-600:#8d949e;--ifm-color-gray-700:#606770;--ifm-color-gray-800:#444950;--ifm-color-gray-900:#1c1e21;--ifm-color-gray-1000:var(--ifm-color-black);--ifm-color-emphasis-0:var(--ifm-color-gray-0);--ifm-color-emphasis-100:var(--ifm-color-gray-100);--ifm-color-emphasis-200:var(--ifm-color-gray-200);--ifm-color-emphasis-300:var(--ifm-color-gray-300);--ifm-color-emphasis-400:var(--ifm-color-gray-400);--ifm-color-emphasis-600:var(--ifm-color-gray-600);--ifm-color-emphasis-700:var(--ifm-color-gray-700);--ifm-color-emphasis-800:var(--ifm-color-gray-800);--ifm-color-emphasis-900:var(--ifm-color-gray-900);--ifm-color-emphasis-1000:var(--ifm-color-gray-1000);--ifm-color-content:var(--ifm-color-emphasis-900);--ifm-color-content-inverse:var(--ifm-color-emphasis-0);--ifm-color-content-secondary:#525860;--ifm-background-color:#0000;--ifm-background-surface-color:var(--ifm-color-content-inverse);--ifm-global-border-width:1px;--ifm-global-radius:0.4rem;--ifm-hover-overlay:#0000000d;--ifm-font-color-base:var(--ifm-color-content);--ifm-font-color-base-inverse:var(--ifm-color-content-inverse);--ifm-font-color-secondary:var(--ifm-color-content-secondary);--ifm-font-family-base:system-ui,-apple-system,Segoe UI,Roboto,Ubuntu,Cantarell,Noto Sans,sans-serif,BlinkMacSystemFont,"Segoe UI",Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";--ifm-font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--ifm-font-size-base:100%;--ifm-font-weight-light:300;--ifm-font-weight-normal:400;--ifm-font-weight-semibold:500;--ifm-font-weight-bold:700;--ifm-font-weight-base:var(--ifm-font-weight-normal);--ifm-line-height-base:1.65;--ifm-global-spacing:1rem;--ifm-spacing-vertical:var(--ifm-global-spacing);--ifm-spacing-horizontal:var(--ifm-global-spacing);--ifm-transition-fast:200ms;--ifm-transition-slow:400ms;--ifm-transition-timing-default:cubic-bezier(0.08,0.52,0.52,1);--ifm-global-shadow-lw:0 1px 2px 0 #0000001a;--ifm-global-shadow-md:0 5px 40px #0003;--ifm-global-shadow-tl:0 12px 28px 0 #0003,0 2px 4px 0 #0000001a;--ifm-z-index-dropdown:100;--ifm-z-index-fixed:200;--ifm-z-index-overlay:400;--ifm-container-width:1140px;--ifm-container-width-xl:1320px;--ifm-code-background:#f6f7f8;--ifm-code-border-radius:var(--ifm-global-radius);--ifm-code-font-size:90%;--ifm-code-padding-horizontal:0.1rem;--ifm-code-padding-vertical:0.1rem;--ifm-pre-background:var(--ifm-code-background);--ifm-pre-border-radius:var(--ifm-code-border-radius);--ifm-pre-color:inherit;--ifm-pre-line-height:1.45;--ifm-pre-padding:1rem;--ifm-heading-color:inherit;--ifm-heading-margin-top:0;--ifm-heading-margin-bottom:var(--ifm-spacing-vertical);--ifm-heading-font-family:var(--ifm-font-family-base);--ifm-heading-font-weight:var(--ifm-font-weight-bold);--ifm-heading-line-height:1.25;--ifm-h1-font-size:2rem;--ifm-h2-font-size:1.5rem;--ifm-h3-font-size:1.25rem;--ifm-h4-font-size:1rem;--ifm-h5-font-size:0.875rem;--ifm-h6-font-size:0.85rem;--ifm-image-alignment-padding:1.25rem;--ifm-leading-desktop:1.25;--ifm-leading:calc(var(--ifm-leading-desktop)*1rem);--ifm-list-left-padding:2rem;--ifm-list-margin:1rem;--ifm-list-item-margin:0.25rem;--ifm-list-paragraph-margin:1rem;--ifm-table-cell-padding:0.75rem;--ifm-table-background:#0000;--ifm-table-stripe-background:#00000008;--ifm-table-border-width:1px;--ifm-table-border-color:var(--ifm-color-emphasis-300);--ifm-table-head-background:inherit;--ifm-table-head-color:inherit;--ifm-table-head-font-weight:var(--ifm-font-weight-bold);--ifm-table-cell-color:inherit;--ifm-link-color:var(--ifm-color-primary);--ifm-link-decoration:none;--ifm-link-hover-color:var(--ifm-link-color);--ifm-link-hover-decoration:underline;--ifm-paragraph-margin-bottom:var(--ifm-leading);--ifm-blockquote-font-size:var(--ifm-font-size-base);--ifm-blockquote-border-left-width:2px;--ifm-blockquote-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-blockquote-padding-vertical:0;--ifm-blockquote-shadow:none;--ifm-blockquote-color:var(--ifm-color-emphasis-800);--ifm-blockquote-border-color:var(--ifm-color-emphasis-300);--ifm-hr-background-color:var(--ifm-color-emphasis-500);--ifm-hr-height:1px;--ifm-hr-margin-vertical:1.5rem;--ifm-scrollbar-size:7px;--ifm-scrollbar-track-background-color:#f1f1f1;--ifm-scrollbar-thumb-background-color:silver;--ifm-scrollbar-thumb-hover-background-color:#a7a7a7;--ifm-alert-background-color:inherit;--ifm-alert-border-color:inherit;--ifm-alert-border-radius:var(--ifm-global-radius);--ifm-alert-border-width:0px;--ifm-alert-border-left-width:5px;--ifm-alert-color:var(--ifm-font-color-base);--ifm-alert-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-alert-padding-vertical:var(--ifm-spacing-vertical);--ifm-alert-shadow:var(--ifm-global-shadow-lw);--ifm-avatar-intro-margin:1rem;--ifm-avatar-intro-alignment:inherit;--ifm-avatar-photo-size:3rem;--ifm-badge-background-color:inherit;--ifm-badge-border-color:inherit;--ifm-badge-border-radius:var(--ifm-global-radius);--ifm-badge-border-width:var(--ifm-global-border-width);--ifm-badge-color:var(--ifm-color-white);--ifm-badge-padding-horizontal:calc(var(--ifm-spacing-horizontal)*0.5);--ifm-badge-padding-vertical:calc(var(--ifm-spacing-vertical)*0.25);--ifm-breadcrumb-border-radius:1.5rem;--ifm-breadcrumb-spacing:0.5rem;--ifm-breadcrumb-color-active:var(--ifm-color-primary);--ifm-breadcrumb-item-background-active:var(--ifm-hover-overlay);--ifm-breadcrumb-padding-horizontal:0.8rem;--ifm-breadcrumb-padding-vertical:0.4rem;--ifm-breadcrumb-size-multiplier:1;--ifm-breadcrumb-separator:url('data:image/svg+xml;utf8,');--ifm-breadcrumb-separator-filter:none;--ifm-breadcrumb-separator-size:0.5rem;--ifm-breadcrumb-separator-size-multiplier:1.25;--ifm-button-background-color:inherit;--ifm-button-border-color:var(--ifm-button-background-color);--ifm-button-border-width:var(--ifm-global-border-width);--ifm-button-font-weight:var(--ifm-font-weight-bold);--ifm-button-padding-horizontal:1.5rem;--ifm-button-padding-vertical:0.375rem;--ifm-button-size-multiplier:1;--ifm-button-transition-duration:var(--ifm-transition-fast);--ifm-button-border-radius:calc(var(--ifm-global-radius)*var(--ifm-button-size-multiplier));--ifm-button-group-spacing:2px;--ifm-card-background-color:var(--ifm-background-surface-color);--ifm-card-border-radius:calc(var(--ifm-global-radius)*2);--ifm-card-horizontal-spacing:var(--ifm-global-spacing);--ifm-card-vertical-spacing:var(--ifm-global-spacing);--ifm-toc-border-color:var(--ifm-color-emphasis-300);--ifm-toc-link-color:var(--ifm-color-content-secondary);--ifm-toc-padding-vertical:0.5rem;--ifm-toc-padding-horizontal:0.5rem;--ifm-dropdown-background-color:var(--ifm-background-surface-color);--ifm-dropdown-font-weight:var(--ifm-font-weight-semibold);--ifm-dropdown-link-color:var(--ifm-font-color-base);--ifm-dropdown-hover-background-color:var(--ifm-hover-overlay);--ifm-footer-background-color:var(--ifm-color-emphasis-100);--ifm-footer-color:inherit;--ifm-footer-link-color:var(--ifm-color-emphasis-700);--ifm-footer-link-hover-color:var(--ifm-color-primary);--ifm-footer-link-horizontal-spacing:0.5rem;--ifm-footer-padding-horizontal:calc(var(--ifm-spacing-horizontal)*2);--ifm-footer-padding-vertical:calc(var(--ifm-spacing-vertical)*2);--ifm-footer-title-color:inherit;--ifm-footer-logo-max-width:min(30rem,90vw);--ifm-hero-background-color:var(--ifm-background-surface-color);--ifm-hero-text-color:var(--ifm-color-emphasis-800);--ifm-menu-color:var(--ifm-color-emphasis-700);--ifm-menu-color-active:var(--ifm-color-primary);--ifm-menu-color-background-active:var(--ifm-hover-overlay);--ifm-menu-color-background-hover:var(--ifm-hover-overlay);--ifm-menu-link-padding-horizontal:0.75rem;--ifm-menu-link-padding-vertical:0.375rem;--ifm-menu-link-sublist-icon:url('data:image/svg+xml;utf8,');--ifm-menu-link-sublist-icon-filter:none;--ifm-navbar-background-color:var(--ifm-background-surface-color);--ifm-navbar-height:3.75rem;--ifm-navbar-item-padding-horizontal:0.75rem;--ifm-navbar-item-padding-vertical:0.25rem;--ifm-navbar-link-color:var(--ifm-font-color-base);--ifm-navbar-link-active-color:var(--ifm-link-color);--ifm-navbar-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-navbar-padding-vertical:calc(var(--ifm-spacing-vertical)*0.5);--ifm-navbar-shadow:var(--ifm-global-shadow-lw);--ifm-navbar-search-input-background-color:var(--ifm-color-emphasis-200);--ifm-navbar-search-input-color:var(--ifm-color-emphasis-800);--ifm-navbar-search-input-placeholder-color:var(--ifm-color-emphasis-500);--ifm-navbar-search-input-icon:url('data:image/svg+xml;utf8,');--ifm-navbar-sidebar-width:83vw;--ifm-pagination-border-radius:var(--ifm-global-radius);--ifm-pagination-color-active:var(--ifm-color-primary);--ifm-pagination-font-size:1rem;--ifm-pagination-item-active-background:var(--ifm-hover-overlay);--ifm-pagination-page-spacing:0.2em;--ifm-pagination-padding-horizontal:calc(var(--ifm-spacing-horizontal)*1);--ifm-pagination-padding-vertical:calc(var(--ifm-spacing-vertical)*0.25);--ifm-pagination-nav-border-radius:var(--ifm-global-radius);--ifm-pagination-nav-color-hover:var(--ifm-color-primary);--ifm-pills-color-active:var(--ifm-color-primary);--ifm-pills-color-background-active:var(--ifm-hover-overlay);--ifm-pills-spacing:0.125rem;--ifm-tabs-color:var(--ifm-font-color-secondary);--ifm-tabs-color-active:var(--ifm-color-primary);--ifm-tabs-color-active-border:var(--ifm-tabs-color-active);--ifm-tabs-padding-horizontal:1rem;--ifm-tabs-padding-vertical:1rem}.badge--danger,.badge--info,.badge--primary,.badge--secondary,.badge--success,.badge--warning{--ifm-badge-border-color:var(--ifm-badge-background-color)}.button--link,.button--outline{--ifm-button-background-color:#0000}html{background-color:var(--ifm-background-color);color:var(--ifm-font-color-base);color-scheme:var(--ifm-color-scheme);font:var(--ifm-font-size-base)/var(--ifm-line-height-base) var(--ifm-font-family-base);-webkit-font-smoothing:antialiased;text-rendering:optimizelegibility;-webkit-text-size-adjust:100%;text-size-adjust:100%;--primary:#54bff7;--background:var(--pale-grey);--text-color:var(--black);--hero-background:var(--bright-white);--kinda-transparent:#ffffffd9;--hyperlink:#ff62ce;--feature-img-shadow:4px 4px 6px #a9a9a980,-2px -2px 4px #0006;--footer-color:var(--white);--footer-text-color:var(--black);scroll-behavior:smooth}body{background:var(--background)}iframe{border:0;color-scheme:auto}.container{margin:0 auto;max-width:var(--ifm-container-width)}.container--fluid{max-width:inherit}.row{display:flex;flex-wrap:wrap;margin:0 calc(var(--ifm-spacing-horizontal)*-1)}.margin-bottom--none,.margin-vert--none,.markdown>:last-child{margin-bottom:0!important}.margin-top--none,.margin-vert--none{margin-top:0!important}.row--no-gutters{margin-left:0;margin-right:0}.margin-horiz--none,.margin-right--none{margin-right:0!important}.row--no-gutters>.col{padding-left:0;padding-right:0}.row--align-top{align-items:flex-start}.row--align-bottom{align-items:flex-end}.menuExternalLink_NmtK,.row--align-center{align-items:center}.row--align-stretch{align-items:stretch}.row--align-baseline{align-items:baseline}.col{--ifm-col-width:100%;flex:1 0;margin-left:0;max-width:var(--ifm-col-width)}.padding-bottom--none,.padding-vert--none{padding-bottom:0!important}.padding-top--none,.padding-vert--none{padding-top:0!important}.padding-horiz--none,.padding-left--none{padding-left:0!important}.padding-horiz--none,.padding-right--none{padding-right:0!important}.col[class*=col--]{flex:0 0 var(--ifm-col-width)}.col--1{--ifm-col-width:8.33333%}.col--offset-1{margin-left:8.33333%}.col--2{--ifm-col-width:16.66667%}.col--offset-2{margin-left:16.66667%}.col--3{--ifm-col-width:25%}.col--offset-3{margin-left:25%}.col--4{--ifm-col-width:33.33333%}.col--offset-4{margin-left:33.33333%}.col--5{--ifm-col-width:41.66667%}.col--offset-5{margin-left:41.66667%}.col--6{--ifm-col-width:50%}.col--offset-6{margin-left:50%}.col--7{--ifm-col-width:58.33333%}.col--offset-7{margin-left:58.33333%}.col--8{--ifm-col-width:66.66667%}.col--offset-8{margin-left:66.66667%}.col--9{--ifm-col-width:75%}.col--offset-9{margin-left:75%}.col--10{--ifm-col-width:83.33333%}.col--offset-10{margin-left:83.33333%}.col--11{--ifm-col-width:91.66667%}.col--offset-11{margin-left:91.66667%}.col--12{--ifm-col-width:100%}.col--offset-12{margin-left:100%}.margin-horiz--none,.margin-left--none{margin-left:0!important}.margin--none{margin:0!important}.margin-bottom--xs,.margin-vert--xs{margin-bottom:.25rem!important}.margin-top--xs,.margin-vert--xs{margin-top:.25rem!important}.margin-horiz--xs,.margin-left--xs{margin-left:.25rem!important}.margin-horiz--xs,.margin-right--xs{margin-right:.25rem!important}.margin--xs{margin:.25rem!important}.margin-bottom--sm,.margin-vert--sm{margin-bottom:.5rem!important}.margin-top--sm,.margin-vert--sm{margin-top:.5rem!important}.margin-horiz--sm,.margin-left--sm{margin-left:.5rem!important}.margin-horiz--sm,.margin-right--sm{margin-right:.5rem!important}.margin--sm{margin:.5rem!important}.margin-bottom--md,.margin-vert--md{margin-bottom:1rem!important}.margin-top--md,.margin-vert--md{margin-top:1rem!important}.margin-horiz--md,.margin-left--md{margin-left:1rem!important}.margin-horiz--md,.margin-right--md{margin-right:1rem!important}.margin--md{margin:1rem!important}.margin-bottom--lg,.margin-vert--lg{margin-bottom:2rem!important}.margin-top--lg,.margin-vert--lg{margin-top:2rem!important}.margin-horiz--lg,.margin-left--lg{margin-left:2rem!important}.margin-horiz--lg,.margin-right--lg{margin-right:2rem!important}.margin--lg{margin:2rem!important}.margin-bottom--xl,.margin-vert--xl{margin-bottom:5rem!important}.margin-top--xl,.margin-vert--xl{margin-top:5rem!important}.margin-horiz--xl,.margin-left--xl{margin-left:5rem!important}.margin-horiz--xl,.margin-right--xl{margin-right:5rem!important}.margin--xl{margin:5rem!important}.padding--none{padding:0!important}.padding-bottom--xs,.padding-vert--xs{padding-bottom:.25rem!important}.padding-top--xs,.padding-vert--xs{padding-top:.25rem!important}.padding-horiz--xs,.padding-left--xs{padding-left:.25rem!important}.padding-horiz--xs,.padding-right--xs{padding-right:.25rem!important}.padding--xs{padding:.25rem!important}.padding-bottom--sm,.padding-vert--sm{padding-bottom:.5rem!important}.padding-top--sm,.padding-vert--sm{padding-top:.5rem!important}.padding-horiz--sm,.padding-left--sm{padding-left:.5rem!important}.padding-horiz--sm,.padding-right--sm{padding-right:.5rem!important}.padding--sm{padding:.5rem!important}.padding-bottom--md,.padding-vert--md{padding-bottom:1rem!important}.padding-top--md,.padding-vert--md{padding-top:1rem!important}.padding-horiz--md,.padding-left--md{padding-left:1rem!important}.padding-horiz--md,.padding-right--md{padding-right:1rem!important}.padding--md{padding:1rem!important}.padding-bottom--lg,.padding-vert--lg{padding-bottom:2rem!important}.padding-top--lg,.padding-vert--lg{padding-top:2rem!important}.padding-horiz--lg,.padding-left--lg{padding-left:2rem!important}.padding-horiz--lg,.padding-right--lg{padding-right:2rem!important}.padding--lg{padding:2rem!important}.padding-bottom--xl,.padding-vert--xl{padding-bottom:5rem!important}.padding-top--xl,.padding-vert--xl{padding-top:5rem!important}.padding-horiz--xl,.padding-left--xl{padding-left:5rem!important}.padding-horiz--xl,.padding-right--xl{padding-right:5rem!important}.padding--xl{padding:5rem!important}code{background-color:var(--ifm-code-background);border:.1rem solid #0000001a;border-radius:var(--ifm-code-border-radius);font-family:var(--ifm-font-family-monospace);font-size:var(--ifm-code-font-size);padding:var(--ifm-code-padding-vertical) var(--ifm-code-padding-horizontal)}a code{color:inherit}pre{background-color:var(--ifm-pre-background);border-radius:var(--ifm-pre-border-radius);color:var(--ifm-pre-color);font:var(--ifm-code-font-size)/var(--ifm-pre-line-height) var(--ifm-font-family-monospace);padding:var(--ifm-pre-padding)}pre code{background-color:initial;border:none;font-size:100%;line-height:inherit;padding:0}kbd{background-color:var(--ifm-color-emphasis-0);border:1px solid var(--ifm-color-emphasis-400);border-radius:.2rem;box-shadow:inset 0 -1px 0 var(--ifm-color-emphasis-400);color:var(--ifm-color-emphasis-800);font:80% var(--ifm-font-family-monospace);padding:.15rem .3rem}h1,h2,h3,h4,h5,h6{color:var(--ifm-heading-color);font-family:var(--ifm-heading-font-family);font-weight:var(--ifm-heading-font-weight);line-height:var(--ifm-heading-line-height);margin:var(--ifm-heading-margin-top) 0 var(--ifm-heading-margin-bottom) 0}article div.markdown h1,h1{font-size:var(--ifm-h1-font-size)}article div.markdown h2,h2{font-size:var(--ifm-h2-font-size)}article div.markdown h3,h3{font-size:var(--ifm-h3-font-size)}h4{font-size:var(--ifm-h4-font-size)}h5{font-size:var(--ifm-h5-font-size)}h6{font-size:var(--ifm-h6-font-size)}.container_lyt7,.container_lyt7>svg,img{max-width:100%}img[align=right]{padding-left:var(--image-alignment-padding)}img[align=left]{padding-right:var(--image-alignment-padding)}.markdown{--ifm-h1-vertical-rhythm-top:3;--ifm-h2-vertical-rhythm-top:2;--ifm-h3-vertical-rhythm-top:1.5;--ifm-heading-vertical-rhythm-top:1.25;--ifm-h1-vertical-rhythm-bottom:1.25;--ifm-heading-vertical-rhythm-bottom:1}.markdown:after,.markdown:before{content:"";display:table}.markdown:after{clear:both}.markdown h1:first-child{--ifm-h1-font-size:3rem;margin-bottom:calc(var(--ifm-h1-vertical-rhythm-bottom)*var(--ifm-leading))}.markdown>h2{--ifm-h2-font-size:2rem;margin-top:calc(var(--ifm-h2-vertical-rhythm-top)*var(--ifm-leading))}.markdown>h3{--ifm-h3-font-size:1.5rem;margin-top:calc(var(--ifm-h3-vertical-rhythm-top)*var(--ifm-leading))}.markdown>h4,.markdown>h5,.markdown>h6{margin-top:calc(var(--ifm-heading-vertical-rhythm-top)*var(--ifm-leading))}.markdown>p,.markdown>pre,.markdown>ul{margin-bottom:var(--ifm-leading)}.markdown li>p{margin-top:var(--ifm-list-paragraph-margin)}.markdown li+li{margin-top:var(--ifm-list-item-margin)}ol,ul{margin:0 0 var(--ifm-list-margin);padding-left:var(--ifm-list-left-padding)}ol ol,ul ol{list-style-type:lower-roman}ol ol ol,ol ul ol,ul ol ol,ul ul ol{list-style-type:lower-alpha}table{border-collapse:collapse;display:block;margin-bottom:var(--ifm-spacing-vertical)}table thead tr{border-bottom:2px solid var(--ifm-table-border-color)}table thead,table tr:nth-child(2n){background-color:var(--ifm-table-stripe-background)}table tr{background-color:var(--ifm-table-background);border-top:var(--ifm-table-border-width) solid var(--ifm-table-border-color)}table td,table th{border:var(--ifm-table-border-width) solid var(--ifm-table-border-color);padding:var(--ifm-table-cell-padding)}table th{background-color:var(--ifm-table-head-background);color:var(--ifm-table-head-color);font-weight:var(--ifm-table-head-font-weight)}table td{color:var(--ifm-table-cell-color)}strong{font-weight:var(--ifm-font-weight-bold)}a{color:var(--ifm-link-color);text-decoration:var(--ifm-link-decoration)}a:hover{color:var(--ifm-link-hover-color);text-decoration:var(--ifm-link-hover-decoration)}.button:hover,.text--no-decoration,.text--no-decoration:hover,a:not([href]){-webkit-text-decoration:none;text-decoration:none}p{margin:0 0 var(--ifm-paragraph-margin-bottom)}blockquote{border-left:var(--ifm-blockquote-border-left-width) solid var(--ifm-blockquote-border-color);box-shadow:var(--ifm-blockquote-shadow);color:var(--ifm-blockquote-color);font-size:var(--ifm-blockquote-font-size);padding:var(--ifm-blockquote-padding-vertical) var(--ifm-blockquote-padding-horizontal)}blockquote>:first-child{margin-top:0}blockquote>:last-child{margin-bottom:0}hr{background-color:var(--ifm-hr-background-color);border:0;height:var(--ifm-hr-height);margin:var(--ifm-hr-margin-vertical) 0}.shadow--lw{box-shadow:var(--ifm-global-shadow-lw)!important}.shadow--md{box-shadow:var(--ifm-global-shadow-md)!important}.shadow--tl{box-shadow:var(--ifm-global-shadow-tl)!important}.text--primary,.wordWrapButtonEnabled_uzNF .wordWrapButtonIcon_b1P5{color:var(--ifm-color-primary)}.text--secondary{color:var(--ifm-color-secondary)}.text--success{color:var(--ifm-color-success)}.text--info{color:var(--ifm-color-info)}.text--warning{color:var(--ifm-color-warning)}.text--danger{color:var(--ifm-color-danger)}.text--center{text-align:center}.text--left{text-align:left}.text--justify{text-align:justify}.text--right{text-align:right}.text--capitalize{text-transform:capitalize}.text--lowercase{text-transform:lowercase}.admonitionHeading_Gvgb,.alert__heading,.text--uppercase{text-transform:uppercase}.text--light{font-weight:var(--ifm-font-weight-light)}.text--normal{font-weight:var(--ifm-font-weight-normal)}.text--semibold{font-weight:var(--ifm-font-weight-semibold)}.text--bold{font-weight:var(--ifm-font-weight-bold)}.text--italic{font-style:italic}.text--truncate{overflow:hidden;white-space:nowrap}.text--break{word-wrap:break-word!important;word-break:break-word!important}.clean-btn{background:none;border:none;color:inherit;cursor:pointer;font-family:inherit;padding:0}.alert,.alert .close{color:var(--ifm-alert-foreground-color)}.clean-list{padding-left:0}.alert--primary{--ifm-alert-background-color:var(--ifm-color-primary-contrast-background);--ifm-alert-background-color-highlight:#3578e526;--ifm-alert-foreground-color:var(--ifm-color-primary-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-primary-dark)}.alert--secondary{--ifm-alert-background-color:var(--ifm-color-secondary-contrast-background);--ifm-alert-background-color-highlight:#ebedf026;--ifm-alert-foreground-color:var(--ifm-color-secondary-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-secondary-dark)}.alert--success{--ifm-alert-background-color:var(--ifm-color-success-contrast-background);--ifm-alert-background-color-highlight:#00a40026;--ifm-alert-foreground-color:var(--ifm-color-success-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-success-dark)}.alert--info{--ifm-alert-background-color:var(--ifm-color-info-contrast-background);--ifm-alert-background-color-highlight:#54c7ec26;--ifm-alert-foreground-color:var(--ifm-color-info-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-info-dark)}.alert--warning{--ifm-alert-background-color:var(--ifm-color-warning-contrast-background);--ifm-alert-background-color-highlight:#ffba0026;--ifm-alert-foreground-color:var(--ifm-color-warning-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-warning-dark)}.alert--danger{--ifm-alert-background-color:var(--ifm-color-danger-contrast-background);--ifm-alert-background-color-highlight:#fa383e26;--ifm-alert-foreground-color:var(--ifm-color-danger-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-danger-dark)}.alert{--ifm-code-background:var(--ifm-alert-background-color-highlight);--ifm-link-color:var(--ifm-alert-foreground-color);--ifm-link-hover-color:var(--ifm-alert-foreground-color);--ifm-link-decoration:underline;--ifm-tabs-color:var(--ifm-alert-foreground-color);--ifm-tabs-color-active:var(--ifm-alert-foreground-color);--ifm-tabs-color-active-border:var(--ifm-alert-border-color);background-color:var(--ifm-alert-background-color);border:var(--ifm-alert-border-width) solid var(--ifm-alert-border-color);border-left-width:var(--ifm-alert-border-left-width);border-radius:var(--ifm-alert-border-radius);box-shadow:var(--ifm-alert-shadow);padding:var(--ifm-alert-padding-vertical) var(--ifm-alert-padding-horizontal)}.alert__heading{align-items:center;display:flex;font:700 var(--ifm-h5-font-size)/var(--ifm-heading-line-height) var(--ifm-heading-font-family);margin-bottom:.5rem}.alert__icon{display:inline-flex;margin-right:.4em}.alert__icon svg{fill:var(--ifm-alert-foreground-color);stroke:var(--ifm-alert-foreground-color);stroke-width:0}.alert .close{margin:calc(var(--ifm-alert-padding-vertical)*-1) calc(var(--ifm-alert-padding-horizontal)*-1) 0 0;opacity:.75}.alert .close:focus,.alert .close:hover{opacity:1}.alert a{text-decoration-color:var(--ifm-alert-border-color)}.alert a:hover{text-decoration-thickness:2px}.avatar{column-gap:var(--ifm-avatar-intro-margin);display:flex}.avatar__photo{border-radius:50%;display:block;height:var(--ifm-avatar-photo-size);overflow:hidden;width:var(--ifm-avatar-photo-size)}.card--full-height,.navbar__logo img,body,html{height:100%}.avatar__photo--sm{--ifm-avatar-photo-size:2rem}.avatar__photo--lg{--ifm-avatar-photo-size:4rem}.avatar__photo--xl{--ifm-avatar-photo-size:6rem}.avatar__intro{display:flex;flex:1 1;flex-direction:column;justify-content:center;text-align:var(--ifm-avatar-intro-alignment)}.badge,.breadcrumbs__item,.breadcrumbs__link,.dropdown>.navbar__link:after,.searchBarContainer_NW3z.searchIndexLoading_EJ1f .searchBarLoadingRing_YnHq{display:inline-block}.avatar__name{font:700 var(--ifm-h4-font-size)/var(--ifm-heading-line-height) var(--ifm-font-family-base)}.avatar__subtitle{margin-top:.25rem}.avatar--vertical{--ifm-avatar-intro-alignment:center;--ifm-avatar-intro-margin:0.5rem;align-items:center;flex-direction:column}.badge{background-color:var(--ifm-badge-background-color);border:var(--ifm-badge-border-width) solid var(--ifm-badge-border-color);border-radius:var(--ifm-badge-border-radius);color:var(--ifm-badge-color);font-size:75%;font-weight:var(--ifm-font-weight-bold);line-height:1;padding:var(--ifm-badge-padding-vertical) var(--ifm-badge-padding-horizontal)}.badge--primary{--ifm-badge-background-color:var(--ifm-color-primary)}.badge--secondary{--ifm-badge-background-color:var(--ifm-color-secondary);color:var(--ifm-color-black)}.breadcrumbs__link,.button.button--secondary.button--outline:not(.button--active):not(:hover){color:var(--ifm-font-color-base)}.badge--success{--ifm-badge-background-color:var(--ifm-color-success)}.badge--info{--ifm-badge-background-color:var(--ifm-color-info)}.badge--warning{--ifm-badge-background-color:var(--ifm-color-warning)}.badge--danger{--ifm-badge-background-color:var(--ifm-color-danger)}.breadcrumbs{margin-bottom:0;padding-left:0}.breadcrumbs__item:not(:last-child):after{background:var(--ifm-breadcrumb-separator) center;content:" ";display:inline-block;filter:var(--ifm-breadcrumb-separator-filter);height:calc(var(--ifm-breadcrumb-separator-size)*var(--ifm-breadcrumb-size-multiplier)*var(--ifm-breadcrumb-separator-size-multiplier));margin:0 var(--ifm-breadcrumb-spacing);opacity:.5;width:calc(var(--ifm-breadcrumb-separator-size)*var(--ifm-breadcrumb-size-multiplier)*var(--ifm-breadcrumb-separator-size-multiplier))}.breadcrumbs__item--active .breadcrumbs__link{background:var(--ifm-breadcrumb-item-background-active);color:var(--ifm-breadcrumb-color-active)}.breadcrumbs__link{border-radius:var(--ifm-breadcrumb-border-radius);font-size:calc(1rem*var(--ifm-breadcrumb-size-multiplier));padding:calc(var(--ifm-breadcrumb-padding-vertical)*var(--ifm-breadcrumb-size-multiplier)) calc(var(--ifm-breadcrumb-padding-horizontal)*var(--ifm-breadcrumb-size-multiplier));transition-duration:var(--ifm-transition-fast);transition-property:background,color}.breadcrumbs__link:any-link:hover,.breadcrumbs__link:link:hover,.breadcrumbs__link:visited:hover,area[href].breadcrumbs__link:hover{background:var(--ifm-breadcrumb-item-background-active);-webkit-text-decoration:none;text-decoration:none}.breadcrumbs--sm{--ifm-breadcrumb-size-multiplier:0.8}.breadcrumbs--lg{--ifm-breadcrumb-size-multiplier:1.2}.button{background-color:var(--ifm-button-background-color);border:var(--ifm-button-border-width) solid var(--ifm-button-border-color);border-radius:var(--ifm-button-border-radius);font-size:calc(.875rem*var(--ifm-button-size-multiplier));font-weight:var(--ifm-button-font-weight);line-height:1.5;padding:calc(var(--ifm-button-padding-vertical)*var(--ifm-button-size-multiplier)) calc(var(--ifm-button-padding-horizontal)*var(--ifm-button-size-multiplier));text-align:center;transition-duration:var(--ifm-button-transition-duration);-webkit-user-select:none;user-select:none;white-space:nowrap}.button,.button:hover{color:var(--ifm-button-color)}.button--outline{--ifm-button-color:var(--ifm-button-border-color)}.button--outline:hover{--ifm-button-background-color:var(--ifm-button-border-color)}.button--link{--ifm-button-border-color:#0000;color:var(--ifm-link-color);text-decoration:var(--ifm-link-decoration)}.button--link.button--active,.button--link:active,.button--link:hover{color:var(--ifm-link-hover-color);text-decoration:var(--ifm-link-hover-decoration)}#carbonads a,.banner_woPo,.button-link-wrapper:hover,.dropdown__link--active,.dropdown__link:hover,.link2_y3x6:hover,.link_ecgS:hover,.menu__link:hover,.navLink_KmGT,.navLink_KmGT:hover,.navbar__brand:hover,.navbar__link--active,.navbar__link:hover,.pagination-nav__link:hover,.pagination__link:hover{-webkit-text-decoration:none;text-decoration:none}.button.disabled,.button:disabled,.button[disabled]{opacity:.65;pointer-events:none}.button--sm{--ifm-button-size-multiplier:0.8}.button--lg{--ifm-button-size-multiplier:1.35}.button--block{display:block;width:100%}.button.button--secondary{color:var(--ifm-color-gray-900)}:where(.button--primary){--ifm-button-background-color:var(--ifm-color-primary);--ifm-button-border-color:var(--ifm-color-primary)}:where(.button--primary):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-primary-dark);--ifm-button-border-color:var(--ifm-color-primary-dark)}.button--primary.button--active,.button--primary:active{--ifm-button-background-color:var(--ifm-color-primary-darker);--ifm-button-border-color:var(--ifm-color-primary-darker)}:where(.button--secondary){--ifm-button-background-color:var(--ifm-color-secondary);--ifm-button-border-color:var(--ifm-color-secondary)}:where(.button--secondary):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-secondary-dark);--ifm-button-border-color:var(--ifm-color-secondary-dark)}.button--secondary.button--active,.button--secondary:active{--ifm-button-background-color:var(--ifm-color-secondary-darker);--ifm-button-border-color:var(--ifm-color-secondary-darker)}:where(.button--success){--ifm-button-background-color:var(--ifm-color-success);--ifm-button-border-color:var(--ifm-color-success)}:where(.button--success):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-success-dark);--ifm-button-border-color:var(--ifm-color-success-dark)}.button--success.button--active,.button--success:active{--ifm-button-background-color:var(--ifm-color-success-darker);--ifm-button-border-color:var(--ifm-color-success-darker)}:where(.button--info){--ifm-button-background-color:var(--ifm-color-info);--ifm-button-border-color:var(--ifm-color-info)}:where(.button--info):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-info-dark);--ifm-button-border-color:var(--ifm-color-info-dark)}.button--info.button--active,.button--info:active{--ifm-button-background-color:var(--ifm-color-info-darker);--ifm-button-border-color:var(--ifm-color-info-darker)}:where(.button--warning){--ifm-button-background-color:var(--ifm-color-warning);--ifm-button-border-color:var(--ifm-color-warning)}:where(.button--warning):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-warning-dark);--ifm-button-border-color:var(--ifm-color-warning-dark)}.button--warning.button--active,.button--warning:active{--ifm-button-background-color:var(--ifm-color-warning-darker);--ifm-button-border-color:var(--ifm-color-warning-darker)}:where(.button--danger){--ifm-button-background-color:var(--ifm-color-danger);--ifm-button-border-color:var(--ifm-color-danger)}:where(.button--danger):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-danger-dark);--ifm-button-border-color:var(--ifm-color-danger-dark)}.button--danger.button--active,.button--danger:active{--ifm-button-background-color:var(--ifm-color-danger-darker);--ifm-button-border-color:var(--ifm-color-danger-darker)}.button-group{display:inline-flex;gap:var(--ifm-button-group-spacing)}.button-group>.button:not(:first-child){border-bottom-left-radius:0;border-top-left-radius:0}.button-group>.button:not(:last-child){border-bottom-right-radius:0;border-top-right-radius:0}.button-group--block{display:flex;justify-content:stretch}.button-group--block>.button{flex-grow:1}.card{background-color:var(--ifm-card-background-color);border-radius:var(--ifm-card-border-radius);box-shadow:var(--ifm-global-shadow-lw);display:flex;flex-direction:column;overflow:hidden}.card__image{padding-top:var(--ifm-card-vertical-spacing)}.card__image:first-child{padding-top:0}.card__body,.card__footer,.card__header{padding:var(--ifm-card-vertical-spacing) var(--ifm-card-horizontal-spacing)}.card__body:not(:last-child),.card__footer:not(:last-child),.card__header:not(:last-child){padding-bottom:0}.card__body>:last-child,.card__footer>:last-child,.card__header>:last-child{margin-bottom:0}.card__footer{margin-top:auto}.table-of-contents{font-size:.8rem;margin-bottom:0;padding:var(--ifm-toc-padding-vertical) 0}.table-of-contents,.table-of-contents ul{list-style:none;padding-left:var(--ifm-toc-padding-horizontal)}.table-of-contents li{margin:var(--ifm-toc-padding-vertical) var(--ifm-toc-padding-horizontal)}.table-of-contents__left-border{border-left:1px solid var(--ifm-toc-border-color)}.table-of-contents__link{color:var(--ifm-toc-link-color);display:block}.table-of-contents__link--active,.table-of-contents__link--active code,.table-of-contents__link:hover,.table-of-contents__link:hover code{color:var(--ifm-color-primary);-webkit-text-decoration:none;text-decoration:none}.close{color:var(--ifm-color-black);font-size:1.5rem;font-weight:var(--ifm-font-weight-bold);line-height:1;opacity:.5;padding:1rem;transition:opacity var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.close:hover{opacity:.7}.close:focus,.theme-code-block-highlighted-line .codeLineNumber_Tfdd:before{opacity:.8}.dropdown{display:inline-flex;font-weight:var(--ifm-dropdown-font-weight);position:relative;vertical-align:top}.dropdown--hoverable:hover .dropdown__menu,.dropdown--show .dropdown__menu{opacity:1;pointer-events:all;transform:translateY(-1px);visibility:visible}.dropdown--right .dropdown__menu{left:inherit;right:0}.dropdown--nocaret .navbar__link:after{content:none!important}.dropdown__menu{background-color:var(--ifm-dropdown-background-color);border-radius:var(--ifm-global-radius);box-shadow:var(--ifm-global-shadow-md);left:0;max-height:80vh;min-width:10rem;opacity:0;overflow-y:auto;padding:.5rem;pointer-events:none;position:absolute;top:calc(100% - var(--ifm-navbar-item-padding-vertical) + .3rem);transform:translateY(-.625rem);transition-duration:var(--ifm-transition-fast);transition-property:opacity,transform,visibility;transition-timing-function:var(--ifm-transition-timing-default);visibility:hidden;z-index:var(--ifm-z-index-dropdown)}.menu__caret,.menu__link,.menu__list-item-collapsible{border-radius:.25rem;transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.dropdown__link{border-radius:.25rem;color:var(--ifm-dropdown-link-color);display:block;font-size:.875rem;margin-top:.2rem;padding:.25rem .5rem;white-space:nowrap}.dropdown__link--active,.dropdown__link:hover{background-color:var(--ifm-dropdown-hover-background-color);color:var(--ifm-dropdown-link-color)}.dropdown__link--active,.dropdown__link--active:hover{--ifm-dropdown-link-color:var(--ifm-link-color)}.dropdown>.navbar__link:after{border-color:currentcolor #0000;border-style:solid;border-width:.4em .4em 0;content:"";margin-left:.3em;position:relative;top:2px;transform:translateY(-50%)}.footer{background-color:var(--ifm-footer-background-color);color:var(--ifm-footer-color);padding:var(--ifm-footer-padding-vertical) var(--ifm-footer-padding-horizontal)}.footer--dark{--ifm-footer-background-color:#303846;--ifm-footer-color:var(--ifm-footer-link-color);--ifm-footer-link-color:var(--ifm-color-secondary);--ifm-footer-title-color:var(--ifm-color-white)}.footer__links{margin-bottom:1rem}.footer__link-item{color:var(--ifm-footer-link-color);line-height:2}.footer__link-item:hover{color:var(--ifm-footer-link-hover-color)}.footer__link-separator{margin:0 var(--ifm-footer-link-horizontal-spacing)}.footer__logo{margin-top:1rem;max-width:var(--ifm-footer-logo-max-width)}.footer__title{color:var(--ifm-footer-title-color);font:700 var(--ifm-h4-font-size)/var(--ifm-heading-line-height) var(--ifm-font-family-base);margin-bottom:var(--ifm-heading-margin-bottom)}.menu,.navbar__link{font-weight:var(--ifm-font-weight-semibold)}.docItemContainer_Djhp article>:first-child,.docItemContainer_Djhp header+*,.footer__item{margin-top:0}.admonitionContent_BuS1>:last-child,.collapsibleContent_i85q p:last-child,.details_lb9f>summary>p:last-child,.footer__items,.searchResultItem_U687>h2{margin-bottom:0}.codeBlockStandalone_MEMb,[type=checkbox]{padding:0}.hero{align-items:center;background-color:var(--ifm-hero-background-color);color:var(--ifm-hero-text-color);display:flex;padding:4rem 2rem}.hero--primary{--ifm-hero-background-color:var(--ifm-color-primary);--ifm-hero-text-color:var(--ifm-font-color-base-inverse)}.hero--dark{--ifm-hero-background-color:#303846;--ifm-hero-text-color:var(--ifm-color-white)}.hero__title,h2{font-size:3rem}.hero__subtitle{font-size:1.5rem}.menu__list{margin:0;padding-left:0}.menu__caret,.menu__link{padding:var(--ifm-menu-link-padding-vertical) var(--ifm-menu-link-padding-horizontal)}.menu__list .menu__list{flex:0 0 100%;margin-top:.25rem;padding-left:var(--ifm-menu-link-padding-horizontal)}.menu__list-item:not(:first-child){margin-top:.25rem}.menu__list-item--collapsed .menu__list{height:0;overflow:hidden}.details_lb9f[data-collapsed=false].isBrowser_bmU9>summary:before,.details_lb9f[open]:not(.isBrowser_bmU9)>summary:before,.menu__list-item--collapsed .menu__caret:before,.menu__list-item--collapsed .menu__link--sublist:after{transform:rotate(90deg)}.menu__list-item-collapsible{display:flex;flex-wrap:wrap;position:relative}.menu__caret:hover,.menu__link:hover,.menu__list-item-collapsible--active,.menu__list-item-collapsible:hover{background:var(--ifm-menu-color-background-hover)}.menu__list-item-collapsible .menu__link--active,.menu__list-item-collapsible .menu__link:hover{background:none!important}.menu__caret,.menu__link{align-items:center;display:flex}.navbar-sidebar,.navbar-sidebar__backdrop{bottom:0;opacity:0;transition-duration:var(--ifm-transition-fast);transition-timing-function:ease-in-out;top:0;left:0;visibility:hidden}.menu__link{color:var(--ifm-menu-color);flex:1;line-height:1.25}.menu__link:hover{color:var(--ifm-menu-color)}.menu__caret:before,.menu__link--sublist-caret:after{height:1.25rem;transform:rotate(180deg);transition:transform var(--ifm-transition-fast) linear;width:1.25rem;content:"";filter:var(--ifm-menu-link-sublist-icon-filter)}.menu__link--sublist-caret:after{background:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem;margin-left:auto;min-width:1.25rem}.menu__link--active,.menu__link--active:hover{color:var(--ifm-menu-color-active)}.navbar__brand,.navbar__link{color:var(--ifm-navbar-link-color)}.menu__link--active:not(.menu__link--sublist){background-color:var(--ifm-menu-color-background-active)}.menu__caret:before{background:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem}.navbar--dark,html[data-theme=dark]{--ifm-menu-link-sublist-icon-filter:invert(100%) sepia(94%) saturate(17%) hue-rotate(223deg) brightness(104%) contrast(98%)}.navbar{background-color:var(--ifm-navbar-background-color);box-shadow:var(--ifm-navbar-shadow);height:var(--ifm-navbar-height);padding:var(--ifm-navbar-padding-vertical) var(--ifm-navbar-padding-horizontal)}.navbar,.navbar>.container,.navbar>.container-fluid{display:flex}.navbar--fixed-top{position:sticky;top:0;z-index:var(--ifm-z-index-fixed)}.navbar__inner{display:flex;flex-wrap:wrap;justify-content:space-between;width:100%}.navbar__brand{align-items:center;display:flex;margin-right:1rem;min-width:0}.navbar__brand:hover{color:var(--ifm-navbar-link-hover-color)}.announcementBarContent_xLdY,.navbar__title{flex:1 1 auto}.navbar__toggle{display:none;margin-right:.5rem}.navbar__logo{flex:0 0 auto;height:2rem;margin-right:.5rem}.navbar__items{align-items:center;display:flex;flex:1;min-width:0}.navbar__items--center{flex:0 0 auto}.fetchErrorCard_M8w3 p,.home-page-features-wrapper .feature .feature-title h3,.navbar__items--center .navbar__brand{margin:0}.navbar__items--center+.navbar__items--right{flex:1}.navbar__items--right{flex:0 0 auto;justify-content:flex-end}.navbar__item{display:inline-block;padding:var(--ifm-navbar-item-padding-vertical) var(--ifm-navbar-item-padding-horizontal)}#nprogress,.navbar__item.dropdown .navbar__link:not([href]){pointer-events:none}.navbar__link--active,.navbar__link:hover{color:var(--ifm-navbar-link-hover-color)}.navbar--dark,.navbar--primary{--ifm-menu-color:var(--ifm-color-gray-300);--ifm-navbar-link-color:var(--ifm-color-gray-100);--ifm-navbar-search-input-background-color:#ffffff1a;--ifm-navbar-search-input-placeholder-color:#ffffff80;color:var(--ifm-color-white)}.navbar--dark{--ifm-navbar-background-color:#242526;--ifm-menu-color-background-active:#ffffff0d;--ifm-navbar-search-input-color:var(--ifm-color-white)}.navbar--primary{--ifm-navbar-background-color:var(--ifm-color-primary);--ifm-navbar-link-hover-color:var(--ifm-color-white);--ifm-menu-color-active:var(--ifm-color-white);--ifm-navbar-search-input-color:var(--ifm-color-emphasis-500)}.navbar__search-input{appearance:none;background:var(--ifm-navbar-search-input-background-color) var(--ifm-navbar-search-input-icon) no-repeat .75rem center/1rem 1rem;border:none;border-radius:2rem;color:var(--ifm-navbar-search-input-color);cursor:text;display:inline-block;font-size:1rem;height:2rem;padding:0 .5rem 0 2.25rem;width:12.5rem}.navbar__search-input::placeholder{color:var(--ifm-navbar-search-input-placeholder-color)}.navbar-sidebar{background-color:var(--ifm-navbar-background-color);box-shadow:var(--ifm-global-shadow-md);position:fixed;transform:translate3d(-100%,0,0);transition-property:opacity,visibility,transform;width:var(--ifm-navbar-sidebar-width)}.navbar-sidebar--show .navbar-sidebar,.navbar-sidebar__items{transform:translateZ(0)}.navbar-sidebar--show .navbar-sidebar,.navbar-sidebar--show .navbar-sidebar__backdrop{opacity:1;visibility:visible}.navbar-sidebar__backdrop{background-color:#0009;position:fixed;right:0;transition-property:opacity,visibility}.navbar-sidebar__brand{align-items:center;box-shadow:var(--ifm-navbar-shadow);display:flex;flex:1;height:var(--ifm-navbar-height);padding:var(--ifm-navbar-padding-vertical) var(--ifm-navbar-padding-horizontal)}.footer,article div.markdown p img,nav.navbar{box-shadow:var(--feature-img-shadow)}.navbar-sidebar__items{display:flex;height:calc(100% - var(--ifm-navbar-height));transition:transform var(--ifm-transition-fast) ease-in-out}.navbar-sidebar__items--show-secondary{transform:translate3d(calc((var(--ifm-navbar-sidebar-width))*-1),0,0)}.navbar-sidebar__item{flex-shrink:0;padding:.5rem;width:calc(var(--ifm-navbar-sidebar-width))}.navbar-sidebar__back{background:var(--ifm-menu-color-background-active);font-size:15px;font-weight:var(--ifm-button-font-weight);margin:0 0 .2rem -.5rem;padding:.6rem 1.5rem;position:relative;text-align:left;top:-.5rem;width:calc(100% + 1rem)}.navbar-sidebar__close{display:flex;margin-left:auto}.pagination{column-gap:var(--ifm-pagination-page-spacing);display:flex;font-size:var(--ifm-pagination-font-size);padding-left:0}.pagination--sm{--ifm-pagination-font-size:0.8rem;--ifm-pagination-padding-horizontal:0.8rem;--ifm-pagination-padding-vertical:0.2rem}.pagination--lg{--ifm-pagination-font-size:1.2rem;--ifm-pagination-padding-horizontal:1.2rem;--ifm-pagination-padding-vertical:0.3rem}.pagination__item{display:inline-flex}.pagination__item>span{padding:var(--ifm-pagination-padding-vertical)}.pagination__item--active .pagination__link{color:var(--ifm-pagination-color-active)}.pagination__item--active .pagination__link,.pagination__item:not(.pagination__item--active):hover .pagination__link{background:var(--ifm-pagination-item-active-background)}.pagination__item--disabled,.pagination__item[disabled]{opacity:.25;pointer-events:none}.pagination__link{border-radius:var(--ifm-pagination-border-radius);color:var(--ifm-font-color-base);display:inline-block;padding:var(--ifm-pagination-padding-vertical) var(--ifm-pagination-padding-horizontal);transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pagination-nav{display:grid;grid-gap:var(--ifm-spacing-horizontal);gap:var(--ifm-spacing-horizontal);grid-template-columns:repeat(2,1fr)}.pagination-nav__link{border:1px solid var(--ifm-color-emphasis-300);border-radius:var(--ifm-pagination-nav-border-radius);display:block;height:100%;line-height:var(--ifm-heading-line-height);padding:var(--ifm-global-spacing);transition:border-color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pagination-nav__link:hover{border-color:var(--ifm-pagination-nav-color-hover)}.content_knG7 a,.hitFooter_E9YW a,.link2_y3x6.link2_y3x6,.link2_y3x6.link2_y3x6:hover,.link_ecgS.link2_y3x6,.link_ecgS.link2_y3x6:hover,.suggestion_fB_2.cursor_eG29 mark,li a:hover,p a:hover{-webkit-text-decoration:underline;text-decoration:underline}.pagination-nav__link--next{grid-column:2/3;text-align:right}.pagination-nav__label{font-size:var(--ifm-h4-font-size);font-weight:var(--ifm-heading-font-weight);word-break:break-word}.pagination-nav__link--prev .pagination-nav__label:before{content:"« "}.pagination-nav__link--next .pagination-nav__label:after{content:" »"}.pagination-nav__sublabel{color:var(--ifm-color-content-secondary);font-size:var(--ifm-h5-font-size);font-weight:var(--ifm-font-weight-semibold);margin-bottom:.25rem}.pills__item,.tabs{font-weight:var(--ifm-font-weight-bold)}.pills{display:flex;gap:var(--ifm-pills-spacing);padding-left:0}.pills__item{border-radius:.5rem;cursor:pointer;display:inline-block;padding:.25rem 1rem;transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pills__item--active{color:var(--ifm-pills-color-active)}.pills__item--active,.pills__item:not(.pills__item--active):hover{background:var(--ifm-pills-color-background-active)}.pills--block{justify-content:stretch}.pills--block .pills__item{flex-grow:1;text-align:center}.tabs{color:var(--ifm-tabs-color);display:flex;margin-bottom:0;overflow-x:auto;padding-left:0}.tabs__item{border-bottom:3px solid #0000;border-radius:var(--ifm-global-radius);cursor:pointer;display:inline-flex;padding:var(--ifm-tabs-padding-vertical) var(--ifm-tabs-padding-horizontal);transition:background-color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.tabs__item--active{border-bottom-color:var(--ifm-tabs-color-active-border);border-bottom-left-radius:0;border-bottom-right-radius:0;color:var(--ifm-tabs-color-active)}.tabs__item:hover{background-color:var(--ifm-hover-overlay)}.tabs--block{justify-content:stretch}.tabs--block .tabs__item{flex-grow:1;justify-content:center}html[data-theme=dark]{--ifm-color-scheme:dark;--ifm-color-emphasis-0:var(--ifm-color-gray-1000);--ifm-color-emphasis-100:var(--ifm-color-gray-900);--ifm-color-emphasis-200:var(--ifm-color-gray-800);--ifm-color-emphasis-300:var(--ifm-color-gray-700);--ifm-color-emphasis-400:var(--ifm-color-gray-600);--ifm-color-emphasis-600:var(--ifm-color-gray-400);--ifm-color-emphasis-700:var(--ifm-color-gray-300);--ifm-color-emphasis-800:var(--ifm-color-gray-200);--ifm-color-emphasis-900:var(--ifm-color-gray-100);--ifm-color-emphasis-1000:var(--ifm-color-gray-0);--ifm-background-color:#1b1b1d;--ifm-background-surface-color:#242526;--ifm-hover-overlay:#ffffff0d;--ifm-color-content:#e3e3e3;--ifm-color-content-secondary:#fff;--ifm-breadcrumb-separator-filter:invert(64%) sepia(11%) saturate(0%) hue-rotate(149deg) brightness(99%) contrast(95%);--ifm-code-background:#ffffff1a;--ifm-scrollbar-track-background-color:#444;--ifm-scrollbar-thumb-background-color:#686868;--ifm-scrollbar-thumb-hover-background-color:#7a7a7a;--ifm-table-stripe-background:#ffffff12;--ifm-toc-border-color:var(--ifm-color-emphasis-200);--ifm-color-primary-contrast-background:#102445;--ifm-color-primary-contrast-foreground:#ebf2fc;--ifm-color-secondary-contrast-background:#474748;--ifm-color-secondary-contrast-foreground:#fdfdfe;--ifm-color-success-contrast-background:#003100;--ifm-color-success-contrast-foreground:#e6f6e6;--ifm-color-info-contrast-background:#193c47;--ifm-color-info-contrast-foreground:#eef9fd;--ifm-color-warning-contrast-background:#4d3800;--ifm-color-warning-contrast-foreground:#fff8e6;--ifm-color-danger-contrast-background:#4b1113;--ifm-color-danger-contrast-foreground:#ffebec;--primary:#db78fc;--background:var(--dark-grey);--text-color:var(--white);--hero-background:var(--black);--kinda-transparent:#000000d9;--hyperlink:var(--primary);--feature-img-shadow:4px 4px 6px #00000080,-2px -2px 4px #0006;--footer-color:var(--black);--footer-text-color:var(--white);--heading-shadow:-5px 4px 0px #000;--sub-heading-shadow:-3px 3px 1px #000}:root{--docusaurus-progress-bar-color:var(--ifm-color-primary);--bright-white:#fff;--white:#f7f7f7;--pale-grey:#e9e9e8;--mid-grey:#a9a9a9;--dark-grey:#18191a;--black:#121212;--pitch-black:#000;--info:#35c9fa;--success:#8f8;--warning:#ece715;--danger:#f80363;--accent-green:#41ef90;--accent-green-dark:#1e9554;--accent-blue:#5c85f7;--accent-blue-dark:#3d48dd;--accent-purple:#9083ed;--accent-purple-dark:#6b3cd6;--accent-pink:#db78fc;--accent-pink-dark:#b83ddd;--accent-yellow:#dcff5a;--accent-yellow-dark:#ceb73f;--ifm-color-primary:var(--background);--ifm-color-primary-dark:#21af90;--ifm-color-primary-darker:#1fa588;--ifm-color-primary-darkest:#1a8870;--ifm-color-primary-light:#46cbae;--ifm-color-primary-lighter:#66d4bd;--ifm-color-primary-lightest:#92e0d0;--ifm-code-font-size:95%;--ifm-navbar-background-color:var(--hero-background);--ifm-navbar-link-color:var(--text-color);--ifm-breadcrumb-color-active:var(--text-color);--docusaurus-tag-list-border:var(--ifm-color-emphasis-300);--docusaurus-announcement-bar-height:auto;--docusaurus-collapse-button-bg:#0000;--docusaurus-collapse-button-bg-hover:#0000001a;--doc-sidebar-width:300px;--doc-sidebar-hidden-width:30px}#nprogress .bar{background:var(--docusaurus-progress-bar-color);height:2px;left:0;position:fixed;top:0;width:100%;z-index:1031}#nprogress .peg{box-shadow:0 0 10px var(--docusaurus-progress-bar-color),0 0 5px var(--docusaurus-progress-bar-color);height:100%;opacity:1;position:absolute;right:0;transform:rotate(3deg) translateY(-4px);width:100px}h1{font-family:Racing Sans One,mono;font-size:4rem}button,h2,h3,h4,h5{font-family:Roboto Mono,monospace}h3{font-size:2rem}button,h4,h5{font-size:1.2rem}a,body,div,p,section,span,ul li{font-family:Poppins,sans-serif;font-size:1.2rem}.link2_y3x6,.link_ecgS,code span{font-family:Roboto Mono,monospace}.hero{--ifm-hero-background-color:var(--hero-background);--ifm-hero-text-color:var(--text-color)}.footer{--ifm-footer-background-color:var(--footer-color);--ifm-footer-color:var(--footer-text-color);--ifm-footer-link-color:var(--footer-text-color);--ifm-footer-title-color:var(--footer-text-color);--ifm-footer-link-hover-color:var(--primary);--ifm-link-color:var(--footer-text-color);--ifm-link-hover-color:var(--primary)}.footer .footer__link-item,.sidebar-ad #carbonads .carbon-text,article div.markdown a,article div.markdown code span,article div.markdown li,article div.markdown ol,article div.markdown p,article div.markdown ul,ul.table-of-contents li a{font-size:1rem}.footer .footer__copyright,.footer .footer__copyright a{font-family:Roboto Mono,monospace;font-size:.9rem;font-weight:700;opacity:.9}.docusaurus-highlight-code-line{background-color:#0000001a;display:block;margin:0 calc(var(--ifm-pre-padding)*-1);padding:0 var(--ifm-pre-padding)}nav.navbar a.navbar__brand .navbar__logo,nav.navbar a.navbar__brand b{transition:.3s}nav.navbar a.navbar__brand:hover .navbar__logo{transform:scale(1.1) rotate(-3deg)}nav.navbar a.navbar__brand:hover b{color:#00eed6}nav.navbar a.navbar__brand b{font-size:1.5rem;font-weight:700}.authorLink_YWtN:hover,nav.navbar .navbar__link--active,nav.navbar .navbar__link:hover{color:var(--primary)}.fetchErrorCard_M8w3 a:hover,.meta_GplF a:hover,.pagination-nav__label,html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg .pagination-nav__label{color:var(--text-color)}li a,li a:hover,p a,p a:hover,ul.table-of-contents li a:hover,ul.table-of-contents li a:hover code{color:var(--hyperlink)}h1,h2,h3,h4,h5{cursor:default}article div.markdown{--ifm-h1-font-size:3.0rem;--ifm-h2-font-size:2.5rem;--ifm-h3-font-size:1.4rem}article div.markdown p img{border-radius:4px}.sponsor_JDQ1,.sponsor_JDQ1 img,article div.markdown table>tr>td>a>img{border-radius:6px}ul.table-of-contents li a.table-of-contents__link--active,ul.table-of-contents li a.table-of-contents__link--active code{color:var(--hyperlink);font-weight:700}ul.menu__list{margin-bottom:1rem}ul.menu__list li.menu__list-item a.active,ul.menu__list li.menu__list-item a.menu__link--active{color:var(--primary);font-weight:700}code span{font-size:.9rem!important}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg{padding:0 1rem!important}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg div>div.col[class*=docItemCol]{background:var(--bright-white);border-right:1px solid var(--ifm-toc-border-color);padding:1rem}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg div>div.col[class*=docItemCol] ul li a{font-weight:700}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg .table-of-contents__left-border{border-left:none}#carbonads *{line-height:normal;margin:initial;padding:initial}#carbonads{--carbon-font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",Helvetica,Arial,sans-serif;--carbon-font-size:14px;--carbon-padding:1.5ch;--carbon-max-char:20ch;--carbon-bg-primary:#fafafa;--carbon-bg-secondary:#ebebeb;--carbon-text-color:#333;font-family:var(--carbon-font-family);font-size:var(--carbon-font-size);margin:0 auto;width:fit-content;z-index:10}#carbonads>span{background-color:var(--carbon-bg-primary);box-shadow:0 0 1px rgba(0,0,0,.085),0 0 2px rgba(0,0,0,.085),0 0 4px rgba(0,0,0,.085),0 0 8px rgba(0,0,0,.085);display:flex;flex-direction:column;gap:var(--carbon-padding);max-inline-size:calc(130px + var(--carbon-max-char) + 8ch);min-inline-size:130px;padding:var(--carbon-padding)}#carbonads .carbon-wrap{display:flex;flex-wrap:wrap;gap:1.5ex}#carbonads .carbon-img{flex:0 0 130px}#carbonads .carbon-img img,.cursor_eG29 .hideAction_vcyE>svg,.tocCollapsibleContent_vkbj a{display:block}#carbonads .carbon-text{flex-basis:var(--carbon-max-char);flex-grow:1;line-height:1.4;text-align:left}#carbonads .carbon-poweredby{background:var(--carbon-bg-secondary);font-size:.6em;font-weight:600;letter-spacing:.2ch;line-height:1.4;padding:6px 8px;text-align:center;text-transform:uppercase}.sidebar-ad{--carbon-bg-primary:var(--background,#18191a);--carbon-bg-secondary:#282a36;--carbon-text-color:#e6e6e6}.sidebar-ad #carbonads{margin:.5rem}.sidebar-ad #carbonads .carbon-img img,.sidebar-ad .avatar__photo-link{border-radius:5px}.mdxPageWrapper_j9I6,.sidebar-ad #carbonads .carbon-wrap{justify-content:center}.banner_woPo,.tagWithCount_h2kH{align-items:center;display:flex}.sidebar-ad #carbonads>span{box-shadow:none}.sidebar-ad #carbonads .carbon-poweredby{border-radius:5px;font-size:.7rem}::selection{background:var(--primary);color:var(--background);text-shadow:none}@keyframes a{0%{opacity:0;transform:translateY(-100%)}to{opacity:1;transform:translateY(0)}}@keyframes b{0%{opacity:1}to{opacity:0;transform:translateY(-100%)}}.banner_woPo{animation:.3s ease-out a;background:var(--accent-green);color:var(--bright-white);flex-wrap:wrap;font-size:.95rem;font-weight:600;gap:.25rem 1.25rem;justify-content:center;padding:.25rem 1rem;position:relative;width:100%;z-index:100}.dismissing_ZrNM{animation:.25s ease-in forwards b}.link2_y3x6,.link_ecgS{color:var(--black);font-size:.95rem;font-weight:400;transition:.2s}.link2_y3x6.link2_y3x6,.link_ecgS.link2_y3x6{opacity:.85}.link2_y3x6.link2_y3x6:hover,.link_ecgS.link2_y3x6:hover{opacity:1;transform:scale(1.03)}.closeBtn_fC0A{background:none;border:none;border-radius:4px;color:var(--black);cursor:pointer;font-size:1.5rem;line-height:1;opacity:.8;padding:0 .25rem .1rem;position:absolute;right:1rem;transition:.2s}.closeBtn_fC0A:hover{background:#1d8c4f;opacity:1}.tag_zVej{border:1px solid var(--docusaurus-tag-list-border);transition:border var(--ifm-transition-fast)}.tag_zVej:hover{--docusaurus-tag-list-border:var(--ifm-link-color);-webkit-text-decoration:none;text-decoration:none}.tagRegular_sFm0{border-radius:var(--ifm-global-radius);font-size:90%;padding:.2rem .5rem .3rem}.tagWithCount_h2kH{border-left:0;padding:0 .5rem 0 1rem;position:relative}.tagWithCount_h2kH:after,.tagWithCount_h2kH:before{border:1px solid var(--docusaurus-tag-list-border);content:"";position:absolute;top:50%;transition:inherit}.tagWithCount_h2kH:before{border-bottom:0;border-right:0;height:1.18rem;right:100%;transform:translate(50%,-50%) rotate(-45deg);width:1.18rem}.tagWithCount_h2kH:after{border-radius:50%;height:.5rem;left:0;transform:translateY(-50%);width:.5rem}.tagWithCount_h2kH span{background:var(--ifm-color-secondary);border-radius:var(--ifm-global-radius);color:var(--ifm-color-black);font-size:.7rem;line-height:1.2;margin-left:.3rem;padding:.1rem .4rem}.tags_jXut{display:inline}.tag_QGVx{display:inline-block;margin:0 .4rem .5rem 0}.backToTopButton_sjWU{background-color:var(--ifm-color-emphasis-200);border-radius:50%;bottom:1.3rem;box-shadow:var(--ifm-global-shadow-lw);height:3rem;opacity:0;position:fixed;right:1.3rem;transform:scale(0);transition:all var(--ifm-transition-fast) var(--ifm-transition-timing-default);visibility:hidden;width:3rem;z-index:calc(var(--ifm-z-index-fixed) - 1)}.backToTopButton_sjWU:after{background-color:var(--ifm-color-emphasis-1000);content:" ";display:inline-block;height:100%;-webkit-mask:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem no-repeat;mask:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem no-repeat;width:100%}.backToTopButtonShow_xfvO{opacity:1;transform:scale(1);visibility:visible}body:not(.navigation-with-keyboard) :not(input):focus{outline:0}#__docusaurus-base-url-issue-banner-container,.docSidebarContainer_YfHR,.hideAction_vcyE>svg,.navbarSearchContainer_Bca1:empty,.sidebarLogo_isFc,.themedComponent_mlkZ,.toggleIcon_g3eP,html[data-announcement-bar-initially-dismissed=true] .announcementBar_mb4j{display:none}.skipToContent_fXgn{background-color:var(--ifm-background-surface-color);color:var(--ifm-color-emphasis-900);left:100%;padding:calc(var(--ifm-global-spacing)/2) var(--ifm-global-spacing);position:fixed;top:1rem;z-index:calc(var(--ifm-z-index-fixed) + 1)}.skipToContent_fXgn:focus{box-shadow:var(--ifm-global-shadow-md);left:1rem}.closeButton_CVFx{line-height:0;padding:0}.content_knG7{font-size:85%;padding:5px 0;text-align:center}.content_knG7 a{color:inherit}.announcementBar_mb4j{align-items:center;background-color:var(--ifm-color-white);border-bottom:1px solid var(--ifm-color-emphasis-100);color:var(--ifm-color-black);display:flex;height:var(--docusaurus-announcement-bar-height)}.announcementBarPlaceholder_vyr4{flex:0 0 10px}.announcementBarClose_gvF7{align-self:stretch;flex:0 0 30px}.toggle_vylO{height:2rem;width:2rem}.toggleButton_gllP{align-items:center;border-radius:50%;display:flex;height:100%;justify-content:center;transition:background var(--ifm-transition-fast);width:100%}.toggleButton_gllP:hover{background:var(--ifm-color-emphasis-200)}[data-theme-choice=dark] .darkToggleIcon_wfgR,[data-theme-choice=light] .lightToggleIcon_pyhR,[data-theme-choice=system] .systemToggleIcon_QzmC,[data-theme=dark] .themedComponent--dark_xIcU,[data-theme=light] .themedComponent--light_NVdE,html:not([data-theme]) .themedComponent--light_NVdE{display:initial}.toggleButtonDisabled_aARS{cursor:not-allowed}.darkNavbarColorModeToggle_X3D1:hover{background:var(--ifm-color-gray-800)}[data-theme=dark]:root{--docusaurus-collapse-button-bg:#ffffff0d;--docusaurus-collapse-button-bg-hover:#ffffff1a}.collapseSidebarButton_PEFL{display:none;margin:0}.categoryLinkLabel_W154,.linkLabel_WmDU{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden}.iconExternalLink_nPIU{margin-left:.3rem}.linkLabel_WmDU{line-clamp:2;-webkit-line-clamp:2}.categoryLink_byQd{overflow:hidden}.menu__link--sublist-caret:after{margin-left:var(--ifm-menu-link-padding-vertical)}.categoryLinkLabel_W154{flex:1;line-clamp:2;-webkit-line-clamp:2}.docMainContainer_TBSr,.docRoot_UBD9{display:flex;width:100%}.docsWrapper_hBAB{display:flex;flex:1 0 auto}.notFoundPage_Wsy5{align-items:center;background:var(--hero-background);display:flex;justify-content:center;min-height:calc(100vh - 4rem);overflow:hidden;position:relative}.container_a6qs{padding:2rem;position:relative;text-align:center;z-index:2}.glitchCode_Pt6F{animation:3s ease-in-out infinite d;color:var(--primary);font-family:Racing Sans One,monospace;font-size:10rem;font-weight:900;line-height:1;position:relative;text-shadow:0 0 20px color-mix(in srgb,var(--primary) 40%,#0000),0 0 60px color-mix(in srgb,var(--primary) 20%,#0000)}.badge_qP1g,.count_Ad5j,.monthHeader_QFki,.sha_JsPS,.tagCard_SluW .entryTitle_RUqs,.tagPill_IKwa{font-family:Roboto Mono,monospace}.title_B9uO{color:var(--text-color);font-size:2rem;font-weight:700;margin:1rem 0 .5rem}.subtitle_Z226{color:var(--mid-grey);font-size:1.1rem;line-height:1.6;margin-bottom:2.5rem;margin-left:auto;margin-right:auto;max-width:480px}.navLinks_NElv{display:flex;flex-wrap:wrap;gap:1rem;justify-content:center;margin-top:1rem}.navLink_KmGT{align-items:center;border-radius:8px;cursor:pointer;display:inline-flex;font-size:1rem;font-weight:600;gap:.5rem;padding:.75rem 1.5rem;transition:.25s}.navLink_KmGT:hover{transform:translateY(-2px)}.primaryLink_UP3n{background:var(--primary);color:var(--hero-background)}.primaryLink_UP3n:hover{box-shadow:0 4px 20px color-mix(in srgb,var(--primary) 50%,#0000);color:var(--hero-background)}.secondaryLink_VStI{background:#0000;border:2px solid var(--primary);color:var(--primary)}.secondaryLink_VStI:hover{background:color-mix(in srgb,var(--primary) 10%,#0000);color:var(--primary)}.particles_xv3x{inset:0;overflow:hidden;position:absolute;z-index:1}.particle_Q0bD{animation:linear infinite c;background:var(--primary);border-radius:50%;height:4px;opacity:.3;position:absolute;width:4px}.particle_Q0bD:first-child{animation-delay:-3s;animation-duration:15s;height:6px;left:59%;opacity:.37;top:21%;width:6px}.particle_Q0bD:nth-child(2){animation-delay:-2s;animation-duration:13s;height:6px;left:69%;opacity:.3;top:22%;width:6px}.particle_Q0bD:nth-child(3){animation-delay:-1s;animation-duration:16s;height:3px;left:44%;opacity:.18;top:54%;width:3px}.particle_Q0bD:nth-child(4){animation-delay:-4s;animation-duration:14s;height:3px;left:29%;opacity:.39;top:63%;width:3px}.particle_Q0bD:nth-child(5){animation-delay:-5s;animation-duration:16s;height:3px;left:63%;opacity:.29;top:50%;width:3px}.particle_Q0bD:nth-child(6){animation-delay:-3s;animation-duration:15s;height:6px;left:70%;opacity:.39;top:78%;width:4px}.particle_Q0bD:nth-child(7){animation-delay:-5s;animation-duration:13s;height:4px;left:20%;opacity:.36;top:41%;width:5px}.particle_Q0bD:nth-child(8){animation-delay:-4s;animation-duration:12s;height:5px;left:2%;opacity:.32;top:63%;width:4px}.particle_Q0bD:nth-child(9){animation-delay:-5s;animation-duration:13s;height:3px;left:83%;opacity:.24;top:64%;width:4px}.particle_Q0bD:nth-child(10){animation-delay:-5s;animation-duration:9s;height:3px;left:90%;opacity:.39;top:43%;width:5px}.particle_Q0bD:nth-child(11){animation-delay:-4s;animation-duration:16s;height:6px;left:59%;opacity:.15;top:42%;width:5px}.particle_Q0bD:nth-child(12){animation-delay:-5s;animation-duration:15s;height:6px;left:34%;opacity:.39;top:93%;width:5px}.particle_Q0bD:nth-child(13){animation-delay:-1s;animation-duration:20s;height:4px;left:75%;opacity:.11;top:76%;width:3px}.particle_Q0bD:nth-child(14){animation-delay:-3s;animation-duration:18s;height:5px;left:97%;opacity:.18;top:95%;width:5px}.particle_Q0bD:nth-child(15){animation-delay:-2s;animation-duration:12s;height:3px;left:28%;opacity:.27;top:57%;width:6px}.particle_Q0bD:nth-child(16){animation-delay:-5s;animation-duration:14s;height:6px;left:3%;opacity:.2;top:90%;width:6px}.particle_Q0bD:nth-child(17){animation-delay:-3s;animation-duration:13s;height:4px;left:76%;opacity:.31;top:1%;width:3px}.particle_Q0bD:nth-child(18){animation-delay:-5s;animation-duration:12s;height:6px;left:37%;opacity:.3;top:1%;width:5px}.particle_Q0bD:nth-child(19){animation-delay:-2s;animation-duration:20s;height:4px;left:27%;opacity:.22;top:77%;width:4px}.particle_Q0bD:nth-child(20){animation-delay:-2s;animation-duration:12s;height:5px;left:65%;opacity:.18;top:67%;width:6px}@keyframes c{0%,to{opacity:0;transform:translateY(0) translateX(0)}10%,90%{opacity:.3}50%{transform:translateY(-100px) translateX(30px)}}@keyframes d{0%,to{opacity:1}50%{opacity:.85}}.dropdownNavbarItemMobile_J0Sd{cursor:pointer}.iconLanguage_nlXk{margin-right:5px;vertical-align:text-bottom}.searchBar_RVTs .dropdownMenu_qbY6{background:var(--search-local-modal-background,#f5f6f7);border-radius:6px;box-shadow:var(--search-local-modal-shadow,inset 1px 1px 0 0 #ffffff80,0 3px 8px 0 #555a64);left:auto!important;margin-top:8px;padding:var(--search-local-spacing,12px);position:relative;right:0!important;width:var(--search-local-modal-width,560px)}.searchInput_YFbd:focus{outline:2px solid var(--search-local-input-active-border-color,var(--ifm-color-primary));outline-offset:0}div.ask-ai,html[data-theme=dark] div.ask-ai{--ask-ai-primary:var(--ifm-color-primary);--ask-ai-primary-hover:var(--ifm-color-primary-light);--ask-ai-foreground:var(--ifm-color-content);--ask-ai-border:var(--ifm-color-emphasis-300);--ask-ai-error:var(--ifm-color-danger);--ask-ai-button-bg:var(--ifm-color-emphasis-200)}.ask-ai{--ask-ai-background:var(--search-local-modal-background,#f5f6f7);--ask-ai-muted:var(--search-local-muted-color,#969faf)}html[data-theme=dark] .ask-ai{--ask-ai-background:var(--search-local-modal-background,var(--ifm-background-color));--ask-ai-muted:var(--search-local-muted-color,var(--ifm-color-secondary-darkest))}html[data-theme=dark] .searchBar_RVTs .dropdownMenu_qbY6{background:var(--search-local-modal-background,var(--ifm-background-color));box-shadow:var(--search-local-modal-shadow,inset 1px 1px 0 0 #2c2e40,0 3px 8px 0 #000309)}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2{align-items:center;background:var(--search-local-hit-background,#fff);border-radius:4px;box-shadow:var(--search-local-hit-shadow,0 1px 3px 0 #d4d9e1);color:var(--search-local-hit-color,#444950);cursor:pointer;display:flex;flex-direction:row;height:var(--search-local-hit-height,56px);padding:0 var(--search-local-spacing,12px);width:100%}html[data-theme=dark] .dropdownMenu_qbY6 .suggestion_fB_2{background:var(--search-local-hit-background,var(--ifm-color-emphasis-100));box-shadow:var(--search-local-hit-shadow,none);color:var(--search-local-hit-color,var(--ifm-font-color-base))}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2:not(:last-child){margin-bottom:4px}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2.cursor_eG29{background-color:var(--search-local-highlight-color,var(--ifm-color-primary))}.hitFooter_E9YW a,.hitIcon_a7Zy,.hitPath_ieM4,.hitTree_kk6K,.noResultsIcon_EBY5{color:var(--search-local-muted-color,#969faf)}html[data-theme=dark] .hitIcon_a7Zy,html[data-theme=dark] .hitPath_ieM4,html[data-theme=dark] .hitTree_kk6K,html[data-theme=dark] .noResultsIcon_EBY5{color:var(--search-local-muted-color,var(--ifm-color-secondary-darkest))}.hitTree_kk6K{align-items:center;display:flex}.hitTree_kk6K>svg{height:var(--search-local-hit-height,56px);opacity:.5;width:24px}.hitIcon_a7Zy,.hitTree_kk6K>svg{stroke-width:var(--search-local-icon-stroke-width,1.4)}.hitAction_NqkB,.hitIcon_a7Zy{height:20px;width:20px}.hitWrapper_sAK8{display:flex;flex:1 1 auto;flex-direction:column;font-weight:500;justify-content:center;margin:0 8px;overflow-x:hidden;width:80%}.hitWrapper_sAK8 mark{background:none;color:var(--search-local-highlight-color,var(--ifm-color-primary))}.hitTitle_vyVt{font-size:.9em}.hitPath_ieM4{font-size:.75em}.hitPath_ieM4,.hitTitle_vyVt{overflow-x:hidden;text-overflow:ellipsis;white-space:nowrap}.noResults_l6Q3{align-items:center;display:flex;flex-direction:column;justify-content:center;padding:var(--search-local-spacing,12px) 0}.noResultsIcon_EBY5{margin-bottom:var(--search-local-spacing,12px)}.hitFooter_E9YW{font-size:.85em;margin-top:var(--search-local-spacing,12px);text-align:center}.suggestion_fB_2.cursor_eG29,.suggestion_fB_2.cursor_eG29 .hitIcon_a7Zy,.suggestion_fB_2.cursor_eG29 .hitPath_ieM4,.suggestion_fB_2.cursor_eG29 .hitTree_kk6K,.suggestion_fB_2.cursor_eG29 mark{color:var(--search-local-hit-active-color,var(--ifm-color-white))!important}.searchBarContainer_NW3z{margin-left:16px}.searchBarContainer_NW3z .searchBarLoadingRing_YnHq{display:none;left:10px;position:absolute;top:6px}.searchBarContainer_NW3z .searchClearButton_qk4g{background:none;border:none;line-height:1rem;padding:0;position:absolute;right:.8rem;top:50%;transform:translateY(-50%)}.navbar__search{position:relative}.searchIndexLoading_EJ1f .navbar__search-input{background-image:none}.searchHintContainer_Pkmr{align-items:center;display:flex;gap:4px;height:100%;justify-content:center;pointer-events:none;position:absolute;right:10px;top:0}.searchHint_iIMx{background-color:var(--ifm-navbar-search-input-background-color);border:1px solid var(--ifm-color-emphasis-500);box-shadow:inset 0 -1px 0 var(--ifm-color-emphasis-500);color:var(--ifm-navbar-search-input-placeholder-color)}html[dir=rtl] .searchHintContainer_Pkmr{left:10px;right:auto}html[dir=rtl] .searchBarContainer_NW3z .searchClearButton_qk4g{left:.8rem;right:auto}html[dir=rtl] .searchBarContainer_NW3z .searchBarLoadingRing_YnHq{left:auto;right:10px}html[dir=rtl] .navbar__search-input{padding:0 2.25em 0 .5em}.loadingRing_RJI3{display:inline-block;height:20px;opacity:var(--search-local-loading-icon-opacity,.5);position:relative;width:20px}.loadingRing_RJI3 div{animation:1.2s cubic-bezier(.5,0,.5,1) infinite e;border:2px solid var(--search-load-loading-icon-color,var(--ifm-navbar-search-input-color));border-color:var(--search-load-loading-icon-color,var(--ifm-navbar-search-input-color)) #0000 #0000 #0000;border-radius:50%;display:block;height:16px;margin:2px;position:absolute;width:16px}.loadingRing_RJI3 div:first-child{animation-delay:-.45s}.loadingRing_RJI3 div:nth-child(2){animation-delay:-.3s}.loadingRing_RJI3 div:nth-child(3){animation-delay:-.15s}@keyframes e{0%{transform:rotate(0)}to{transform:rotate(1turn)}}.navbarHideable_m1mJ{transition:transform var(--ifm-transition-fast) ease}.navbarHidden_jGov{transform:translate3d(0,calc(-100% - 2px),0)}.errorBoundaryError_a6uf{color:red;white-space:pre-wrap}.badge_qP1g,.commitMeta_teNK,.commitMsg_XeKn,.name_mmMI{white-space:nowrap}.errorBoundaryFallback_VBag{color:red;padding:.55rem}.iconEdit_Z9Sw{margin-right:.3em;vertical-align:sub}.navbar__items--right>:last-child{padding-right:0}.lastUpdated_JAkA{font-size:smaller;font-style:italic;margin-top:.2rem}.tocCollapsibleButton_TO0P{align-items:center;display:flex;font-size:inherit;justify-content:space-between;padding:.4rem .8rem;width:100%}.tocCollapsibleButton_TO0P:after{background:var(--ifm-menu-link-sublist-icon) 50% 50%/2rem 2rem no-repeat;content:"";filter:var(--ifm-menu-link-sublist-icon-filter);height:1.25rem;transform:rotate(180deg);transition:transform var(--ifm-transition-fast);width:1.25rem}.tocCollapsibleButtonExpanded_MG3E:after,.tocCollapsibleExpanded_sAul{transform:none}.tocCollapsible_ETCw{background-color:var(--ifm-menu-color-background-active);border-radius:var(--ifm-global-radius);margin:1rem 0}.tocCollapsibleContent_vkbj>ul{border-left:none;border-top:1px solid var(--ifm-color-emphasis-300);font-size:15px;padding:.2rem 0}.tocCollapsibleContent_vkbj ul li{margin:.4rem .8rem}.footerLogoLink_BH7S{opacity:.5;transition:opacity var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.footerLogoLink_BH7S:hover,.hash-link:focus,:hover>.hash-link,header.heroBanner_qw_p img.starButton_Vbnu:hover{opacity:1}.anchorTargetStickyNavbar_Vzrq{scroll-margin-top:calc(var(--ifm-navbar-height) + .5rem)}.anchorTargetHideOnScrollNavbar_vjPI{scroll-margin-top:.5rem}.hash-link{opacity:0;padding-left:.5rem;transition:opacity var(--ifm-transition-fast);-webkit-user-select:none;user-select:none}.hash-link:before{content:"#"}.mainWrapper_z2l0{display:flex;flex:1 0 auto;flex-direction:column}.docusaurus-mt-lg{margin-top:3rem}#__docusaurus{display:flex;flex-direction:column;min-height:100%}.searchContextInput_mXoe,.searchQueryInput_CFBF{background:var(--ifm-background-color);border:var(--ifm-global-border-width) solid var(--ifm-color-content-secondary);border-radius:var(--ifm-global-radius);color:var(--ifm-font-color-base);font-size:var(--ifm-font-size-base);margin-bottom:1rem;padding:.5rem;width:100%}.monthHeader_QFki,.updatesPage_z4Av,header.heroBanner_qw_p,main.docs{background:var(--hero-background)}.releaseCard_aa3v,.tagCard_SluW{border-radius:8px;box-shadow:var(--feature-img-shadow)}.searchResultItem_U687{border-bottom:1px solid #dfe3e8;padding:1rem 0}.searchResultItemPath_uIbk{color:var(--ifm-color-content-secondary);font-size:.8rem;margin:.5rem 0 0}.searchResultItemSummary_oZHr{font-style:italic;margin:.5rem 0 0}.updatesPage_z4Av{min-height:80vh;padding:2rem 1rem 4rem}.pageHeader_g4VR{margin-bottom:2.5rem;text-align:center}.pageHeader_g4VR h1{color:var(--text-color);font-family:Racing Sans One,cursive;font-size:2.5rem;margin-bottom:.25rem}.pageHeader_g4VR p{color:var(--mid-grey);font-family:Raleway,sans-serif;font-size:1.1rem;margin:0}.timeline_WTQQ{margin:0 auto;max-width:1000px;padding-left:2.5rem;position:relative;width:80%}.timeline_WTQQ:before{background:var(--mid-grey);bottom:0;content:"";left:0;opacity:.4;position:absolute;top:0;width:2px}.monthHeader_QFki{color:var(--mid-grey);font-size:1rem;margin:0;padding:.75rem 0 .5rem;position:sticky;top:60px;z-index:1}.entry_mTt2{margin-bottom:1rem;position:relative}.entry_mTt2:before{border-radius:50%;content:"";position:absolute;z-index:1}.releaseEntry_ZG2V:before{background:var(--accent-pink);height:14px;left:-2.5rem;margin-left:-6px;top:1rem;width:14px}.tagEntry_ORGt:before{background:var(--accent-blue);height:10px;left:-2.5rem;margin-left:-4px;top:1rem;width:10px}.commitEntry_JtMn:before{background:var(--mid-grey);height:6px;left:-2.5rem;margin-left:-2px;top:.9rem;width:6px}.commitRow_RIZe:hover,.releaseCard_aa3v,.tagCard_SluW{background:var(--background)}.releaseCard_aa3v{border-left:4px solid var(--accent-pink);padding:1.25rem 1.5rem}.releaseCard_aa3v .entryTitle_RUqs{align-items:center;color:var(--text-color);display:flex;flex-wrap:wrap;font-family:Roboto Mono,monospace;font-size:1.3rem;gap:.5rem;margin:0 0 .5rem}.tagCard_SluW{border-left:4px solid var(--accent-blue);padding:1rem 1.25rem}.badgeRelease_agcf,.button.color-btn.btn-pink{background:var(--accent-pink)}.tagCard_SluW .entryTitle_RUqs{font-size:1.1rem;gap:.5rem;margin:0 0 .25rem}.commitRow_RIZe,.tagCard_SluW .entryTitle_RUqs{align-items:center;color:var(--text-color);display:flex}.commitLink_QhHv,.meta_GplF{align-items:center;gap:.75rem}.commitRow_RIZe{border-bottom:1px solid #80808026;gap:.75rem;padding:.5rem 0;transition:background .15s}.commitLink_QhHv{display:flex;overflow:hidden}.commitLink_QhHv,.commitLink_QhHv:hover,.tile_CRAN,.tile_CRAN:hover{color:var(--text-color);-webkit-text-decoration:none;text-decoration:none}.fetchErrorCard_M8w3 a,.keepReading_G_Ux,.meta_GplF a,main.docs p.see-repo-note a{-webkit-text-decoration:underline;text-decoration:underline}.badge_qP1g{border-radius:4px;color:#fff;display:inline-block;font-size:.7rem;line-height:1.4;padding:.1rem .5rem}.badgeTag_gC_o{background:var(--accent-blue)}.tagPill_IKwa{border:1px solid var(--mid-grey);border-radius:4px;color:var(--mid-grey);font-size:.8rem;padding:.05rem .4rem}.avatarSmall_luV7,.avatar_Vzby,.contributorAvatar_eGpt img{border-radius:50%;vertical-align:middle}.errorMessage_lztz,.fetchErrorCard_M8w3{box-shadow:var(--feature-img-shadow);padding:1.25rem 1.5rem;text-align:center}.bodyPreview_iKaD{color:var(--mid-grey);display:-webkit-box;font-size:.9rem;-webkit-line-clamp:3;margin:.25rem 0 .5rem;-webkit-box-orient:vertical;line-height:1.5;overflow:hidden}.meta_GplF,.sha_JsPS{color:var(--mid-grey);font-size:.85rem}.meta_GplF{display:flex;flex-wrap:wrap}.meta_GplF a{color:var(--mid-grey)}.linksList_CKaa img,.meta_GplF a .providerIcon_NdOm{margin-right:.25rem;vertical-align:middle;width:1.35rem}.avatar_Vzby{height:24px;width:24px}.authorInfo_pi2M{align-items:center;display:flex;gap:.35rem}.sha_JsPS{flex-shrink:0}.commitMsg_XeKn{font-size:.95rem;max-width:500px;overflow:hidden}.authorLink_YWtN{color:var(--text-color);opacity:.75;transition:.2s}.authorLink_YWtN:hover .avatarSmall_luV7{transform:scale(1.08)}.avatarSmall_luV7{height:18px;margin-right:.2rem;transition:.2s;width:18px}.errorMessage_lztz,.fetchErrorCard_M8w3,.skeleton_NVtD{border-radius:8px;background:var(--background)}.commitMeta_teNK{align-items:center;color:var(--mid-grey);display:flex;flex-shrink:0;font-size:.8rem;gap:.3rem;margin-left:auto}.contributors_QyJV{align-items:center;display:flex;flex-wrap:wrap;gap:.35rem;margin-bottom:.5rem}.contributorAvatar_eGpt{display:block;flex-shrink:0;height:26px;width:26px}.contributorAvatar_eGpt img{height:26px;transition:transform .2s;width:26px}.contributorAvatar_eGpt:hover img{transform:scale(1.15)}.skeleton_NVtD{animation:1.5s ease-in-out infinite f;height:80px;margin-bottom:1rem}@keyframes f{0%,to{opacity:.4}50%{opacity:1}}.loadMore_YatW{display:flex;justify-content:center;margin:2rem auto}.fetchErrorCard_M8w3{color:var(--mid-grey);font-size:.95rem;margin:1.5rem auto;max-width:1000px;width:80%}.fetchErrorCard_M8w3 a{color:var(--accent-blue)}.errorMessage_lztz{color:var(--mid-grey);font-size:1.1rem;margin:0 auto;width:fit-content}.errorMessage_lztz p{margin:.25rem 0}html[data-theme=light]{--heading-shadow:-2px 3px 2px #a2a1a1;--sub-heading-shadow:-1px 1px 1px #a2a1a1}header.heroBanner_qw_p{min-height:calc(100vh - 4rem);overflow:hidden;padding:4rem 1rem 2rem;position:relative;text-align:center}header.heroBanner_qw_p h1.heroTitle_CteR{font-family:Racing Sans One,mono;font-size:8rem;text-shadow:var(--heading-shadow)}header.heroBanner_qw_p h3.heroSubTitle_p8ER{text-shadow:var(--sub-heading-shadow)}header.heroBanner_qw_p img.starButton_Vbnu{opacity:.85;position:absolute;right:1rem;top:1rem;width:12rem}.buttons_LJn2{align-items:center;display:flex;justify-content:center;margin:1.5rem}.dashyDescription_iLL2{font-size:1.2rem;margin:1rem auto;max-width:680px;text-align:left}.dashyDescription_iLL2>p,.dashyDescription_iLL2>span{margin-left:auto;margin-right:auto;max-width:680px}.keepReading_G_Ux{cursor:pointer;font-size:1.2rem;margin-left:.5rem}.scrollDown_cjGa{bottom:0;cursor:pointer;display:flex;left:42%;margin:2rem auto;position:absolute;width:fit-content}.scrollDownText_HIIX{color:var(--text-color);font-size:1.2rem;font-weight:700;margin:0 1rem}.scrollDownIcon_UZ2V{width:2rem}.scrollDownIcon_UZ2V path{fill:var(--text-color)}.sponsorContainer_zSLg{display:flex;flex-direction:column;gap:1rem;margin:1.5rem auto;max-width:680px}.sponsor_JDQ1{background:var(--background);color:var(--text-color);display:flex;flex-direction:column-reverse;gap:.5rem;justify-content:left;padding:.5rem;width:100%}.sponsor_JDQ1 span{font-size:.9rem;text-align:center}.sponsor_JDQ1 span a{color:var(--text-color);font-size:.9rem}.button{border:none;border-radius:4px;color:var(--black);cursor:pointer;display:inline-block;display:flex;font-size:1.3rem;font-weight:700;margin:.5rem;min-width:12rem;padding:.75rem 1rem;position:relative;transition:none}.button svg{margin-right:.5rem;width:1.8rem}.button.btn-small{font-size:1rem;max-width:12rem;min-width:auto;padding:.5rem .75rem;width:100%}.button.btn-small svg{margin-right:.35rem;width:1.2rem}.button.color-btn:hover{color:rgba(0,0,0,.702);top:2px}.button.color-btn:active{box-shadow:none!important;color:rgba(0,0,0,.502);top:4px}.button.color-btn.btn-pink{box-shadow:0 4px var(--accent-pink-dark)}.button.color-btn.btn-pink:hover{box-shadow:0 2px var(--accent-pink-dark)}.button.color-btn.btn-purple{background:var(--accent-purple);box-shadow:0 4px var(--accent-purple-dark)}.button.color-btn.btn-purple:hover{box-shadow:0 2px var(--accent-purple-dark)}.button.color-btn.btn-blue{background:var(--accent-blue);box-shadow:0 4px var(--accent-blue-dark)}.button.color-btn.btn-blue:hover{box-shadow:0 2px var(--accent-blue-dark)}.button.color-btn.btn-green{background:var(--accent-green);box-shadow:0 4px var(--accent-green-dark)}.button.color-btn.btn-green:hover{box-shadow:0 2px var(--accent-green-dark)}.button.color-btn.btn-yellow{background:var(--accent-yellow);box-shadow:0 4px var(--accent-yellow-dark)}.button.color-btn.btn-yellow:hover{box-shadow:0 2px var(--accent-yellow-dark)}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0,.home-page-features-wrapper .feature .feature-half.assets,.statItem_at6_,.tile_CRAN:hover,main.docs{box-shadow:var(--feature-img-shadow)}main.docs{margin:1rem auto;max-width:1250px;padding:1rem}main.docs h1.docs-title{font-size:5rem;margin:0;text-align:center}main.docs p.see-repo-note{background:rgba(255,255,170,.839);border-radius:3px;display:none;padding:.25rem 1rem;width:fit-content}main.docs p.see-repo-note a,main.docs p.see-repo-note b,main.docs p.see-repo-note span{color:#000;font-size:1rem;font-style:italic}main.docs p.see-repo-note a{color:#000;font-size:1rem;font-weight:700}main.docs div.row1{display:flex;flex-direction:row}main.docs div.docs-links-section{display:flex;flex-wrap:wrap}main.docs div.docs-contents{color:var(--text-color);display:flex;flex-direction:column;justify-content:space-around}main.docs div.docs-contents h3.section-title{margin:1rem .5rem .5rem}main.docs div.docs-contents a{display:flex;flex-direction:column;margin:.5rem .75rem;min-height:5rem;padding:.25rem .5rem;width:calc(33% - 1.5rem)}main.docs div.docs-contents a p.name{font-size:1.4rem;margin:0;text-align:left}main.docs div.docs-contents a span.description{font-size:1rem;text-align:left;white-space:pre-wrap}.color-pink{--feature-color:var(--accent-pink)}.color-blue{--feature-color:var(--accent-blue)}.color-green{--feature-color:var(--accent-green)}.color-yellow{--feature-color:var(--accent-yellow)}.color-white{--feature-color:#fff}html[data-theme=light] .color-pink{--feature-color:var(--accent-pink-dark)}html[data-theme=light] .color-blue{--feature-color:var(--accent-purple-dark)}html[data-theme=light] .color-green{--feature-color:var(--accent-green-dark)}html[data-theme=light] .color-yellow{--feature-color:var(--accent-yellow-dark)}html[data-theme=light] .color-white{--feature-color:#000}.home-page-features-wrapper{display:flex;flex-direction:column;font-size:1.2rem}.home-page-features-wrapper .feature{align-items:center;display:flex;justify-content:center;margin:0;padding:.5rem}.home-page-features-wrapper .feature.align-left{background:var(--background);box-shadow:0 1px 4px #000;flex-direction:row}.home-page-features-wrapper .feature.align-right{background:var(--hero-background);box-shadow:0 -1px 4px rgba(0,0,0,.702);flex-direction:row-reverse}.home-page-features-wrapper .feature .feature-half{margin:1rem;padding:.25rem;width:40%}.home-page-features-wrapper .feature .feature-half.assets{align-items:center;background:var(--feature-color);border-radius:8px;display:flex;height:fit-content;justify-content:center;margin:.5rem}.home-page-features-wrapper .feature .feature-half.assets img.demo,.home-page-features-wrapper .feature .feature-half.assets span.not-demo{background:var(--kinda-transparent);border-radius:8px;box-shadow:var(--feature-img-shadow);height:100%;max-height:36rem;min-height:12rem;object-fit:contain;width:100%}.home-page-features-wrapper .feature .feature-half.assets span.not-demo{align-items:center;display:flex;justify-content:center}.home-page-features-wrapper .feature .feature-title{align-items:flex-end;color:var(--feature-color);display:flex;margin:1rem 0}.home-page-features-wrapper .feature .feature-title svg{margin-right:.5rem;width:2rem}.home-page-features-wrapper .feature div.read-the-docs a.button-link-wrapper{margin:0 1rem 1rem}.home-page-features-wrapper .feature div.read-the-docs small{margin:0 .5rem;opacity:.75}.authorsSection_t3c7{background:var(--hero-background);padding:3rem 1rem}.inner_rVjr{margin:.5rem auto;width:80%}.heading_j3J0{color:var(--text-color);font-size:2rem}.enjoyingWrap_FXPn{display:flex;gap:2rem;margin-bottom:2rem}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0{background:var(--background);border-radius:8px;padding:.5rem 1rem;width:50%}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0 p{margin:0 0 .5rem}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0 .authorInnerWrap_y4gi{align-items:center;display:flex;gap:.75rem}.subHeading_D3Yw{color:var(--text-color);font-size:1.2rem;margin:.25rem 0}.builtBy_GoGF,.name_mmMI{color:var(--mid-grey);font-size:.9rem}.grid_qTnk{display:grid;gap:.75rem;grid-template-columns:repeat(auto-fill,minmax(120px,1fr));margin:0 auto 2rem;max-width:1800px;padding:0}.tile_CRAN{align-items:center;background:var(--background);border-radius:8px;display:flex;flex-direction:column;gap:.4rem;padding:.6rem;position:relative;transition:transform .2s,box-shadow .2s}.ctaSection_oQB2,.statItem_at6_{align-items:center;display:flex}.tile_CRAN:hover{transform:translateY(-2px)}.avatar__HA9{border-radius:50%;height:48px;width:48px}.name_mmMI{max-width:120px;overflow:hidden}.star_JoTu{position:absolute;right:.5rem;top:.5rem}.fallback_O_bA{margin:0 auto;max-width:100%}.fallback_O_bA img{height:auto;max-width:100%}.ctaSection_oQB2{flex-direction:row;gap:.5rem;justify-content:center;margin-top:.25rem}.builtBy_GoGF{margin-bottom:1rem}.statsSection_Rc3S{background:var(--hero-background);padding:.25rem 1rem 3rem}.inner_YtnN{display:flex;flex-wrap:wrap;gap:1.5rem;justify-content:space-around;margin:0 auto;max-width:1000px;width:80%}.statItem_at6_{background:var(--background);border-radius:8px;flex:1 1 180px;flex-direction:column;max-width:240px;padding:1.5rem 2rem;text-align:center}.icon_vhSc{height:3rem;margin-bottom:.25rem;width:3rem}.count_Ad5j,.icon_vhSc{color:var(--stat-color)}.count_Ad5j{font-size:2.8rem;font-weight:400;line-height:1.1}.label_wElN{color:var(--mid-grey);font-size:1rem;margin-top:.25rem}.buttonGroup_M5ko button,.codeBlockContainer_Ckt0{background:var(--prism-background-color);color:var(--prism-color)}iframe#dashy-survey{border:none;height:92vh;width:100%}.codeBlockContainer_Ckt0{border-radius:var(--ifm-code-border-radius);box-shadow:var(--ifm-global-shadow-lw);margin-bottom:var(--ifm-leading)}.codeBlock_bY9V{--ifm-pre-background:var(--prism-background-color);margin:0;padding:0}.codeBlockLines_e6Vv{float:left;font:inherit;min-width:100%;padding:var(--ifm-pre-padding)}.codeBlockLinesWithNumbering_o6Pm{display:table;padding:var(--ifm-pre-padding) 0}:where(:root){--docusaurus-highlighted-code-line-bg:#484d5b}:where([data-theme=dark]){--docusaurus-highlighted-code-line-bg:#646464}.theme-code-block-highlighted-line{background-color:var(--docusaurus-highlighted-code-line-bg);display:block;margin:0 calc(var(--ifm-pre-padding)*-1);padding:0 var(--ifm-pre-padding)}.codeLine_lJS_{counter-increment:line-count;display:table-row}.codeLineNumber_Tfdd{background:var(--ifm-pre-background);display:table-cell;left:0;overflow-wrap:normal;padding:0 var(--ifm-pre-padding);position:sticky;text-align:right;width:1%}.codeLineNumber_Tfdd:before{content:counter(line-count);opacity:.4}.codeLineContent_feaV{padding-right:var(--ifm-pre-padding)}.theme-code-block:hover .copyButtonCopied_Vdqa{opacity:1!important}.copyButtonIcons_IEyt{height:1.125rem;position:relative;width:1.125rem}.copyButtonIcon_TrPX,.copyButtonSuccessIcon_cVMy{fill:currentColor;height:inherit;left:0;opacity:inherit;position:absolute;top:0;transition:all var(--ifm-transition-fast) ease;width:inherit}.copyButtonSuccessIcon_cVMy{color:#00d600;left:50%;opacity:0;top:50%;transform:translate(-50%,-50%) scale(.33)}.copyButtonCopied_Vdqa .copyButtonIcon_TrPX{opacity:0;transform:scale(.33)}.copyButtonCopied_Vdqa .copyButtonSuccessIcon_cVMy{opacity:1;transform:translate(-50%,-50%) scale(1);transition-delay:75ms}.wordWrapButtonIcon_b1P5{height:1.2rem;width:1.2rem}.buttonGroup_M5ko{column-gap:.2rem;display:flex;position:absolute;right:calc(var(--ifm-pre-padding)/2);top:calc(var(--ifm-pre-padding)/2)}.buttonGroup_M5ko button{align-items:center;border:1px solid var(--ifm-color-emphasis-300);border-radius:var(--ifm-global-radius);display:flex;line-height:0;opacity:0;padding:.4rem;transition:opacity var(--ifm-transition-fast) ease-in-out}.buttonGroup_M5ko button:focus-visible,.buttonGroup_M5ko button:hover{opacity:1!important}.theme-code-block:hover .buttonGroup_M5ko button{opacity:.4}.codeBlockContent_QJqH{border-radius:inherit;direction:ltr;position:relative}.codeBlockTitle_OeMC{border-bottom:1px solid var(--ifm-color-emphasis-300);border-top-left-radius:inherit;border-top-right-radius:inherit;font-size:var(--ifm-code-font-size);font-weight:500;padding:.75rem var(--ifm-pre-padding)}.codeBlockTitle_OeMC+.codeBlockContent_QJqH .codeBlock_a8dz{border-top-left-radius:0;border-top-right-radius:0}.details_lb9f{--docusaurus-details-summary-arrow-size:0.38rem;--docusaurus-details-transition:transform 200ms ease;--docusaurus-details-decoration-color:grey}.details_lb9f>summary{cursor:pointer;padding-left:1rem;position:relative}.details_lb9f>summary::-webkit-details-marker{display:none}.details_lb9f>summary:before{border-color:#0000 #0000 #0000 var(--docusaurus-details-decoration-color);border-style:solid;border-width:var(--docusaurus-details-summary-arrow-size);content:"";left:0;position:absolute;top:.45rem;transform:rotate(0);transform-origin:calc(var(--docusaurus-details-summary-arrow-size)/2) 50%;transition:var(--docusaurus-details-transition)}.collapsibleContent_i85q{border-top:1px solid var(--docusaurus-details-decoration-color);margin-top:1rem;padding-top:1rem}.details_b_Ee{--docusaurus-details-decoration-color:var(--ifm-alert-border-color);--docusaurus-details-transition:transform var(--ifm-transition-fast) ease;border:1px solid var(--ifm-alert-border-color);margin:0 0 var(--ifm-spacing-vertical)}:not(.containsTaskList_mC6p>li)>.containsTaskList_mC6p{padding-left:0}.img_ev3q{height:auto}.admonition_xJq3{margin-bottom:1em}.admonitionHeading_Gvgb{font:var(--ifm-heading-font-weight) var(--ifm-h5-font-size)/var(--ifm-heading-line-height) var(--ifm-heading-font-family)}.admonitionHeading_Gvgb:not(:last-child){margin-bottom:.3rem}.admonitionHeading_Gvgb code{text-transform:none}.admonitionIcon_Rf37{display:inline-block;margin-right:.4em;vertical-align:middle}.admonitionIcon_Rf37 svg{display:inline-block;fill:var(--ifm-alert-foreground-color);height:1.6em;width:1.6em}.tableOfContents_bqdL{max-height:calc(100vh - var(--ifm-navbar-height) - 2rem);overflow-y:auto;position:sticky;top:calc(var(--ifm-navbar-height) + 1rem)}.breadcrumbHomeIcon_YNFT{height:1.1rem;position:relative;top:1px;vertical-align:top;width:1.1rem}.breadcrumbsContainer_Z_bl{--ifm-breadcrumb-size-multiplier:0.8;margin-bottom:.8rem}@media (min-width:997px){.collapseSidebarButton_PEFL,.expandButton_TmdG{background-color:var(--docusaurus-collapse-button-bg)}:root{--docusaurus-announcement-bar-height:30px}.announcementBarClose_gvF7,.announcementBarPlaceholder_vyr4{flex-basis:50px}.collapseSidebarButton_PEFL{border:1px solid var(--ifm-toc-border-color);border-radius:0;bottom:0;display:block!important;height:40px;position:sticky}.collapseSidebarButtonIcon_kv0_{margin-top:4px;transform:rotate(180deg)}.expandButtonIcon_i1dp,[dir=rtl] .collapseSidebarButtonIcon_kv0_{transform:rotate(0)}.collapseSidebarButton_PEFL:focus,.collapseSidebarButton_PEFL:hover,.expandButton_TmdG:focus,.expandButton_TmdG:hover{background-color:var(--docusaurus-collapse-button-bg-hover)}.menuHtmlItem_M9Kj{padding:var(--ifm-menu-link-padding-vertical) var(--ifm-menu-link-padding-horizontal)}.menu_SIkG{flex-grow:1;padding:.5rem}@supports (scrollbar-gutter:stable){.menu_SIkG{padding:.5rem 0 .5rem .5rem;scrollbar-gutter:stable}}.menuWithAnnouncementBar_GW3s{margin-bottom:var(--docusaurus-announcement-bar-height)}.sidebar_njMd{display:flex;flex-direction:column;height:100%;padding-top:var(--ifm-navbar-height);width:var(--doc-sidebar-width)}.sidebarWithHideableNavbar_wUlq{padding-top:0}.sidebarHidden_VK0M{opacity:0;visibility:hidden}.sidebarLogo_isFc{align-items:center;color:inherit!important;display:flex!important;margin:0 var(--ifm-navbar-padding-horizontal);max-height:var(--ifm-navbar-height);min-height:var(--ifm-navbar-height);-webkit-text-decoration:none!important;text-decoration:none!important}.sidebarLogo_isFc img{height:2rem;margin-right:.5rem}.expandButton_TmdG{align-items:center;display:flex;height:100%;justify-content:center;position:absolute;right:0;top:0;transition:background-color var(--ifm-transition-fast) ease;width:100%}[dir=rtl] .expandButtonIcon_i1dp{transform:rotate(180deg)}.docSidebarContainer_YfHR{border-right:1px solid var(--ifm-toc-border-color);clip-path:inset(0);display:block;margin-top:calc(var(--ifm-navbar-height)*-1);transition:width var(--ifm-transition-fast) ease;width:var(--doc-sidebar-width);will-change:width}.docSidebarContainerHidden_DPk8{cursor:pointer;width:var(--doc-sidebar-hidden-width)}.sidebarViewport_aRkj{height:100%;max-height:100vh;position:sticky;top:0}.docMainContainer_TBSr{flex-grow:1;max-width:calc(100% - var(--doc-sidebar-width))}.docMainContainerEnhanced_lQrH{max-width:calc(100% - var(--doc-sidebar-hidden-width))}.docItemWrapperEnhanced_JWYK{max-width:calc(var(--ifm-container-width) + var(--doc-sidebar-width))!important}.navbarSearchContainer_Bca1{padding:0 var(--ifm-navbar-item-padding-horizontal)}.lastUpdated_JAkA{text-align:right}.tocMobile_ITEo{display:none}.docItemCol_VOVn{max-width:75%!important}}@media (min-width:997px)and (max-width:1200px){nav.navbar .navbar__item.navbar__link{font-size:1rem}nav.navbar .navbar__item.navbar__link svg{display:none}}@media screen and (min-width:1200px){.sponsorContainer_zSLg{flex-direction:row;justify-content:center;max-width:1000px}.sponsor_JDQ1{flex:1 1 0;min-height:100%}}@media (min-width:1440px){.container{max-width:var(--ifm-container-width-xl)}}@media (max-width:996px){.col{--ifm-col-width:100%;flex-basis:var(--ifm-col-width);margin-left:0}.footer{--ifm-footer-padding-horizontal:0}.colorModeToggle_DEke,.footer__link-separator,.navbar__item,.tableOfContents_bqdL{display:none}.footer__col{margin-bottom:calc(var(--ifm-spacing-vertical)*3)}.footer__link-item{display:block;width:max-content}.hero{padding-left:0;padding-right:0}.navbar>.container,.navbar>.container-fluid{padding:0}.navbar__toggle{display:inherit}.navbar__search-input{width:9rem}.pills--block,.tabs--block{flex-direction:column}.navbarSearchContainer_Bca1{position:absolute;right:var(--ifm-navbar-padding-horizontal)}.docItemContainer_F8PC{padding:0 .3rem}}@media not (max-width:996px){.searchBar_RVTs.searchBarLeft_MXDe .dropdownMenu_qbY6{left:0!important;right:auto!important}}@media only screen and (max-width:996px){.searchQueryColumn_q7nx{max-width:60%!important}.searchContextColumn_oWAF{max-width:40%!important}}@media (max-width:966px){.timeline_WTQQ{padding-left:2rem;width:100%}.commitEntry_JtMn:before,.releaseEntry_ZG2V:before,.tagEntry_ORGt:before{left:-2rem}.commitRow_RIZe{flex-wrap:wrap}.commitMsg_XeKn{max-width:100%}.commitMeta_teNK{margin-left:0}.bodyPreview_iKaD{-webkit-line-clamp:2}header.heroBanner_qw_p h1.heroTitle_CteR{font-size:6rem}header.heroBanner_qw_p h3.heroSubTitle_p8ER{font-size:1.5rem}.scrollDown_cjGa{display:none}.button{width:80%}.ctaSection_oQB2,.enjoyingWrap_FXPn,.home-page-features-wrapper .feature.align-left,.home-page-features-wrapper .feature.align-right{flex-direction:column}.home-page-features-wrapper .feature.align-left .feature-half,.home-page-features-wrapper .feature.align-right .feature-half{width:95%}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0,.inner_rVjr{width:100%}.grid_qTnk{grid-template-columns:repeat(auto-fill,minmax(100px,1fr))}.inner_YtnN{display:grid;gap:1rem;grid-template-columns:1fr 1fr;width:100%}.statItem_at6_{max-width:none;padding:1.25rem 1rem}.count_Ad5j{font-size:2.2rem}}@media screen and (max-width:966px){header.heroBanner_qw_p{padding:2rem}.buttons_LJn2{flex-direction:column}}@media (max-width:768px){.glitchCode_Pt6F{font-size:6rem}}@media (max-width:600px){.banner_woPo{font-size:.85rem;padding:.4rem .75rem}.closeBtn_fC0A{font-size:1.1rem;right:.5rem}}@media (max-width:576px){.markdown h1:first-child{--ifm-h1-font-size:2rem}.markdown>h2{--ifm-h2-font-size:1.5rem}.markdown>h3{--ifm-h3-font-size:1.25rem}.navbar__search-input:not(:focus){width:2rem}.searchBar_RVTs .dropdownMenu_qbY6{max-width:calc(100vw - var(--ifm-navbar-padding-horizontal)*2);width:var(--search-local-modal-width-sm,340px)}.searchBarContainer_NW3z:not(.focused_OWtg) .searchClearButton_qk4g,.searchHintContainer_Pkmr{display:none}}@media screen and (max-width:576px){.searchQueryColumn_q7nx{max-width:100%!important}.searchContextColumn_oWAF{max-width:100%!important;padding-left:var(--ifm-spacing-horizontal)!important}}@media (max-width:500px){main.docs div.docs-contents a{width:100%}}@media (hover:hover){.backToTopButton_sjWU:hover{background-color:var(--ifm-color-emphasis-300)}}@media (pointer:fine){.thin-scrollbar{scrollbar-width:thin}.thin-scrollbar::-webkit-scrollbar{height:var(--ifm-scrollbar-size);width:var(--ifm-scrollbar-size)}.thin-scrollbar::-webkit-scrollbar-track{background:var(--ifm-scrollbar-track-background-color);border-radius:10px}.thin-scrollbar::-webkit-scrollbar-thumb{background:var(--ifm-scrollbar-thumb-background-color);border-radius:10px}.thin-scrollbar::-webkit-scrollbar-thumb:hover{background:var(--ifm-scrollbar-thumb-hover-background-color)}}@media (prefers-color-scheme:dark){#carbonads{--carbon-bg-primary:#1f1f1f;--carbon-bg-secondary:#262626;--carbon-text-color:#e6e6e6}}@media (prefers-reduced-motion:reduce){:root{--ifm-transition-fast:0ms;--ifm-transition-slow:0ms}.skeleton_NVtD{animation:none;opacity:.6}}@media print{.announcementBar_mb4j,.footer,.menu,.navbar,.noPrint_WFHX,.pagination-nav,.table-of-contents,.tocMobile_ITEo{display:none}.tabs{page-break-inside:avoid}.codeBlockLines_e6Vv{white-space:pre-wrap}} \ No newline at end of file +@import url(https://fonts.googleapis.com/css2?family=Racing+Sans+One&family=Poppins:wght@300&family=Roboto+Mono:wght@500;300&display=swap);.col,.container{padding:0 var(--ifm-spacing-horizontal);width:100%}.markdown>h2,.markdown>h3,.markdown>h4,.markdown>h5,.markdown>h6{margin-bottom:calc(var(--ifm-heading-vertical-rhythm-bottom)*var(--ifm-leading))}.markdown li,body{word-wrap:break-word}body,ol ol,ol ul,ul ol,ul ul{margin:0}pre,table{overflow:auto}blockquote,pre{margin:0 0 var(--ifm-spacing-vertical)}.breadcrumbs__link,.button{transition-timing-function:var(--ifm-transition-timing-default)}.button,code{vertical-align:middle}.button--outline.button--active,.button--outline:active,.button--outline:hover,:root{--ifm-button-color:var(--ifm-font-color-base-inverse)}.menu__link:hover,a{transition:color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.navbar--dark,:root{--ifm-navbar-link-hover-color:var(--ifm-color-primary)}.menu,.navbar-sidebar{overflow-x:hidden}:root,html[data-theme=dark]{--ifm-color-emphasis-500:var(--ifm-color-gray-500)}#carbonads .carbon-poweredby,#carbonads a,#carbonads a:hover{color:var(--carbon-text-color)}.toggleButton_gllP,html{-webkit-tap-highlight-color:transparent}*,.loadingRing_RJI3 div{box-sizing:border-box}.close,.home-page-features-wrapper .feature div.read-the-docs{float:right}.commitMsg_XeKn,.name_mmMI,.text--truncate{text-overflow:ellipsis}.clean-list,.containsTaskList_mC6p,.details_lb9f>summary,.dropdown__menu,.grid_qTnk,.menu__list{list-style:none}:root{--ifm-color-scheme:light;--ifm-dark-value:10%;--ifm-darker-value:15%;--ifm-darkest-value:30%;--ifm-light-value:15%;--ifm-lighter-value:30%;--ifm-lightest-value:50%;--ifm-contrast-background-value:90%;--ifm-contrast-foreground-value:70%;--ifm-contrast-background-dark-value:70%;--ifm-contrast-foreground-dark-value:90%;--ifm-color-primary:#3578e5;--ifm-color-secondary:#ebedf0;--ifm-color-success:#00a400;--ifm-color-info:#54c7ec;--ifm-color-warning:#ffba00;--ifm-color-danger:#fa383e;--ifm-color-primary-dark:#306cce;--ifm-color-primary-darker:#2d66c3;--ifm-color-primary-darkest:#2554a0;--ifm-color-primary-light:#538ce9;--ifm-color-primary-lighter:#72a1ed;--ifm-color-primary-lightest:#9abcf2;--ifm-color-primary-contrast-background:#ebf2fc;--ifm-color-primary-contrast-foreground:#102445;--ifm-color-secondary-dark:#d4d5d8;--ifm-color-secondary-darker:#c8c9cc;--ifm-color-secondary-darkest:#a4a6a8;--ifm-color-secondary-light:#eef0f2;--ifm-color-secondary-lighter:#f1f2f5;--ifm-color-secondary-lightest:#f5f6f8;--ifm-color-secondary-contrast-background:#fdfdfe;--ifm-color-secondary-contrast-foreground:#474748;--ifm-color-success-dark:#009400;--ifm-color-success-darker:#008b00;--ifm-color-success-darkest:#007300;--ifm-color-success-light:#26b226;--ifm-color-success-lighter:#4dbf4d;--ifm-color-success-lightest:#80d280;--ifm-color-success-contrast-background:#e6f6e6;--ifm-color-success-contrast-foreground:#003100;--ifm-color-info-dark:#4cb3d4;--ifm-color-info-darker:#47a9c9;--ifm-color-info-darkest:#3b8ba5;--ifm-color-info-light:#6ecfef;--ifm-color-info-lighter:#87d8f2;--ifm-color-info-lightest:#aae3f6;--ifm-color-info-contrast-background:#eef9fd;--ifm-color-info-contrast-foreground:#193c47;--ifm-color-warning-dark:#e6a700;--ifm-color-warning-darker:#d99e00;--ifm-color-warning-darkest:#b38200;--ifm-color-warning-light:#ffc426;--ifm-color-warning-lighter:#ffcf4d;--ifm-color-warning-lightest:#ffdd80;--ifm-color-warning-contrast-background:#fff8e6;--ifm-color-warning-contrast-foreground:#4d3800;--ifm-color-danger-dark:#e13238;--ifm-color-danger-darker:#d53035;--ifm-color-danger-darkest:#af272b;--ifm-color-danger-light:#fb565b;--ifm-color-danger-lighter:#fb7478;--ifm-color-danger-lightest:#fd9c9f;--ifm-color-danger-contrast-background:#ffebec;--ifm-color-danger-contrast-foreground:#4b1113;--ifm-color-white:#fff;--ifm-color-black:#000;--ifm-color-gray-0:var(--ifm-color-white);--ifm-color-gray-100:#f5f6f7;--ifm-color-gray-200:#ebedf0;--ifm-color-gray-300:#dadde1;--ifm-color-gray-400:#ccd0d5;--ifm-color-gray-500:#bec3c9;--ifm-color-gray-600:#8d949e;--ifm-color-gray-700:#606770;--ifm-color-gray-800:#444950;--ifm-color-gray-900:#1c1e21;--ifm-color-gray-1000:var(--ifm-color-black);--ifm-color-emphasis-0:var(--ifm-color-gray-0);--ifm-color-emphasis-100:var(--ifm-color-gray-100);--ifm-color-emphasis-200:var(--ifm-color-gray-200);--ifm-color-emphasis-300:var(--ifm-color-gray-300);--ifm-color-emphasis-400:var(--ifm-color-gray-400);--ifm-color-emphasis-600:var(--ifm-color-gray-600);--ifm-color-emphasis-700:var(--ifm-color-gray-700);--ifm-color-emphasis-800:var(--ifm-color-gray-800);--ifm-color-emphasis-900:var(--ifm-color-gray-900);--ifm-color-emphasis-1000:var(--ifm-color-gray-1000);--ifm-color-content:var(--ifm-color-emphasis-900);--ifm-color-content-inverse:var(--ifm-color-emphasis-0);--ifm-color-content-secondary:#525860;--ifm-background-color:#0000;--ifm-background-surface-color:var(--ifm-color-content-inverse);--ifm-global-border-width:1px;--ifm-global-radius:0.4rem;--ifm-hover-overlay:#0000000d;--ifm-font-color-base:var(--ifm-color-content);--ifm-font-color-base-inverse:var(--ifm-color-content-inverse);--ifm-font-color-secondary:var(--ifm-color-content-secondary);--ifm-font-family-base:system-ui,-apple-system,Segoe UI,Roboto,Ubuntu,Cantarell,Noto Sans,sans-serif,BlinkMacSystemFont,"Segoe UI",Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";--ifm-font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--ifm-font-size-base:100%;--ifm-font-weight-light:300;--ifm-font-weight-normal:400;--ifm-font-weight-semibold:500;--ifm-font-weight-bold:700;--ifm-font-weight-base:var(--ifm-font-weight-normal);--ifm-line-height-base:1.65;--ifm-global-spacing:1rem;--ifm-spacing-vertical:var(--ifm-global-spacing);--ifm-spacing-horizontal:var(--ifm-global-spacing);--ifm-transition-fast:200ms;--ifm-transition-slow:400ms;--ifm-transition-timing-default:cubic-bezier(0.08,0.52,0.52,1);--ifm-global-shadow-lw:0 1px 2px 0 #0000001a;--ifm-global-shadow-md:0 5px 40px #0003;--ifm-global-shadow-tl:0 12px 28px 0 #0003,0 2px 4px 0 #0000001a;--ifm-z-index-dropdown:100;--ifm-z-index-fixed:200;--ifm-z-index-overlay:400;--ifm-container-width:1140px;--ifm-container-width-xl:1320px;--ifm-code-background:#f6f7f8;--ifm-code-border-radius:var(--ifm-global-radius);--ifm-code-font-size:90%;--ifm-code-padding-horizontal:0.1rem;--ifm-code-padding-vertical:0.1rem;--ifm-pre-background:var(--ifm-code-background);--ifm-pre-border-radius:var(--ifm-code-border-radius);--ifm-pre-color:inherit;--ifm-pre-line-height:1.45;--ifm-pre-padding:1rem;--ifm-heading-color:inherit;--ifm-heading-margin-top:0;--ifm-heading-margin-bottom:var(--ifm-spacing-vertical);--ifm-heading-font-family:var(--ifm-font-family-base);--ifm-heading-font-weight:var(--ifm-font-weight-bold);--ifm-heading-line-height:1.25;--ifm-h1-font-size:2rem;--ifm-h2-font-size:1.5rem;--ifm-h3-font-size:1.25rem;--ifm-h4-font-size:1rem;--ifm-h5-font-size:0.875rem;--ifm-h6-font-size:0.85rem;--ifm-image-alignment-padding:1.25rem;--ifm-leading-desktop:1.25;--ifm-leading:calc(var(--ifm-leading-desktop)*1rem);--ifm-list-left-padding:2rem;--ifm-list-margin:1rem;--ifm-list-item-margin:0.25rem;--ifm-list-paragraph-margin:1rem;--ifm-table-cell-padding:0.75rem;--ifm-table-background:#0000;--ifm-table-stripe-background:#00000008;--ifm-table-border-width:1px;--ifm-table-border-color:var(--ifm-color-emphasis-300);--ifm-table-head-background:inherit;--ifm-table-head-color:inherit;--ifm-table-head-font-weight:var(--ifm-font-weight-bold);--ifm-table-cell-color:inherit;--ifm-link-color:var(--ifm-color-primary);--ifm-link-decoration:none;--ifm-link-hover-color:var(--ifm-link-color);--ifm-link-hover-decoration:underline;--ifm-paragraph-margin-bottom:var(--ifm-leading);--ifm-blockquote-font-size:var(--ifm-font-size-base);--ifm-blockquote-border-left-width:2px;--ifm-blockquote-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-blockquote-padding-vertical:0;--ifm-blockquote-shadow:none;--ifm-blockquote-color:var(--ifm-color-emphasis-800);--ifm-blockquote-border-color:var(--ifm-color-emphasis-300);--ifm-hr-background-color:var(--ifm-color-emphasis-500);--ifm-hr-height:1px;--ifm-hr-margin-vertical:1.5rem;--ifm-scrollbar-size:7px;--ifm-scrollbar-track-background-color:#f1f1f1;--ifm-scrollbar-thumb-background-color:silver;--ifm-scrollbar-thumb-hover-background-color:#a7a7a7;--ifm-alert-background-color:inherit;--ifm-alert-border-color:inherit;--ifm-alert-border-radius:var(--ifm-global-radius);--ifm-alert-border-width:0px;--ifm-alert-border-left-width:5px;--ifm-alert-color:var(--ifm-font-color-base);--ifm-alert-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-alert-padding-vertical:var(--ifm-spacing-vertical);--ifm-alert-shadow:var(--ifm-global-shadow-lw);--ifm-avatar-intro-margin:1rem;--ifm-avatar-intro-alignment:inherit;--ifm-avatar-photo-size:3rem;--ifm-badge-background-color:inherit;--ifm-badge-border-color:inherit;--ifm-badge-border-radius:var(--ifm-global-radius);--ifm-badge-border-width:var(--ifm-global-border-width);--ifm-badge-color:var(--ifm-color-white);--ifm-badge-padding-horizontal:calc(var(--ifm-spacing-horizontal)*0.5);--ifm-badge-padding-vertical:calc(var(--ifm-spacing-vertical)*0.25);--ifm-breadcrumb-border-radius:1.5rem;--ifm-breadcrumb-spacing:0.5rem;--ifm-breadcrumb-color-active:var(--ifm-color-primary);--ifm-breadcrumb-item-background-active:var(--ifm-hover-overlay);--ifm-breadcrumb-padding-horizontal:0.8rem;--ifm-breadcrumb-padding-vertical:0.4rem;--ifm-breadcrumb-size-multiplier:1;--ifm-breadcrumb-separator:url('data:image/svg+xml;utf8,');--ifm-breadcrumb-separator-filter:none;--ifm-breadcrumb-separator-size:0.5rem;--ifm-breadcrumb-separator-size-multiplier:1.25;--ifm-button-background-color:inherit;--ifm-button-border-color:var(--ifm-button-background-color);--ifm-button-border-width:var(--ifm-global-border-width);--ifm-button-font-weight:var(--ifm-font-weight-bold);--ifm-button-padding-horizontal:1.5rem;--ifm-button-padding-vertical:0.375rem;--ifm-button-size-multiplier:1;--ifm-button-transition-duration:var(--ifm-transition-fast);--ifm-button-border-radius:calc(var(--ifm-global-radius)*var(--ifm-button-size-multiplier));--ifm-button-group-spacing:2px;--ifm-card-background-color:var(--ifm-background-surface-color);--ifm-card-border-radius:calc(var(--ifm-global-radius)*2);--ifm-card-horizontal-spacing:var(--ifm-global-spacing);--ifm-card-vertical-spacing:var(--ifm-global-spacing);--ifm-toc-border-color:var(--ifm-color-emphasis-300);--ifm-toc-link-color:var(--ifm-color-content-secondary);--ifm-toc-padding-vertical:0.5rem;--ifm-toc-padding-horizontal:0.5rem;--ifm-dropdown-background-color:var(--ifm-background-surface-color);--ifm-dropdown-font-weight:var(--ifm-font-weight-semibold);--ifm-dropdown-link-color:var(--ifm-font-color-base);--ifm-dropdown-hover-background-color:var(--ifm-hover-overlay);--ifm-footer-background-color:var(--ifm-color-emphasis-100);--ifm-footer-color:inherit;--ifm-footer-link-color:var(--ifm-color-emphasis-700);--ifm-footer-link-hover-color:var(--ifm-color-primary);--ifm-footer-link-horizontal-spacing:0.5rem;--ifm-footer-padding-horizontal:calc(var(--ifm-spacing-horizontal)*2);--ifm-footer-padding-vertical:calc(var(--ifm-spacing-vertical)*2);--ifm-footer-title-color:inherit;--ifm-footer-logo-max-width:min(30rem,90vw);--ifm-hero-background-color:var(--ifm-background-surface-color);--ifm-hero-text-color:var(--ifm-color-emphasis-800);--ifm-menu-color:var(--ifm-color-emphasis-700);--ifm-menu-color-active:var(--ifm-color-primary);--ifm-menu-color-background-active:var(--ifm-hover-overlay);--ifm-menu-color-background-hover:var(--ifm-hover-overlay);--ifm-menu-link-padding-horizontal:0.75rem;--ifm-menu-link-padding-vertical:0.375rem;--ifm-menu-link-sublist-icon:url('data:image/svg+xml;utf8,');--ifm-menu-link-sublist-icon-filter:none;--ifm-navbar-background-color:var(--ifm-background-surface-color);--ifm-navbar-height:3.75rem;--ifm-navbar-item-padding-horizontal:0.75rem;--ifm-navbar-item-padding-vertical:0.25rem;--ifm-navbar-link-color:var(--ifm-font-color-base);--ifm-navbar-link-active-color:var(--ifm-link-color);--ifm-navbar-padding-horizontal:var(--ifm-spacing-horizontal);--ifm-navbar-padding-vertical:calc(var(--ifm-spacing-vertical)*0.5);--ifm-navbar-shadow:var(--ifm-global-shadow-lw);--ifm-navbar-search-input-background-color:var(--ifm-color-emphasis-200);--ifm-navbar-search-input-color:var(--ifm-color-emphasis-800);--ifm-navbar-search-input-placeholder-color:var(--ifm-color-emphasis-500);--ifm-navbar-search-input-icon:url('data:image/svg+xml;utf8,');--ifm-navbar-sidebar-width:83vw;--ifm-pagination-border-radius:var(--ifm-global-radius);--ifm-pagination-color-active:var(--ifm-color-primary);--ifm-pagination-font-size:1rem;--ifm-pagination-item-active-background:var(--ifm-hover-overlay);--ifm-pagination-page-spacing:0.2em;--ifm-pagination-padding-horizontal:calc(var(--ifm-spacing-horizontal)*1);--ifm-pagination-padding-vertical:calc(var(--ifm-spacing-vertical)*0.25);--ifm-pagination-nav-border-radius:var(--ifm-global-radius);--ifm-pagination-nav-color-hover:var(--ifm-color-primary);--ifm-pills-color-active:var(--ifm-color-primary);--ifm-pills-color-background-active:var(--ifm-hover-overlay);--ifm-pills-spacing:0.125rem;--ifm-tabs-color:var(--ifm-font-color-secondary);--ifm-tabs-color-active:var(--ifm-color-primary);--ifm-tabs-color-active-border:var(--ifm-tabs-color-active);--ifm-tabs-padding-horizontal:1rem;--ifm-tabs-padding-vertical:1rem}.badge--danger,.badge--info,.badge--primary,.badge--secondary,.badge--success,.badge--warning{--ifm-badge-border-color:var(--ifm-badge-background-color)}.button--link,.button--outline{--ifm-button-background-color:#0000}html{background-color:var(--ifm-background-color);color:var(--ifm-font-color-base);color-scheme:var(--ifm-color-scheme);font:var(--ifm-font-size-base)/var(--ifm-line-height-base) var(--ifm-font-family-base);-webkit-font-smoothing:antialiased;text-rendering:optimizelegibility;-webkit-text-size-adjust:100%;text-size-adjust:100%;--primary:#54bff7;--background:var(--pale-grey);--text-color:var(--black);--hero-background:var(--bright-white);--kinda-transparent:#ffffffd9;--hyperlink:#ff62ce;--feature-img-shadow:4px 4px 6px #a9a9a980,-2px -2px 4px #0006;--footer-color:var(--white);--footer-text-color:var(--black);scroll-behavior:smooth}body{background:var(--background)}iframe{border:0;color-scheme:auto}.container{margin:0 auto;max-width:var(--ifm-container-width)}.container--fluid{max-width:inherit}.row{display:flex;flex-wrap:wrap;margin:0 calc(var(--ifm-spacing-horizontal)*-1)}.margin-bottom--none,.margin-vert--none,.markdown>:last-child{margin-bottom:0!important}.margin-top--none,.margin-vert--none{margin-top:0!important}.row--no-gutters{margin-left:0;margin-right:0}.margin-horiz--none,.margin-right--none{margin-right:0!important}.row--no-gutters>.col{padding-left:0;padding-right:0}.row--align-top{align-items:flex-start}.row--align-bottom{align-items:flex-end}.menuExternalLink_NmtK,.row--align-center{align-items:center}.row--align-stretch{align-items:stretch}.row--align-baseline{align-items:baseline}.col{--ifm-col-width:100%;flex:1 0;margin-left:0;max-width:var(--ifm-col-width)}.padding-bottom--none,.padding-vert--none{padding-bottom:0!important}.padding-top--none,.padding-vert--none{padding-top:0!important}.padding-horiz--none,.padding-left--none{padding-left:0!important}.padding-horiz--none,.padding-right--none{padding-right:0!important}.col[class*=col--]{flex:0 0 var(--ifm-col-width)}.col--1{--ifm-col-width:8.33333%}.col--offset-1{margin-left:8.33333%}.col--2{--ifm-col-width:16.66667%}.col--offset-2{margin-left:16.66667%}.col--3{--ifm-col-width:25%}.col--offset-3{margin-left:25%}.col--4{--ifm-col-width:33.33333%}.col--offset-4{margin-left:33.33333%}.col--5{--ifm-col-width:41.66667%}.col--offset-5{margin-left:41.66667%}.col--6{--ifm-col-width:50%}.col--offset-6{margin-left:50%}.col--7{--ifm-col-width:58.33333%}.col--offset-7{margin-left:58.33333%}.col--8{--ifm-col-width:66.66667%}.col--offset-8{margin-left:66.66667%}.col--9{--ifm-col-width:75%}.col--offset-9{margin-left:75%}.col--10{--ifm-col-width:83.33333%}.col--offset-10{margin-left:83.33333%}.col--11{--ifm-col-width:91.66667%}.col--offset-11{margin-left:91.66667%}.col--12{--ifm-col-width:100%}.col--offset-12{margin-left:100%}.margin-horiz--none,.margin-left--none{margin-left:0!important}.margin--none{margin:0!important}.margin-bottom--xs,.margin-vert--xs{margin-bottom:.25rem!important}.margin-top--xs,.margin-vert--xs{margin-top:.25rem!important}.margin-horiz--xs,.margin-left--xs{margin-left:.25rem!important}.margin-horiz--xs,.margin-right--xs{margin-right:.25rem!important}.margin--xs{margin:.25rem!important}.margin-bottom--sm,.margin-vert--sm{margin-bottom:.5rem!important}.margin-top--sm,.margin-vert--sm{margin-top:.5rem!important}.margin-horiz--sm,.margin-left--sm{margin-left:.5rem!important}.margin-horiz--sm,.margin-right--sm{margin-right:.5rem!important}.margin--sm{margin:.5rem!important}.margin-bottom--md,.margin-vert--md{margin-bottom:1rem!important}.margin-top--md,.margin-vert--md{margin-top:1rem!important}.margin-horiz--md,.margin-left--md{margin-left:1rem!important}.margin-horiz--md,.margin-right--md{margin-right:1rem!important}.margin--md{margin:1rem!important}.margin-bottom--lg,.margin-vert--lg{margin-bottom:2rem!important}.margin-top--lg,.margin-vert--lg{margin-top:2rem!important}.margin-horiz--lg,.margin-left--lg{margin-left:2rem!important}.margin-horiz--lg,.margin-right--lg{margin-right:2rem!important}.margin--lg{margin:2rem!important}.margin-bottom--xl,.margin-vert--xl{margin-bottom:5rem!important}.margin-top--xl,.margin-vert--xl{margin-top:5rem!important}.margin-horiz--xl,.margin-left--xl{margin-left:5rem!important}.margin-horiz--xl,.margin-right--xl{margin-right:5rem!important}.margin--xl{margin:5rem!important}.padding--none{padding:0!important}.padding-bottom--xs,.padding-vert--xs{padding-bottom:.25rem!important}.padding-top--xs,.padding-vert--xs{padding-top:.25rem!important}.padding-horiz--xs,.padding-left--xs{padding-left:.25rem!important}.padding-horiz--xs,.padding-right--xs{padding-right:.25rem!important}.padding--xs{padding:.25rem!important}.padding-bottom--sm,.padding-vert--sm{padding-bottom:.5rem!important}.padding-top--sm,.padding-vert--sm{padding-top:.5rem!important}.padding-horiz--sm,.padding-left--sm{padding-left:.5rem!important}.padding-horiz--sm,.padding-right--sm{padding-right:.5rem!important}.padding--sm{padding:.5rem!important}.padding-bottom--md,.padding-vert--md{padding-bottom:1rem!important}.padding-top--md,.padding-vert--md{padding-top:1rem!important}.padding-horiz--md,.padding-left--md{padding-left:1rem!important}.padding-horiz--md,.padding-right--md{padding-right:1rem!important}.padding--md{padding:1rem!important}.padding-bottom--lg,.padding-vert--lg{padding-bottom:2rem!important}.padding-top--lg,.padding-vert--lg{padding-top:2rem!important}.padding-horiz--lg,.padding-left--lg{padding-left:2rem!important}.padding-horiz--lg,.padding-right--lg{padding-right:2rem!important}.padding--lg{padding:2rem!important}.padding-bottom--xl,.padding-vert--xl{padding-bottom:5rem!important}.padding-top--xl,.padding-vert--xl{padding-top:5rem!important}.padding-horiz--xl,.padding-left--xl{padding-left:5rem!important}.padding-horiz--xl,.padding-right--xl{padding-right:5rem!important}.padding--xl{padding:5rem!important}code{background-color:var(--ifm-code-background);border:.1rem solid #0000001a;border-radius:var(--ifm-code-border-radius);font-family:var(--ifm-font-family-monospace);font-size:var(--ifm-code-font-size);padding:var(--ifm-code-padding-vertical) var(--ifm-code-padding-horizontal)}a code{color:inherit}pre{background-color:var(--ifm-pre-background);border-radius:var(--ifm-pre-border-radius);color:var(--ifm-pre-color);font:var(--ifm-code-font-size)/var(--ifm-pre-line-height) var(--ifm-font-family-monospace);padding:var(--ifm-pre-padding)}pre code{background-color:initial;border:none;font-size:100%;line-height:inherit;padding:0}kbd{background-color:var(--ifm-color-emphasis-0);border:1px solid var(--ifm-color-emphasis-400);border-radius:.2rem;box-shadow:inset 0 -1px 0 var(--ifm-color-emphasis-400);color:var(--ifm-color-emphasis-800);font:80% var(--ifm-font-family-monospace);padding:.15rem .3rem}h1,h2,h3,h4,h5,h6{color:var(--ifm-heading-color);font-family:var(--ifm-heading-font-family);font-weight:var(--ifm-heading-font-weight);line-height:var(--ifm-heading-line-height);margin:var(--ifm-heading-margin-top) 0 var(--ifm-heading-margin-bottom) 0}article div.markdown h1,h1{font-size:var(--ifm-h1-font-size)}article div.markdown h2,h2{font-size:var(--ifm-h2-font-size)}article div.markdown h3,h3{font-size:var(--ifm-h3-font-size)}h4{font-size:var(--ifm-h4-font-size)}h5{font-size:var(--ifm-h5-font-size)}h6{font-size:var(--ifm-h6-font-size)}.container_lyt7,.container_lyt7>svg,img{max-width:100%}img[align=right]{padding-left:var(--image-alignment-padding)}img[align=left]{padding-right:var(--image-alignment-padding)}.markdown{--ifm-h1-vertical-rhythm-top:3;--ifm-h2-vertical-rhythm-top:2;--ifm-h3-vertical-rhythm-top:1.5;--ifm-heading-vertical-rhythm-top:1.25;--ifm-h1-vertical-rhythm-bottom:1.25;--ifm-heading-vertical-rhythm-bottom:1}.markdown:after,.markdown:before{content:"";display:table}.markdown:after{clear:both}.markdown h1:first-child{--ifm-h1-font-size:3rem;margin-bottom:calc(var(--ifm-h1-vertical-rhythm-bottom)*var(--ifm-leading))}.markdown>h2{--ifm-h2-font-size:2rem;margin-top:calc(var(--ifm-h2-vertical-rhythm-top)*var(--ifm-leading))}.markdown>h3{--ifm-h3-font-size:1.5rem;margin-top:calc(var(--ifm-h3-vertical-rhythm-top)*var(--ifm-leading))}.markdown>h4,.markdown>h5,.markdown>h6{margin-top:calc(var(--ifm-heading-vertical-rhythm-top)*var(--ifm-leading))}.markdown>p,.markdown>pre,.markdown>ul{margin-bottom:var(--ifm-leading)}.markdown li>p{margin-top:var(--ifm-list-paragraph-margin)}.markdown li+li{margin-top:var(--ifm-list-item-margin)}ol,ul{margin:0 0 var(--ifm-list-margin);padding-left:var(--ifm-list-left-padding)}ol ol,ul ol{list-style-type:lower-roman}ol ol ol,ol ul ol,ul ol ol,ul ul ol{list-style-type:lower-alpha}table{border-collapse:collapse;display:block;margin-bottom:var(--ifm-spacing-vertical)}table thead tr{border-bottom:2px solid var(--ifm-table-border-color)}table thead,table tr:nth-child(2n){background-color:var(--ifm-table-stripe-background)}table tr{background-color:var(--ifm-table-background);border-top:var(--ifm-table-border-width) solid var(--ifm-table-border-color)}table td,table th{border:var(--ifm-table-border-width) solid var(--ifm-table-border-color);padding:var(--ifm-table-cell-padding)}table th{background-color:var(--ifm-table-head-background);color:var(--ifm-table-head-color);font-weight:var(--ifm-table-head-font-weight)}table td{color:var(--ifm-table-cell-color)}strong{font-weight:var(--ifm-font-weight-bold)}a{color:var(--ifm-link-color);text-decoration:var(--ifm-link-decoration)}a:hover{color:var(--ifm-link-hover-color);text-decoration:var(--ifm-link-hover-decoration)}.button:hover,.text--no-decoration,.text--no-decoration:hover,a:not([href]){-webkit-text-decoration:none;text-decoration:none}p{margin:0 0 var(--ifm-paragraph-margin-bottom)}blockquote{border-left:var(--ifm-blockquote-border-left-width) solid var(--ifm-blockquote-border-color);box-shadow:var(--ifm-blockquote-shadow);color:var(--ifm-blockquote-color);font-size:var(--ifm-blockquote-font-size);padding:var(--ifm-blockquote-padding-vertical) var(--ifm-blockquote-padding-horizontal)}blockquote>:first-child{margin-top:0}blockquote>:last-child{margin-bottom:0}hr{background-color:var(--ifm-hr-background-color);border:0;height:var(--ifm-hr-height);margin:var(--ifm-hr-margin-vertical) 0}.shadow--lw{box-shadow:var(--ifm-global-shadow-lw)!important}.shadow--md{box-shadow:var(--ifm-global-shadow-md)!important}.shadow--tl{box-shadow:var(--ifm-global-shadow-tl)!important}.text--primary,.wordWrapButtonEnabled_uzNF .wordWrapButtonIcon_b1P5{color:var(--ifm-color-primary)}.text--secondary{color:var(--ifm-color-secondary)}.text--success{color:var(--ifm-color-success)}.text--info{color:var(--ifm-color-info)}.text--warning{color:var(--ifm-color-warning)}.text--danger{color:var(--ifm-color-danger)}.text--center{text-align:center}.text--left{text-align:left}.text--justify{text-align:justify}.text--right{text-align:right}.text--capitalize{text-transform:capitalize}.text--lowercase{text-transform:lowercase}.admonitionHeading_Gvgb,.alert__heading,.text--uppercase{text-transform:uppercase}.text--light{font-weight:var(--ifm-font-weight-light)}.text--normal{font-weight:var(--ifm-font-weight-normal)}.text--semibold{font-weight:var(--ifm-font-weight-semibold)}.text--bold{font-weight:var(--ifm-font-weight-bold)}.text--italic{font-style:italic}.text--truncate{overflow:hidden;white-space:nowrap}.text--break{word-wrap:break-word!important;word-break:break-word!important}.clean-btn{background:none;border:none;color:inherit;cursor:pointer;font-family:inherit;padding:0}.alert,.alert .close{color:var(--ifm-alert-foreground-color)}.clean-list{padding-left:0}.alert--primary{--ifm-alert-background-color:var(--ifm-color-primary-contrast-background);--ifm-alert-background-color-highlight:#3578e526;--ifm-alert-foreground-color:var(--ifm-color-primary-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-primary-dark)}.alert--secondary{--ifm-alert-background-color:var(--ifm-color-secondary-contrast-background);--ifm-alert-background-color-highlight:#ebedf026;--ifm-alert-foreground-color:var(--ifm-color-secondary-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-secondary-dark)}.alert--success{--ifm-alert-background-color:var(--ifm-color-success-contrast-background);--ifm-alert-background-color-highlight:#00a40026;--ifm-alert-foreground-color:var(--ifm-color-success-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-success-dark)}.alert--info{--ifm-alert-background-color:var(--ifm-color-info-contrast-background);--ifm-alert-background-color-highlight:#54c7ec26;--ifm-alert-foreground-color:var(--ifm-color-info-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-info-dark)}.alert--warning{--ifm-alert-background-color:var(--ifm-color-warning-contrast-background);--ifm-alert-background-color-highlight:#ffba0026;--ifm-alert-foreground-color:var(--ifm-color-warning-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-warning-dark)}.alert--danger{--ifm-alert-background-color:var(--ifm-color-danger-contrast-background);--ifm-alert-background-color-highlight:#fa383e26;--ifm-alert-foreground-color:var(--ifm-color-danger-contrast-foreground);--ifm-alert-border-color:var(--ifm-color-danger-dark)}.alert{--ifm-code-background:var(--ifm-alert-background-color-highlight);--ifm-link-color:var(--ifm-alert-foreground-color);--ifm-link-hover-color:var(--ifm-alert-foreground-color);--ifm-link-decoration:underline;--ifm-tabs-color:var(--ifm-alert-foreground-color);--ifm-tabs-color-active:var(--ifm-alert-foreground-color);--ifm-tabs-color-active-border:var(--ifm-alert-border-color);background-color:var(--ifm-alert-background-color);border:var(--ifm-alert-border-width) solid var(--ifm-alert-border-color);border-left-width:var(--ifm-alert-border-left-width);border-radius:var(--ifm-alert-border-radius);box-shadow:var(--ifm-alert-shadow);padding:var(--ifm-alert-padding-vertical) var(--ifm-alert-padding-horizontal)}.alert__heading{align-items:center;display:flex;font:700 var(--ifm-h5-font-size)/var(--ifm-heading-line-height) var(--ifm-heading-font-family);margin-bottom:.5rem}.alert__icon{display:inline-flex;margin-right:.4em}.alert__icon svg{fill:var(--ifm-alert-foreground-color);stroke:var(--ifm-alert-foreground-color);stroke-width:0}.alert .close{margin:calc(var(--ifm-alert-padding-vertical)*-1) calc(var(--ifm-alert-padding-horizontal)*-1) 0 0;opacity:.75}.alert .close:focus,.alert .close:hover{opacity:1}.alert a{text-decoration-color:var(--ifm-alert-border-color)}.alert a:hover{text-decoration-thickness:2px}.avatar{column-gap:var(--ifm-avatar-intro-margin);display:flex}.avatar__photo{border-radius:50%;display:block;height:var(--ifm-avatar-photo-size);overflow:hidden;width:var(--ifm-avatar-photo-size)}.card--full-height,.navbar__logo img,body,html{height:100%}.avatar__photo--sm{--ifm-avatar-photo-size:2rem}.avatar__photo--lg{--ifm-avatar-photo-size:4rem}.avatar__photo--xl{--ifm-avatar-photo-size:6rem}.avatar__intro{display:flex;flex:1 1;flex-direction:column;justify-content:center;text-align:var(--ifm-avatar-intro-alignment)}.badge,.breadcrumbs__item,.breadcrumbs__link,.dropdown>.navbar__link:after,.searchBarContainer_NW3z.searchIndexLoading_EJ1f .searchBarLoadingRing_YnHq{display:inline-block}.avatar__name{font:700 var(--ifm-h4-font-size)/var(--ifm-heading-line-height) var(--ifm-font-family-base)}.avatar__subtitle{margin-top:.25rem}.avatar--vertical{--ifm-avatar-intro-alignment:center;--ifm-avatar-intro-margin:0.5rem;align-items:center;flex-direction:column}.badge{background-color:var(--ifm-badge-background-color);border:var(--ifm-badge-border-width) solid var(--ifm-badge-border-color);border-radius:var(--ifm-badge-border-radius);color:var(--ifm-badge-color);font-size:75%;font-weight:var(--ifm-font-weight-bold);line-height:1;padding:var(--ifm-badge-padding-vertical) var(--ifm-badge-padding-horizontal)}.badge--primary{--ifm-badge-background-color:var(--ifm-color-primary)}.badge--secondary{--ifm-badge-background-color:var(--ifm-color-secondary);color:var(--ifm-color-black)}.breadcrumbs__link,.button.button--secondary.button--outline:not(.button--active):not(:hover){color:var(--ifm-font-color-base)}.badge--success{--ifm-badge-background-color:var(--ifm-color-success)}.badge--info{--ifm-badge-background-color:var(--ifm-color-info)}.badge--warning{--ifm-badge-background-color:var(--ifm-color-warning)}.badge--danger{--ifm-badge-background-color:var(--ifm-color-danger)}.breadcrumbs{margin-bottom:0;padding-left:0}.breadcrumbs__item:not(:last-child):after{background:var(--ifm-breadcrumb-separator) center;content:" ";display:inline-block;filter:var(--ifm-breadcrumb-separator-filter);height:calc(var(--ifm-breadcrumb-separator-size)*var(--ifm-breadcrumb-size-multiplier)*var(--ifm-breadcrumb-separator-size-multiplier));margin:0 var(--ifm-breadcrumb-spacing);opacity:.5;width:calc(var(--ifm-breadcrumb-separator-size)*var(--ifm-breadcrumb-size-multiplier)*var(--ifm-breadcrumb-separator-size-multiplier))}.breadcrumbs__item--active .breadcrumbs__link{background:var(--ifm-breadcrumb-item-background-active);color:var(--ifm-breadcrumb-color-active)}.breadcrumbs__link{border-radius:var(--ifm-breadcrumb-border-radius);font-size:calc(1rem*var(--ifm-breadcrumb-size-multiplier));padding:calc(var(--ifm-breadcrumb-padding-vertical)*var(--ifm-breadcrumb-size-multiplier)) calc(var(--ifm-breadcrumb-padding-horizontal)*var(--ifm-breadcrumb-size-multiplier));transition-duration:var(--ifm-transition-fast);transition-property:background,color}.breadcrumbs__link:any-link:hover,.breadcrumbs__link:link:hover,.breadcrumbs__link:visited:hover,area[href].breadcrumbs__link:hover{background:var(--ifm-breadcrumb-item-background-active);-webkit-text-decoration:none;text-decoration:none}.breadcrumbs--sm{--ifm-breadcrumb-size-multiplier:0.8}.breadcrumbs--lg{--ifm-breadcrumb-size-multiplier:1.2}.button{background-color:var(--ifm-button-background-color);border:var(--ifm-button-border-width) solid var(--ifm-button-border-color);border-radius:var(--ifm-button-border-radius);font-size:calc(.875rem*var(--ifm-button-size-multiplier));font-weight:var(--ifm-button-font-weight);line-height:1.5;padding:calc(var(--ifm-button-padding-vertical)*var(--ifm-button-size-multiplier)) calc(var(--ifm-button-padding-horizontal)*var(--ifm-button-size-multiplier));text-align:center;transition-duration:var(--ifm-button-transition-duration);-webkit-user-select:none;user-select:none;white-space:nowrap}.button,.button:hover{color:var(--ifm-button-color)}.button--outline{--ifm-button-color:var(--ifm-button-border-color)}.button--outline:hover{--ifm-button-background-color:var(--ifm-button-border-color)}.button--link{--ifm-button-border-color:#0000;color:var(--ifm-link-color);text-decoration:var(--ifm-link-decoration)}.button--link.button--active,.button--link:active,.button--link:hover{color:var(--ifm-link-hover-color);text-decoration:var(--ifm-link-hover-decoration)}#carbonads a,.banner_woPo,.button-link-wrapper:hover,.dropdown__link--active,.dropdown__link:hover,.link2_y3x6:hover,.link_ecgS:hover,.menu__link:hover,.navLink_KmGT,.navLink_KmGT:hover,.navbar__brand:hover,.navbar__link--active,.navbar__link:hover,.pagination-nav__link:hover,.pagination__link:hover{-webkit-text-decoration:none;text-decoration:none}.button.disabled,.button:disabled,.button[disabled]{opacity:.65;pointer-events:none}.button--sm{--ifm-button-size-multiplier:0.8}.button--lg{--ifm-button-size-multiplier:1.35}.button--block{display:block;width:100%}.button.button--secondary{color:var(--ifm-color-gray-900)}:where(.button--primary){--ifm-button-background-color:var(--ifm-color-primary);--ifm-button-border-color:var(--ifm-color-primary)}:where(.button--primary):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-primary-dark);--ifm-button-border-color:var(--ifm-color-primary-dark)}.button--primary.button--active,.button--primary:active{--ifm-button-background-color:var(--ifm-color-primary-darker);--ifm-button-border-color:var(--ifm-color-primary-darker)}:where(.button--secondary){--ifm-button-background-color:var(--ifm-color-secondary);--ifm-button-border-color:var(--ifm-color-secondary)}:where(.button--secondary):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-secondary-dark);--ifm-button-border-color:var(--ifm-color-secondary-dark)}.button--secondary.button--active,.button--secondary:active{--ifm-button-background-color:var(--ifm-color-secondary-darker);--ifm-button-border-color:var(--ifm-color-secondary-darker)}:where(.button--success){--ifm-button-background-color:var(--ifm-color-success);--ifm-button-border-color:var(--ifm-color-success)}:where(.button--success):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-success-dark);--ifm-button-border-color:var(--ifm-color-success-dark)}.button--success.button--active,.button--success:active{--ifm-button-background-color:var(--ifm-color-success-darker);--ifm-button-border-color:var(--ifm-color-success-darker)}:where(.button--info){--ifm-button-background-color:var(--ifm-color-info);--ifm-button-border-color:var(--ifm-color-info)}:where(.button--info):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-info-dark);--ifm-button-border-color:var(--ifm-color-info-dark)}.button--info.button--active,.button--info:active{--ifm-button-background-color:var(--ifm-color-info-darker);--ifm-button-border-color:var(--ifm-color-info-darker)}:where(.button--warning){--ifm-button-background-color:var(--ifm-color-warning);--ifm-button-border-color:var(--ifm-color-warning)}:where(.button--warning):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-warning-dark);--ifm-button-border-color:var(--ifm-color-warning-dark)}.button--warning.button--active,.button--warning:active{--ifm-button-background-color:var(--ifm-color-warning-darker);--ifm-button-border-color:var(--ifm-color-warning-darker)}:where(.button--danger){--ifm-button-background-color:var(--ifm-color-danger);--ifm-button-border-color:var(--ifm-color-danger)}:where(.button--danger):not(.button--outline):hover{--ifm-button-background-color:var(--ifm-color-danger-dark);--ifm-button-border-color:var(--ifm-color-danger-dark)}.button--danger.button--active,.button--danger:active{--ifm-button-background-color:var(--ifm-color-danger-darker);--ifm-button-border-color:var(--ifm-color-danger-darker)}.button-group{display:inline-flex;gap:var(--ifm-button-group-spacing)}.button-group>.button:not(:first-child){border-bottom-left-radius:0;border-top-left-radius:0}.button-group>.button:not(:last-child){border-bottom-right-radius:0;border-top-right-radius:0}.button-group--block{display:flex;justify-content:stretch}.button-group--block>.button{flex-grow:1}.card{background-color:var(--ifm-card-background-color);border-radius:var(--ifm-card-border-radius);box-shadow:var(--ifm-global-shadow-lw);display:flex;flex-direction:column;overflow:hidden}.card__image{padding-top:var(--ifm-card-vertical-spacing)}.card__image:first-child{padding-top:0}.card__body,.card__footer,.card__header{padding:var(--ifm-card-vertical-spacing) var(--ifm-card-horizontal-spacing)}.card__body:not(:last-child),.card__footer:not(:last-child),.card__header:not(:last-child){padding-bottom:0}.card__body>:last-child,.card__footer>:last-child,.card__header>:last-child{margin-bottom:0}.card__footer{margin-top:auto}.table-of-contents{font-size:.8rem;margin-bottom:0;padding:var(--ifm-toc-padding-vertical) 0}.table-of-contents,.table-of-contents ul{list-style:none;padding-left:var(--ifm-toc-padding-horizontal)}.table-of-contents li{margin:var(--ifm-toc-padding-vertical) var(--ifm-toc-padding-horizontal)}.table-of-contents__left-border{border-left:1px solid var(--ifm-toc-border-color)}.table-of-contents__link{color:var(--ifm-toc-link-color);display:block}.table-of-contents__link--active,.table-of-contents__link--active code,.table-of-contents__link:hover,.table-of-contents__link:hover code{color:var(--ifm-color-primary);-webkit-text-decoration:none;text-decoration:none}.close{color:var(--ifm-color-black);font-size:1.5rem;font-weight:var(--ifm-font-weight-bold);line-height:1;opacity:.5;padding:1rem;transition:opacity var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.close:hover{opacity:.7}.close:focus,.theme-code-block-highlighted-line .codeLineNumber_Tfdd:before{opacity:.8}.dropdown{display:inline-flex;font-weight:var(--ifm-dropdown-font-weight);position:relative;vertical-align:top}.dropdown--hoverable:hover .dropdown__menu,.dropdown--show .dropdown__menu{opacity:1;pointer-events:all;transform:translateY(-1px);visibility:visible}.dropdown--right .dropdown__menu{left:inherit;right:0}.dropdown--nocaret .navbar__link:after{content:none!important}.dropdown__menu{background-color:var(--ifm-dropdown-background-color);border-radius:var(--ifm-global-radius);box-shadow:var(--ifm-global-shadow-md);left:0;max-height:80vh;min-width:10rem;opacity:0;overflow-y:auto;padding:.5rem;pointer-events:none;position:absolute;top:calc(100% - var(--ifm-navbar-item-padding-vertical) + .3rem);transform:translateY(-.625rem);transition-duration:var(--ifm-transition-fast);transition-property:opacity,transform,visibility;transition-timing-function:var(--ifm-transition-timing-default);visibility:hidden;z-index:var(--ifm-z-index-dropdown)}.menu__caret,.menu__link,.menu__list-item-collapsible{border-radius:.25rem;transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.dropdown__link{border-radius:.25rem;color:var(--ifm-dropdown-link-color);display:block;font-size:.875rem;margin-top:.2rem;padding:.25rem .5rem;white-space:nowrap}.dropdown__link--active,.dropdown__link:hover{background-color:var(--ifm-dropdown-hover-background-color);color:var(--ifm-dropdown-link-color)}.dropdown__link--active,.dropdown__link--active:hover{--ifm-dropdown-link-color:var(--ifm-link-color)}.dropdown>.navbar__link:after{border-color:currentcolor #0000;border-style:solid;border-width:.4em .4em 0;content:"";margin-left:.3em;position:relative;top:2px;transform:translateY(-50%)}.footer{background-color:var(--ifm-footer-background-color);color:var(--ifm-footer-color);padding:var(--ifm-footer-padding-vertical) var(--ifm-footer-padding-horizontal)}.footer--dark{--ifm-footer-background-color:#303846;--ifm-footer-color:var(--ifm-footer-link-color);--ifm-footer-link-color:var(--ifm-color-secondary);--ifm-footer-title-color:var(--ifm-color-white)}.footer__links{margin-bottom:1rem}.footer__link-item{color:var(--ifm-footer-link-color);line-height:2}.footer__link-item:hover{color:var(--ifm-footer-link-hover-color)}.footer__link-separator{margin:0 var(--ifm-footer-link-horizontal-spacing)}.footer__logo{margin-top:1rem;max-width:var(--ifm-footer-logo-max-width)}.footer__title{color:var(--ifm-footer-title-color);font:700 var(--ifm-h4-font-size)/var(--ifm-heading-line-height) var(--ifm-font-family-base);margin-bottom:var(--ifm-heading-margin-bottom)}.menu,.navbar__link{font-weight:var(--ifm-font-weight-semibold)}.docItemContainer_Djhp article>:first-child,.docItemContainer_Djhp header+*,.footer__item{margin-top:0}.admonitionContent_BuS1>:last-child,.collapsibleContent_i85q p:last-child,.details_lb9f>summary>p:last-child,.footer__items,.searchResultItem_U687>h2{margin-bottom:0}.codeBlockStandalone_MEMb,[type=checkbox]{padding:0}.hero{align-items:center;background-color:var(--ifm-hero-background-color);color:var(--ifm-hero-text-color);display:flex;padding:4rem 2rem}.hero--primary{--ifm-hero-background-color:var(--ifm-color-primary);--ifm-hero-text-color:var(--ifm-font-color-base-inverse)}.hero--dark{--ifm-hero-background-color:#303846;--ifm-hero-text-color:var(--ifm-color-white)}.hero__title,h2{font-size:3rem}.hero__subtitle{font-size:1.5rem}.menu__list{margin:0;padding-left:0}.menu__caret,.menu__link{padding:var(--ifm-menu-link-padding-vertical) var(--ifm-menu-link-padding-horizontal)}.menu__list .menu__list{flex:0 0 100%;margin-top:.25rem;padding-left:var(--ifm-menu-link-padding-horizontal)}.menu__list-item:not(:first-child){margin-top:.25rem}.menu__list-item--collapsed .menu__list{height:0;overflow:hidden}.details_lb9f[data-collapsed=false].isBrowser_bmU9>summary:before,.details_lb9f[open]:not(.isBrowser_bmU9)>summary:before,.menu__list-item--collapsed .menu__caret:before,.menu__list-item--collapsed .menu__link--sublist:after{transform:rotate(90deg)}.menu__list-item-collapsible{display:flex;flex-wrap:wrap;position:relative}.menu__caret:hover,.menu__link:hover,.menu__list-item-collapsible--active,.menu__list-item-collapsible:hover{background:var(--ifm-menu-color-background-hover)}.menu__list-item-collapsible .menu__link--active,.menu__list-item-collapsible .menu__link:hover{background:none!important}.menu__caret,.menu__link{align-items:center;display:flex}.navbar-sidebar,.navbar-sidebar__backdrop{bottom:0;opacity:0;transition-duration:var(--ifm-transition-fast);transition-timing-function:ease-in-out;top:0;left:0;visibility:hidden}.menu__link{color:var(--ifm-menu-color);flex:1;line-height:1.25}.menu__link:hover{color:var(--ifm-menu-color)}.menu__caret:before,.menu__link--sublist-caret:after{height:1.25rem;transform:rotate(180deg);transition:transform var(--ifm-transition-fast) linear;width:1.25rem;content:"";filter:var(--ifm-menu-link-sublist-icon-filter)}.menu__link--sublist-caret:after{background:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem;margin-left:auto;min-width:1.25rem}.menu__link--active,.menu__link--active:hover{color:var(--ifm-menu-color-active)}.navbar__brand,.navbar__link{color:var(--ifm-navbar-link-color)}.menu__link--active:not(.menu__link--sublist){background-color:var(--ifm-menu-color-background-active)}.menu__caret:before{background:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem}.navbar--dark,html[data-theme=dark]{--ifm-menu-link-sublist-icon-filter:invert(100%) sepia(94%) saturate(17%) hue-rotate(223deg) brightness(104%) contrast(98%)}.navbar{background-color:var(--ifm-navbar-background-color);box-shadow:var(--ifm-navbar-shadow);height:var(--ifm-navbar-height);padding:var(--ifm-navbar-padding-vertical) var(--ifm-navbar-padding-horizontal)}.navbar,.navbar>.container,.navbar>.container-fluid{display:flex}.navbar--fixed-top{position:sticky;top:0;z-index:var(--ifm-z-index-fixed)}.navbar__inner{display:flex;flex-wrap:wrap;justify-content:space-between;width:100%}.navbar__brand{align-items:center;display:flex;margin-right:1rem;min-width:0}.navbar__brand:hover{color:var(--ifm-navbar-link-hover-color)}.announcementBarContent_xLdY,.navbar__title{flex:1 1 auto}.navbar__toggle{display:none;margin-right:.5rem}.navbar__logo{flex:0 0 auto;height:2rem;margin-right:.5rem}.navbar__items{align-items:center;display:flex;flex:1;min-width:0}.navbar__items--center{flex:0 0 auto}.fetchErrorCard_M8w3 p,.home-page-features-wrapper .feature .feature-title h3,.navbar__items--center .navbar__brand{margin:0}.navbar__items--center+.navbar__items--right{flex:1}.navbar__items--right{flex:0 0 auto;justify-content:flex-end}.navbar__item{display:inline-block;padding:var(--ifm-navbar-item-padding-vertical) var(--ifm-navbar-item-padding-horizontal)}#nprogress,.navbar__item.dropdown .navbar__link:not([href]){pointer-events:none}.navbar__link--active,.navbar__link:hover{color:var(--ifm-navbar-link-hover-color)}.navbar--dark,.navbar--primary{--ifm-menu-color:var(--ifm-color-gray-300);--ifm-navbar-link-color:var(--ifm-color-gray-100);--ifm-navbar-search-input-background-color:#ffffff1a;--ifm-navbar-search-input-placeholder-color:#ffffff80;color:var(--ifm-color-white)}.navbar--dark{--ifm-navbar-background-color:#242526;--ifm-menu-color-background-active:#ffffff0d;--ifm-navbar-search-input-color:var(--ifm-color-white)}.navbar--primary{--ifm-navbar-background-color:var(--ifm-color-primary);--ifm-navbar-link-hover-color:var(--ifm-color-white);--ifm-menu-color-active:var(--ifm-color-white);--ifm-navbar-search-input-color:var(--ifm-color-emphasis-500)}.navbar__search-input{appearance:none;background:var(--ifm-navbar-search-input-background-color) var(--ifm-navbar-search-input-icon) no-repeat .75rem center/1rem 1rem;border:none;border-radius:2rem;color:var(--ifm-navbar-search-input-color);cursor:text;display:inline-block;font-size:1rem;height:2rem;padding:0 .5rem 0 2.25rem;width:12.5rem}.navbar__search-input::placeholder{color:var(--ifm-navbar-search-input-placeholder-color)}.navbar-sidebar{background-color:var(--ifm-navbar-background-color);box-shadow:var(--ifm-global-shadow-md);position:fixed;transform:translate3d(-100%,0,0);transition-property:opacity,visibility,transform;width:var(--ifm-navbar-sidebar-width)}.navbar-sidebar--show .navbar-sidebar,.navbar-sidebar__items{transform:translateZ(0)}.navbar-sidebar--show .navbar-sidebar,.navbar-sidebar--show .navbar-sidebar__backdrop{opacity:1;visibility:visible}.navbar-sidebar__backdrop{background-color:#0009;position:fixed;right:0;transition-property:opacity,visibility}.navbar-sidebar__brand{align-items:center;box-shadow:var(--ifm-navbar-shadow);display:flex;flex:1;height:var(--ifm-navbar-height);padding:var(--ifm-navbar-padding-vertical) var(--ifm-navbar-padding-horizontal)}.footer,article div.markdown p img,nav.navbar{box-shadow:var(--feature-img-shadow)}.navbar-sidebar__items{display:flex;height:calc(100% - var(--ifm-navbar-height));transition:transform var(--ifm-transition-fast) ease-in-out}.navbar-sidebar__items--show-secondary{transform:translate3d(calc((var(--ifm-navbar-sidebar-width))*-1),0,0)}.navbar-sidebar__item{flex-shrink:0;padding:.5rem;width:calc(var(--ifm-navbar-sidebar-width))}.navbar-sidebar__back{background:var(--ifm-menu-color-background-active);font-size:15px;font-weight:var(--ifm-button-font-weight);margin:0 0 .2rem -.5rem;padding:.6rem 1.5rem;position:relative;text-align:left;top:-.5rem;width:calc(100% + 1rem)}.navbar-sidebar__close{display:flex;margin-left:auto}.pagination{column-gap:var(--ifm-pagination-page-spacing);display:flex;font-size:var(--ifm-pagination-font-size);padding-left:0}.pagination--sm{--ifm-pagination-font-size:0.8rem;--ifm-pagination-padding-horizontal:0.8rem;--ifm-pagination-padding-vertical:0.2rem}.pagination--lg{--ifm-pagination-font-size:1.2rem;--ifm-pagination-padding-horizontal:1.2rem;--ifm-pagination-padding-vertical:0.3rem}.pagination__item{display:inline-flex}.pagination__item>span{padding:var(--ifm-pagination-padding-vertical)}.pagination__item--active .pagination__link{color:var(--ifm-pagination-color-active)}.pagination__item--active .pagination__link,.pagination__item:not(.pagination__item--active):hover .pagination__link{background:var(--ifm-pagination-item-active-background)}.pagination__item--disabled,.pagination__item[disabled]{opacity:.25;pointer-events:none}.pagination__link{border-radius:var(--ifm-pagination-border-radius);color:var(--ifm-font-color-base);display:inline-block;padding:var(--ifm-pagination-padding-vertical) var(--ifm-pagination-padding-horizontal);transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pagination-nav{display:grid;grid-gap:var(--ifm-spacing-horizontal);gap:var(--ifm-spacing-horizontal);grid-template-columns:repeat(2,1fr)}.pagination-nav__link{border:1px solid var(--ifm-color-emphasis-300);border-radius:var(--ifm-pagination-nav-border-radius);display:block;height:100%;line-height:var(--ifm-heading-line-height);padding:var(--ifm-global-spacing);transition:border-color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pagination-nav__link:hover{border-color:var(--ifm-pagination-nav-color-hover)}.content_knG7 a,.hitFooter_E9YW a,.link2_y3x6.link2_y3x6,.link2_y3x6.link2_y3x6:hover,.link_ecgS.link2_y3x6,.link_ecgS.link2_y3x6:hover,.suggestion_fB_2.cursor_eG29 mark,li a:hover,p a:hover{-webkit-text-decoration:underline;text-decoration:underline}.pagination-nav__link--next{grid-column:2/3;text-align:right}.pagination-nav__label{font-size:var(--ifm-h4-font-size);font-weight:var(--ifm-heading-font-weight);word-break:break-word}.pagination-nav__link--prev .pagination-nav__label:before{content:"« "}.pagination-nav__link--next .pagination-nav__label:after{content:" »"}.pagination-nav__sublabel{color:var(--ifm-color-content-secondary);font-size:var(--ifm-h5-font-size);font-weight:var(--ifm-font-weight-semibold);margin-bottom:.25rem}.pills__item,.tabs{font-weight:var(--ifm-font-weight-bold)}.pills{display:flex;gap:var(--ifm-pills-spacing);padding-left:0}.pills__item{border-radius:.5rem;cursor:pointer;display:inline-block;padding:.25rem 1rem;transition:background var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.pills__item--active{color:var(--ifm-pills-color-active)}.pills__item--active,.pills__item:not(.pills__item--active):hover{background:var(--ifm-pills-color-background-active)}.pills--block{justify-content:stretch}.pills--block .pills__item{flex-grow:1;text-align:center}.tabs{color:var(--ifm-tabs-color);display:flex;margin-bottom:0;overflow-x:auto;padding-left:0}.tabs__item{border-bottom:3px solid #0000;border-radius:var(--ifm-global-radius);cursor:pointer;display:inline-flex;padding:var(--ifm-tabs-padding-vertical) var(--ifm-tabs-padding-horizontal);transition:background-color var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.tabs__item--active{border-bottom-color:var(--ifm-tabs-color-active-border);border-bottom-left-radius:0;border-bottom-right-radius:0;color:var(--ifm-tabs-color-active)}.tabs__item:hover{background-color:var(--ifm-hover-overlay)}.tabs--block{justify-content:stretch}.tabs--block .tabs__item{flex-grow:1;justify-content:center}html[data-theme=dark]{--ifm-color-scheme:dark;--ifm-color-emphasis-0:var(--ifm-color-gray-1000);--ifm-color-emphasis-100:var(--ifm-color-gray-900);--ifm-color-emphasis-200:var(--ifm-color-gray-800);--ifm-color-emphasis-300:var(--ifm-color-gray-700);--ifm-color-emphasis-400:var(--ifm-color-gray-600);--ifm-color-emphasis-600:var(--ifm-color-gray-400);--ifm-color-emphasis-700:var(--ifm-color-gray-300);--ifm-color-emphasis-800:var(--ifm-color-gray-200);--ifm-color-emphasis-900:var(--ifm-color-gray-100);--ifm-color-emphasis-1000:var(--ifm-color-gray-0);--ifm-background-color:#1b1b1d;--ifm-background-surface-color:#242526;--ifm-hover-overlay:#ffffff0d;--ifm-color-content:#e3e3e3;--ifm-color-content-secondary:#fff;--ifm-breadcrumb-separator-filter:invert(64%) sepia(11%) saturate(0%) hue-rotate(149deg) brightness(99%) contrast(95%);--ifm-code-background:#ffffff1a;--ifm-scrollbar-track-background-color:#444;--ifm-scrollbar-thumb-background-color:#686868;--ifm-scrollbar-thumb-hover-background-color:#7a7a7a;--ifm-table-stripe-background:#ffffff12;--ifm-toc-border-color:var(--ifm-color-emphasis-200);--ifm-color-primary-contrast-background:#102445;--ifm-color-primary-contrast-foreground:#ebf2fc;--ifm-color-secondary-contrast-background:#474748;--ifm-color-secondary-contrast-foreground:#fdfdfe;--ifm-color-success-contrast-background:#003100;--ifm-color-success-contrast-foreground:#e6f6e6;--ifm-color-info-contrast-background:#193c47;--ifm-color-info-contrast-foreground:#eef9fd;--ifm-color-warning-contrast-background:#4d3800;--ifm-color-warning-contrast-foreground:#fff8e6;--ifm-color-danger-contrast-background:#4b1113;--ifm-color-danger-contrast-foreground:#ffebec;--primary:#db78fc;--background:var(--dark-grey);--text-color:var(--white);--hero-background:var(--black);--kinda-transparent:#000000d9;--hyperlink:var(--primary);--feature-img-shadow:4px 4px 6px #00000080,-2px -2px 4px #0006;--footer-color:var(--black);--footer-text-color:var(--white);--heading-shadow:-5px 4px 0px #000;--sub-heading-shadow:-3px 3px 1px #000}:root{--docusaurus-progress-bar-color:var(--ifm-color-primary);--bright-white:#fff;--white:#f7f7f7;--pale-grey:#e9e9e8;--mid-grey:#a9a9a9;--dark-grey:#18191a;--black:#121212;--pitch-black:#000;--info:#35c9fa;--success:#8f8;--warning:#ece715;--danger:#f80363;--accent-green:#41ef90;--accent-green-dark:#1e9554;--accent-blue:#5c85f7;--accent-blue-dark:#3d48dd;--accent-purple:#9083ed;--accent-purple-dark:#6b3cd6;--accent-pink:#db78fc;--accent-pink-dark:#b83ddd;--accent-yellow:#dcff5a;--accent-yellow-dark:#ceb73f;--ifm-color-primary:var(--background);--ifm-color-primary-dark:#21af90;--ifm-color-primary-darker:#1fa588;--ifm-color-primary-darkest:#1a8870;--ifm-color-primary-light:#46cbae;--ifm-color-primary-lighter:#66d4bd;--ifm-color-primary-lightest:#92e0d0;--ifm-code-font-size:95%;--ifm-navbar-background-color:var(--hero-background);--ifm-navbar-link-color:var(--text-color);--ifm-breadcrumb-color-active:var(--text-color);--docusaurus-tag-list-border:var(--ifm-color-emphasis-300);--docusaurus-announcement-bar-height:auto;--docusaurus-collapse-button-bg:#0000;--docusaurus-collapse-button-bg-hover:#0000001a;--doc-sidebar-width:300px;--doc-sidebar-hidden-width:30px}#nprogress .bar{background:var(--docusaurus-progress-bar-color);height:2px;left:0;position:fixed;top:0;width:100%;z-index:1031}#nprogress .peg{box-shadow:0 0 10px var(--docusaurus-progress-bar-color),0 0 5px var(--docusaurus-progress-bar-color);height:100%;opacity:1;position:absolute;right:0;transform:rotate(3deg) translateY(-4px);width:100px}h1{font-family:Racing Sans One,mono;font-size:4rem}button,h2,h3,h4,h5{font-family:Roboto Mono,monospace}h3{font-size:2rem}button,h4,h5{font-size:1.2rem}a,body,div,p,section,span,ul li{font-family:Poppins,sans-serif;font-size:1.2rem}.link2_y3x6,.link_ecgS,code span{font-family:Roboto Mono,monospace}.hero{--ifm-hero-background-color:var(--hero-background);--ifm-hero-text-color:var(--text-color)}.footer{--ifm-footer-background-color:var(--footer-color);--ifm-footer-color:var(--footer-text-color);--ifm-footer-link-color:var(--footer-text-color);--ifm-footer-title-color:var(--footer-text-color);--ifm-footer-link-hover-color:var(--primary);--ifm-link-color:var(--footer-text-color);--ifm-link-hover-color:var(--primary)}.footer .footer__link-item,.sidebar-ad #carbonads .carbon-text,article div.markdown a,article div.markdown code span,article div.markdown li,article div.markdown ol,article div.markdown p,article div.markdown ul,ul.table-of-contents li a{font-size:1rem}.footer .footer__copyright,.footer .footer__copyright a{font-family:Roboto Mono,monospace;font-size:.9rem;font-weight:700;opacity:.9}.docusaurus-highlight-code-line{background-color:#0000001a;display:block;margin:0 calc(var(--ifm-pre-padding)*-1);padding:0 var(--ifm-pre-padding)}nav.navbar a.navbar__brand .navbar__logo,nav.navbar a.navbar__brand b{transition:.3s}nav.navbar a.navbar__brand:hover .navbar__logo{transform:scale(1.1) rotate(-3deg)}nav.navbar a.navbar__brand:hover b{color:#00eed6}nav.navbar a.navbar__brand b{font-size:1.5rem;font-weight:700}.authorLink_YWtN:hover,nav.navbar .navbar__link--active,nav.navbar .navbar__link:hover{color:var(--primary)}.fetchErrorCard_M8w3 a:hover,.meta_GplF a:hover,.pagination-nav__label,html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg .pagination-nav__label{color:var(--text-color)}li a,li a:hover,p a,p a:hover,ul.table-of-contents li a:hover,ul.table-of-contents li a:hover code{color:var(--hyperlink)}h1,h2,h3,h4,h5{cursor:default}article div.markdown{--ifm-h1-font-size:3.0rem;--ifm-h2-font-size:2.5rem;--ifm-h3-font-size:1.4rem}article div.markdown p img{border-radius:4px}.sponsor_JDQ1,.sponsor_JDQ1 img,article div.markdown table>tr>td>a>img{border-radius:6px}ul.table-of-contents li a.table-of-contents__link--active,ul.table-of-contents li a.table-of-contents__link--active code{color:var(--hyperlink);font-weight:700}ul.menu__list{margin-bottom:1rem}ul.menu__list li.menu__list-item a.active,ul.menu__list li.menu__list-item a.menu__link--active{color:var(--primary);font-weight:700}code span{font-size:.9rem!important}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg{padding:0 1rem!important}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg div>div.col[class*=docItemCol]{background:var(--bright-white);border-right:1px solid var(--ifm-toc-border-color);padding:1rem}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg div>div.col[class*=docItemCol] ul li a{font-weight:700}html[data-theme=light] main div.container.padding-top--md.padding-bottom--lg .table-of-contents__left-border{border-left:none}#carbonads *{line-height:normal;margin:initial;padding:initial}#carbonads{--carbon-font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,Oxygen-Sans,Ubuntu,Cantarell,"Helvetica Neue",Helvetica,Arial,sans-serif;--carbon-font-size:14px;--carbon-padding:1.5ch;--carbon-max-char:20ch;--carbon-bg-primary:#fafafa;--carbon-bg-secondary:#ebebeb;--carbon-text-color:#333;font-family:var(--carbon-font-family);font-size:var(--carbon-font-size);margin:0 auto;width:fit-content;z-index:10}#carbonads>span{background-color:var(--carbon-bg-primary);box-shadow:0 0 1px rgba(0,0,0,.085),0 0 2px rgba(0,0,0,.085),0 0 4px rgba(0,0,0,.085),0 0 8px rgba(0,0,0,.085);display:flex;flex-direction:column;gap:var(--carbon-padding);max-inline-size:calc(130px + var(--carbon-max-char) + 8ch);min-inline-size:130px;padding:var(--carbon-padding)}#carbonads .carbon-wrap{display:flex;flex-wrap:wrap;gap:1.5ex}#carbonads .carbon-img{flex:0 0 130px}#carbonads .carbon-img img,.cursor_eG29 .hideAction_vcyE>svg,.tocCollapsibleContent_vkbj a{display:block}#carbonads .carbon-text{flex-basis:var(--carbon-max-char);flex-grow:1;line-height:1.4;text-align:left}#carbonads .carbon-poweredby{background:var(--carbon-bg-secondary);font-size:.6em;font-weight:600;letter-spacing:.2ch;line-height:1.4;padding:6px 8px;text-align:center;text-transform:uppercase}.sidebar-ad{--carbon-bg-primary:var(--background,#18191a);--carbon-bg-secondary:#282a36;--carbon-text-color:#e6e6e6}.sidebar-ad #carbonads{margin:.5rem}.sidebar-ad #carbonads .carbon-img img,.sidebar-ad .avatar__photo-link{border-radius:5px}.mdxPageWrapper_j9I6,.sidebar-ad #carbonads .carbon-wrap{justify-content:center}.banner_woPo,.tagWithCount_h2kH{align-items:center;display:flex}.sidebar-ad #carbonads>span{box-shadow:none}.sidebar-ad #carbonads .carbon-poweredby{border-radius:5px;font-size:.7rem}::selection{background:var(--primary);color:var(--background);text-shadow:none}@keyframes a{0%{opacity:0;transform:translateY(-100%)}to{opacity:1;transform:translateY(0)}}@keyframes b{0%{opacity:1}to{opacity:0;transform:translateY(-100%)}}.banner_woPo{animation:.3s ease-out a;background:var(--accent-green);color:var(--bright-white);flex-wrap:wrap;font-size:.95rem;font-weight:600;gap:.25rem 1.25rem;justify-content:center;padding:.25rem 1rem;position:relative;width:100%;z-index:100}.dismissing_ZrNM{animation:.25s ease-in forwards b}.link2_y3x6,.link_ecgS{color:var(--black);font-size:.95rem;font-weight:400;transition:.2s}.link2_y3x6.link2_y3x6,.link_ecgS.link2_y3x6{opacity:.85}.link2_y3x6.link2_y3x6:hover,.link_ecgS.link2_y3x6:hover{opacity:1;transform:scale(1.03)}.closeBtn_fC0A{background:none;border:none;border-radius:4px;color:var(--black);cursor:pointer;font-size:1.5rem;line-height:1;opacity:.8;padding:0 .25rem .1rem;position:absolute;right:1rem;transition:.2s}.closeBtn_fC0A:hover{background:#1d8c4f;opacity:1}.tag_zVej{border:1px solid var(--docusaurus-tag-list-border);transition:border var(--ifm-transition-fast)}.tag_zVej:hover{--docusaurus-tag-list-border:var(--ifm-link-color);-webkit-text-decoration:none;text-decoration:none}.tagRegular_sFm0{border-radius:var(--ifm-global-radius);font-size:90%;padding:.2rem .5rem .3rem}.tagWithCount_h2kH{border-left:0;padding:0 .5rem 0 1rem;position:relative}.tagWithCount_h2kH:after,.tagWithCount_h2kH:before{border:1px solid var(--docusaurus-tag-list-border);content:"";position:absolute;top:50%;transition:inherit}.tagWithCount_h2kH:before{border-bottom:0;border-right:0;height:1.18rem;right:100%;transform:translate(50%,-50%) rotate(-45deg);width:1.18rem}.tagWithCount_h2kH:after{border-radius:50%;height:.5rem;left:0;transform:translateY(-50%);width:.5rem}.tagWithCount_h2kH span{background:var(--ifm-color-secondary);border-radius:var(--ifm-global-radius);color:var(--ifm-color-black);font-size:.7rem;line-height:1.2;margin-left:.3rem;padding:.1rem .4rem}.tags_jXut{display:inline}.tag_QGVx{display:inline-block;margin:0 .4rem .5rem 0}.backToTopButton_sjWU{background-color:var(--ifm-color-emphasis-200);border-radius:50%;bottom:1.3rem;box-shadow:var(--ifm-global-shadow-lw);height:3rem;opacity:0;position:fixed;right:1.3rem;transform:scale(0);transition:all var(--ifm-transition-fast) var(--ifm-transition-timing-default);visibility:hidden;width:3rem;z-index:calc(var(--ifm-z-index-fixed) - 1)}.backToTopButton_sjWU:after{background-color:var(--ifm-color-emphasis-1000);content:" ";display:inline-block;height:100%;-webkit-mask:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem no-repeat;mask:var(--ifm-menu-link-sublist-icon) 50%/2rem 2rem no-repeat;width:100%}.backToTopButtonShow_xfvO{opacity:1;transform:scale(1);visibility:visible}body:not(.navigation-with-keyboard) :not(input):focus{outline:0}#__docusaurus-base-url-issue-banner-container,.docSidebarContainer_YfHR,.hideAction_vcyE>svg,.navbarSearchContainer_Bca1:empty,.sidebarLogo_isFc,.themedComponent_mlkZ,.toggleIcon_g3eP,html[data-announcement-bar-initially-dismissed=true] .announcementBar_mb4j{display:none}.skipToContent_fXgn{background-color:var(--ifm-background-surface-color);color:var(--ifm-color-emphasis-900);left:100%;padding:calc(var(--ifm-global-spacing)/2) var(--ifm-global-spacing);position:fixed;top:1rem;z-index:calc(var(--ifm-z-index-fixed) + 1)}.skipToContent_fXgn:focus{box-shadow:var(--ifm-global-shadow-md);left:1rem}.closeButton_CVFx{line-height:0;padding:0}.content_knG7{font-size:85%;padding:5px 0;text-align:center}.content_knG7 a{color:inherit}.announcementBar_mb4j{align-items:center;background-color:var(--ifm-color-white);border-bottom:1px solid var(--ifm-color-emphasis-100);color:var(--ifm-color-black);display:flex;height:var(--docusaurus-announcement-bar-height)}.announcementBarPlaceholder_vyr4{flex:0 0 10px}.announcementBarClose_gvF7{align-self:stretch;flex:0 0 30px}.toggle_vylO{height:2rem;width:2rem}.toggleButton_gllP{align-items:center;border-radius:50%;display:flex;height:100%;justify-content:center;transition:background var(--ifm-transition-fast);width:100%}.toggleButton_gllP:hover{background:var(--ifm-color-emphasis-200)}[data-theme-choice=dark] .darkToggleIcon_wfgR,[data-theme-choice=light] .lightToggleIcon_pyhR,[data-theme-choice=system] .systemToggleIcon_QzmC,[data-theme=dark] .themedComponent--dark_xIcU,[data-theme=light] .themedComponent--light_NVdE,html:not([data-theme]) .themedComponent--light_NVdE{display:initial}.toggleButtonDisabled_aARS{cursor:not-allowed}.darkNavbarColorModeToggle_X3D1:hover{background:var(--ifm-color-gray-800)}[data-theme=dark]:root{--docusaurus-collapse-button-bg:#ffffff0d;--docusaurus-collapse-button-bg-hover:#ffffff1a}.collapseSidebarButton_PEFL{display:none;margin:0}.categoryLinkLabel_W154,.linkLabel_WmDU{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden}.iconExternalLink_nPIU{margin-left:.3rem}.linkLabel_WmDU{line-clamp:2;-webkit-line-clamp:2}.categoryLink_byQd{overflow:hidden}.menu__link--sublist-caret:after{margin-left:var(--ifm-menu-link-padding-vertical)}.categoryLinkLabel_W154{flex:1;line-clamp:2;-webkit-line-clamp:2}.docMainContainer_TBSr,.docRoot_UBD9{display:flex;width:100%}.docsWrapper_hBAB{display:flex;flex:1 0 auto}.notFoundPage_Wsy5{align-items:center;background:var(--hero-background);display:flex;justify-content:center;min-height:calc(100vh - 4rem);overflow:hidden;position:relative}.container_a6qs{padding:2rem;position:relative;text-align:center;z-index:2}.glitchCode_Pt6F{animation:3s ease-in-out infinite d;color:var(--primary);font-family:Racing Sans One,monospace;font-size:10rem;font-weight:900;line-height:1;position:relative;text-shadow:0 0 20px color-mix(in srgb,var(--primary) 40%,#0000),0 0 60px color-mix(in srgb,var(--primary) 20%,#0000)}.badge_qP1g,.count_Ad5j,.monthHeader_QFki,.sha_JsPS,.tagCard_SluW .entryTitle_RUqs,.tagPill_IKwa{font-family:Roboto Mono,monospace}.title_B9uO{color:var(--text-color);font-size:2rem;font-weight:700;margin:1rem 0 .5rem}.subtitle_Z226{color:var(--mid-grey);font-size:1.1rem;line-height:1.6;margin-bottom:2.5rem;margin-left:auto;margin-right:auto;max-width:480px}.navLinks_NElv{display:flex;flex-wrap:wrap;gap:1rem;justify-content:center;margin-top:1rem}.navLink_KmGT{align-items:center;border-radius:8px;cursor:pointer;display:inline-flex;font-size:1rem;font-weight:600;gap:.5rem;padding:.75rem 1.5rem;transition:.25s}.navLink_KmGT:hover{transform:translateY(-2px)}.primaryLink_UP3n{background:var(--primary);color:var(--hero-background)}.primaryLink_UP3n:hover{box-shadow:0 4px 20px color-mix(in srgb,var(--primary) 50%,#0000);color:var(--hero-background)}.secondaryLink_VStI{background:#0000;border:2px solid var(--primary);color:var(--primary)}.secondaryLink_VStI:hover{background:color-mix(in srgb,var(--primary) 10%,#0000);color:var(--primary)}.particles_xv3x{inset:0;overflow:hidden;position:absolute;z-index:1}.particle_Q0bD{animation:linear infinite c;background:var(--primary);border-radius:50%;height:4px;opacity:.3;position:absolute;width:4px}.particle_Q0bD:first-child{animation-delay:-5s;animation-duration:17s;height:3px;left:88%;opacity:.13;top:58%;width:6px}.particle_Q0bD:nth-child(2){animation-delay:-2s;animation-duration:13s;height:5px;left:20%;opacity:.29;top:90%;width:5px}.particle_Q0bD:nth-child(3){animation-delay:-2s;animation-duration:16s;height:3px;left:87%;opacity:.11;top:95%;width:3px}.particle_Q0bD:nth-child(4){animation-delay:-4s;animation-duration:18s;height:4px;left:56%;opacity:.15;top:47%;width:5px}.particle_Q0bD:nth-child(5){animation-delay:-1s;animation-duration:13s;height:6px;left:91%;opacity:.33;top:70%;width:5px}.particle_Q0bD:nth-child(6){animation-delay:-3s;animation-duration:15s;height:3px;left:22%;opacity:.12;top:5%;width:3px}.particle_Q0bD:nth-child(7){animation-delay:-5s;animation-duration:15s;height:5px;left:42%;opacity:.14;top:98%;width:3px}.particle_Q0bD:nth-child(8){animation-delay:-3s;animation-duration:15s;height:5px;left:23%;opacity:.19;top:53%;width:6px}.particle_Q0bD:nth-child(9){animation-delay:-3s;animation-duration:19s;height:6px;left:59%;opacity:.26;top:5%;width:6px}.particle_Q0bD:nth-child(10){animation-delay:-5s;animation-duration:9s;height:3px;left:60%;opacity:.36;top:70%;width:5px}.particle_Q0bD:nth-child(11){animation-delay:-2s;animation-duration:17s;height:3px;left:46%;opacity:.36;top:35%;width:4px}.particle_Q0bD:nth-child(12){animation-delay:-2s;animation-duration:10s;height:6px;left:15%;opacity:.14;top:47%;width:5px}.particle_Q0bD:nth-child(13){animation-delay:-5s;animation-duration:16s;height:3px;left:22%;opacity:.28;top:66%;width:4px}.particle_Q0bD:nth-child(14){animation-delay:-3s;animation-duration:10s;height:5px;left:59%;opacity:.35;top:6%;width:6px}.particle_Q0bD:nth-child(15){animation-delay:-2s;animation-duration:18s;height:4px;left:24%;opacity:.38;top:51%;width:4px}.particle_Q0bD:nth-child(16){animation-delay:-2s;animation-duration:9s;height:4px;left:33%;opacity:.11;top:71%;width:4px}.particle_Q0bD:nth-child(17){animation-delay:-1s;animation-duration:11s;height:5px;left:13%;opacity:.16;top:84%;width:6px}.particle_Q0bD:nth-child(18){animation-delay:-2s;animation-duration:14s;height:4px;left:87%;opacity:.16;top:33%;width:5px}.particle_Q0bD:nth-child(19){animation-delay:-2s;animation-duration:10s;height:3px;left:29%;opacity:.21;top:60%;width:4px}.particle_Q0bD:nth-child(20){animation-delay:-2s;animation-duration:13s;height:4px;left:10%;opacity:.33;top:96%;width:6px}@keyframes c{0%,to{opacity:0;transform:translateY(0) translateX(0)}10%,90%{opacity:.3}50%{transform:translateY(-100px) translateX(30px)}}@keyframes d{0%,to{opacity:1}50%{opacity:.85}}.dropdownNavbarItemMobile_J0Sd{cursor:pointer}.iconLanguage_nlXk{margin-right:5px;vertical-align:text-bottom}.searchBar_RVTs .dropdownMenu_qbY6{background:var(--search-local-modal-background,#f5f6f7);border-radius:6px;box-shadow:var(--search-local-modal-shadow,inset 1px 1px 0 0 #ffffff80,0 3px 8px 0 #555a64);left:auto!important;margin-top:8px;padding:var(--search-local-spacing,12px);position:relative;right:0!important;width:var(--search-local-modal-width,560px)}.searchInput_YFbd:focus{outline:2px solid var(--search-local-input-active-border-color,var(--ifm-color-primary));outline-offset:0}div.ask-ai,html[data-theme=dark] div.ask-ai{--ask-ai-primary:var(--ifm-color-primary);--ask-ai-primary-hover:var(--ifm-color-primary-light);--ask-ai-foreground:var(--ifm-color-content);--ask-ai-border:var(--ifm-color-emphasis-300);--ask-ai-error:var(--ifm-color-danger);--ask-ai-button-bg:var(--ifm-color-emphasis-200)}.ask-ai{--ask-ai-background:var(--search-local-modal-background,#f5f6f7);--ask-ai-muted:var(--search-local-muted-color,#969faf)}html[data-theme=dark] .ask-ai{--ask-ai-background:var(--search-local-modal-background,var(--ifm-background-color));--ask-ai-muted:var(--search-local-muted-color,var(--ifm-color-secondary-darkest))}html[data-theme=dark] .searchBar_RVTs .dropdownMenu_qbY6{background:var(--search-local-modal-background,var(--ifm-background-color));box-shadow:var(--search-local-modal-shadow,inset 1px 1px 0 0 #2c2e40,0 3px 8px 0 #000309)}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2{align-items:center;background:var(--search-local-hit-background,#fff);border-radius:4px;box-shadow:var(--search-local-hit-shadow,0 1px 3px 0 #d4d9e1);color:var(--search-local-hit-color,#444950);cursor:pointer;display:flex;flex-direction:row;height:var(--search-local-hit-height,56px);padding:0 var(--search-local-spacing,12px);width:100%}html[data-theme=dark] .dropdownMenu_qbY6 .suggestion_fB_2{background:var(--search-local-hit-background,var(--ifm-color-emphasis-100));box-shadow:var(--search-local-hit-shadow,none);color:var(--search-local-hit-color,var(--ifm-font-color-base))}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2:not(:last-child){margin-bottom:4px}.searchBar_RVTs .dropdownMenu_qbY6 .suggestion_fB_2.cursor_eG29{background-color:var(--search-local-highlight-color,var(--ifm-color-primary))}.hitFooter_E9YW a,.hitIcon_a7Zy,.hitPath_ieM4,.hitTree_kk6K,.noResultsIcon_EBY5{color:var(--search-local-muted-color,#969faf)}html[data-theme=dark] .hitIcon_a7Zy,html[data-theme=dark] .hitPath_ieM4,html[data-theme=dark] .hitTree_kk6K,html[data-theme=dark] .noResultsIcon_EBY5{color:var(--search-local-muted-color,var(--ifm-color-secondary-darkest))}.hitTree_kk6K{align-items:center;display:flex}.hitTree_kk6K>svg{height:var(--search-local-hit-height,56px);opacity:.5;width:24px}.hitIcon_a7Zy,.hitTree_kk6K>svg{stroke-width:var(--search-local-icon-stroke-width,1.4)}.hitAction_NqkB,.hitIcon_a7Zy{height:20px;width:20px}.hitWrapper_sAK8{display:flex;flex:1 1 auto;flex-direction:column;font-weight:500;justify-content:center;margin:0 8px;overflow-x:hidden;width:80%}.hitWrapper_sAK8 mark{background:none;color:var(--search-local-highlight-color,var(--ifm-color-primary))}.hitTitle_vyVt{font-size:.9em}.hitPath_ieM4{font-size:.75em}.hitPath_ieM4,.hitTitle_vyVt{overflow-x:hidden;text-overflow:ellipsis;white-space:nowrap}.noResults_l6Q3{align-items:center;display:flex;flex-direction:column;justify-content:center;padding:var(--search-local-spacing,12px) 0}.noResultsIcon_EBY5{margin-bottom:var(--search-local-spacing,12px)}.hitFooter_E9YW{font-size:.85em;margin-top:var(--search-local-spacing,12px);text-align:center}.suggestion_fB_2.cursor_eG29,.suggestion_fB_2.cursor_eG29 .hitIcon_a7Zy,.suggestion_fB_2.cursor_eG29 .hitPath_ieM4,.suggestion_fB_2.cursor_eG29 .hitTree_kk6K,.suggestion_fB_2.cursor_eG29 mark{color:var(--search-local-hit-active-color,var(--ifm-color-white))!important}.searchBarContainer_NW3z{margin-left:16px}.searchBarContainer_NW3z .searchBarLoadingRing_YnHq{display:none;left:10px;position:absolute;top:6px}.searchBarContainer_NW3z .searchClearButton_qk4g{background:none;border:none;line-height:1rem;padding:0;position:absolute;right:.8rem;top:50%;transform:translateY(-50%)}.navbar__search{position:relative}.searchIndexLoading_EJ1f .navbar__search-input{background-image:none}.searchHintContainer_Pkmr{align-items:center;display:flex;gap:4px;height:100%;justify-content:center;pointer-events:none;position:absolute;right:10px;top:0}.searchHint_iIMx{background-color:var(--ifm-navbar-search-input-background-color);border:1px solid var(--ifm-color-emphasis-500);box-shadow:inset 0 -1px 0 var(--ifm-color-emphasis-500);color:var(--ifm-navbar-search-input-placeholder-color)}html[dir=rtl] .searchHintContainer_Pkmr{left:10px;right:auto}html[dir=rtl] .searchBarContainer_NW3z .searchClearButton_qk4g{left:.8rem;right:auto}html[dir=rtl] .searchBarContainer_NW3z .searchBarLoadingRing_YnHq{left:auto;right:10px}html[dir=rtl] .navbar__search-input{padding:0 2.25em 0 .5em}.loadingRing_RJI3{display:inline-block;height:20px;opacity:var(--search-local-loading-icon-opacity,.5);position:relative;width:20px}.loadingRing_RJI3 div{animation:1.2s cubic-bezier(.5,0,.5,1) infinite e;border:2px solid var(--search-load-loading-icon-color,var(--ifm-navbar-search-input-color));border-color:var(--search-load-loading-icon-color,var(--ifm-navbar-search-input-color)) #0000 #0000 #0000;border-radius:50%;display:block;height:16px;margin:2px;position:absolute;width:16px}.loadingRing_RJI3 div:first-child{animation-delay:-.45s}.loadingRing_RJI3 div:nth-child(2){animation-delay:-.3s}.loadingRing_RJI3 div:nth-child(3){animation-delay:-.15s}@keyframes e{0%{transform:rotate(0)}to{transform:rotate(1turn)}}.navbarHideable_m1mJ{transition:transform var(--ifm-transition-fast) ease}.navbarHidden_jGov{transform:translate3d(0,calc(-100% - 2px),0)}.errorBoundaryError_a6uf{color:red;white-space:pre-wrap}.badge_qP1g,.commitMeta_teNK,.commitMsg_XeKn,.name_mmMI{white-space:nowrap}.errorBoundaryFallback_VBag{color:red;padding:.55rem}.iconEdit_Z9Sw{margin-right:.3em;vertical-align:sub}.navbar__items--right>:last-child{padding-right:0}.lastUpdated_JAkA{font-size:smaller;font-style:italic;margin-top:.2rem}.tocCollapsibleButton_TO0P{align-items:center;display:flex;font-size:inherit;justify-content:space-between;padding:.4rem .8rem;width:100%}.tocCollapsibleButton_TO0P:after{background:var(--ifm-menu-link-sublist-icon) 50% 50%/2rem 2rem no-repeat;content:"";filter:var(--ifm-menu-link-sublist-icon-filter);height:1.25rem;transform:rotate(180deg);transition:transform var(--ifm-transition-fast);width:1.25rem}.tocCollapsibleButtonExpanded_MG3E:after,.tocCollapsibleExpanded_sAul{transform:none}.tocCollapsible_ETCw{background-color:var(--ifm-menu-color-background-active);border-radius:var(--ifm-global-radius);margin:1rem 0}.tocCollapsibleContent_vkbj>ul{border-left:none;border-top:1px solid var(--ifm-color-emphasis-300);font-size:15px;padding:.2rem 0}.tocCollapsibleContent_vkbj ul li{margin:.4rem .8rem}.footerLogoLink_BH7S{opacity:.5;transition:opacity var(--ifm-transition-fast) var(--ifm-transition-timing-default)}.footerLogoLink_BH7S:hover,.hash-link:focus,:hover>.hash-link,header.heroBanner_qw_p img.starButton_Vbnu:hover{opacity:1}.anchorTargetStickyNavbar_Vzrq{scroll-margin-top:calc(var(--ifm-navbar-height) + .5rem)}.anchorTargetHideOnScrollNavbar_vjPI{scroll-margin-top:.5rem}.hash-link{opacity:0;padding-left:.5rem;transition:opacity var(--ifm-transition-fast);-webkit-user-select:none;user-select:none}.hash-link:before{content:"#"}.mainWrapper_z2l0{display:flex;flex:1 0 auto;flex-direction:column}.docusaurus-mt-lg{margin-top:3rem}#__docusaurus{display:flex;flex-direction:column;min-height:100%}.searchContextInput_mXoe,.searchQueryInput_CFBF{background:var(--ifm-background-color);border:var(--ifm-global-border-width) solid var(--ifm-color-content-secondary);border-radius:var(--ifm-global-radius);color:var(--ifm-font-color-base);font-size:var(--ifm-font-size-base);margin-bottom:1rem;padding:.5rem;width:100%}.monthHeader_QFki,.updatesPage_z4Av,header.heroBanner_qw_p,main.docs{background:var(--hero-background)}.releaseCard_aa3v,.tagCard_SluW{border-radius:8px;box-shadow:var(--feature-img-shadow)}.searchResultItem_U687{border-bottom:1px solid #dfe3e8;padding:1rem 0}.searchResultItemPath_uIbk{color:var(--ifm-color-content-secondary);font-size:.8rem;margin:.5rem 0 0}.searchResultItemSummary_oZHr{font-style:italic;margin:.5rem 0 0}.updatesPage_z4Av{min-height:80vh;padding:2rem 1rem 4rem}.pageHeader_g4VR{margin-bottom:2.5rem;text-align:center}.pageHeader_g4VR h1{color:var(--text-color);font-family:Racing Sans One,cursive;font-size:2.5rem;margin-bottom:.25rem}.pageHeader_g4VR p{color:var(--mid-grey);font-family:Raleway,sans-serif;font-size:1.1rem;margin:0}.timeline_WTQQ{margin:0 auto;max-width:1000px;padding-left:2.5rem;position:relative;width:80%}.timeline_WTQQ:before{background:var(--mid-grey);bottom:0;content:"";left:0;opacity:.4;position:absolute;top:0;width:2px}.monthHeader_QFki{color:var(--mid-grey);font-size:1rem;margin:0;padding:.75rem 0 .5rem;position:sticky;top:60px;z-index:1}.entry_mTt2{margin-bottom:1rem;position:relative}.entry_mTt2:before{border-radius:50%;content:"";position:absolute;z-index:1}.releaseEntry_ZG2V:before{background:var(--accent-pink);height:14px;left:-2.5rem;margin-left:-6px;top:1rem;width:14px}.tagEntry_ORGt:before{background:var(--accent-blue);height:10px;left:-2.5rem;margin-left:-4px;top:1rem;width:10px}.commitEntry_JtMn:before{background:var(--mid-grey);height:6px;left:-2.5rem;margin-left:-2px;top:.9rem;width:6px}.commitRow_RIZe:hover,.releaseCard_aa3v,.tagCard_SluW{background:var(--background)}.releaseCard_aa3v{border-left:4px solid var(--accent-pink);padding:1.25rem 1.5rem}.releaseCard_aa3v .entryTitle_RUqs{align-items:center;color:var(--text-color);display:flex;flex-wrap:wrap;font-family:Roboto Mono,monospace;font-size:1.3rem;gap:.5rem;margin:0 0 .5rem}.tagCard_SluW{border-left:4px solid var(--accent-blue);padding:1rem 1.25rem}.badgeRelease_agcf,.button.color-btn.btn-pink{background:var(--accent-pink)}.tagCard_SluW .entryTitle_RUqs{font-size:1.1rem;gap:.5rem;margin:0 0 .25rem}.commitRow_RIZe,.tagCard_SluW .entryTitle_RUqs{align-items:center;color:var(--text-color);display:flex}.commitLink_QhHv,.meta_GplF{align-items:center;gap:.75rem}.commitRow_RIZe{border-bottom:1px solid #80808026;gap:.75rem;padding:.5rem 0;transition:background .15s}.commitLink_QhHv{display:flex;overflow:hidden}.commitLink_QhHv,.commitLink_QhHv:hover,.tile_CRAN,.tile_CRAN:hover{color:var(--text-color);-webkit-text-decoration:none;text-decoration:none}.fetchErrorCard_M8w3 a,.keepReading_G_Ux,.meta_GplF a,main.docs p.see-repo-note a{-webkit-text-decoration:underline;text-decoration:underline}.badge_qP1g{border-radius:4px;color:#fff;display:inline-block;font-size:.7rem;line-height:1.4;padding:.1rem .5rem}.badgeTag_gC_o{background:var(--accent-blue)}.tagPill_IKwa{border:1px solid var(--mid-grey);border-radius:4px;color:var(--mid-grey);font-size:.8rem;padding:.05rem .4rem}.avatarSmall_luV7,.avatar_Vzby,.contributorAvatar_eGpt img{border-radius:50%;vertical-align:middle}.errorMessage_lztz,.fetchErrorCard_M8w3{box-shadow:var(--feature-img-shadow);padding:1.25rem 1.5rem;text-align:center}.bodyPreview_iKaD{color:var(--mid-grey);display:-webkit-box;font-size:.9rem;-webkit-line-clamp:3;margin:.25rem 0 .5rem;-webkit-box-orient:vertical;line-height:1.5;overflow:hidden}.meta_GplF,.sha_JsPS{color:var(--mid-grey);font-size:.85rem}.meta_GplF{display:flex;flex-wrap:wrap}.meta_GplF a{color:var(--mid-grey)}.linksList_CKaa img,.meta_GplF a .providerIcon_NdOm{margin-right:.25rem;vertical-align:middle;width:1.35rem}.avatar_Vzby{height:24px;width:24px}.authorInfo_pi2M{align-items:center;display:flex;gap:.35rem}.sha_JsPS{flex-shrink:0}.commitMsg_XeKn{font-size:.95rem;max-width:500px;overflow:hidden}.authorLink_YWtN{color:var(--text-color);opacity:.75;transition:.2s}.authorLink_YWtN:hover .avatarSmall_luV7{transform:scale(1.08)}.avatarSmall_luV7{height:18px;margin-right:.2rem;transition:.2s;width:18px}.errorMessage_lztz,.fetchErrorCard_M8w3,.skeleton_NVtD{border-radius:8px;background:var(--background)}.commitMeta_teNK{align-items:center;color:var(--mid-grey);display:flex;flex-shrink:0;font-size:.8rem;gap:.3rem;margin-left:auto}.contributors_QyJV{align-items:center;display:flex;flex-wrap:wrap;gap:.35rem;margin-bottom:.5rem}.contributorAvatar_eGpt{display:block;flex-shrink:0;height:26px;width:26px}.contributorAvatar_eGpt img{height:26px;transition:transform .2s;width:26px}.contributorAvatar_eGpt:hover img{transform:scale(1.15)}.skeleton_NVtD{animation:1.5s ease-in-out infinite f;height:80px;margin-bottom:1rem}@keyframes f{0%,to{opacity:.4}50%{opacity:1}}.loadMore_YatW{display:flex;justify-content:center;margin:2rem auto}.fetchErrorCard_M8w3{color:var(--mid-grey);font-size:.95rem;margin:1.5rem auto;max-width:1000px;width:80%}.fetchErrorCard_M8w3 a{color:var(--accent-blue)}.errorMessage_lztz{color:var(--mid-grey);font-size:1.1rem;margin:0 auto;width:fit-content}.errorMessage_lztz p{margin:.25rem 0}html[data-theme=light]{--heading-shadow:-2px 3px 2px #a2a1a1;--sub-heading-shadow:-1px 1px 1px #a2a1a1}header.heroBanner_qw_p{min-height:calc(100vh - 4rem);overflow:hidden;padding:4rem 1rem 2rem;position:relative;text-align:center}header.heroBanner_qw_p h1.heroTitle_CteR{font-family:Racing Sans One,mono;font-size:8rem;text-shadow:var(--heading-shadow)}header.heroBanner_qw_p h3.heroSubTitle_p8ER{text-shadow:var(--sub-heading-shadow)}header.heroBanner_qw_p img.starButton_Vbnu{opacity:.85;position:absolute;right:1rem;top:1rem;width:12rem}.buttons_LJn2{align-items:center;display:flex;justify-content:center;margin:1.5rem}.dashyDescription_iLL2{font-size:1.2rem;margin:1rem auto;max-width:680px;text-align:left}.dashyDescription_iLL2>p,.dashyDescription_iLL2>span{margin-left:auto;margin-right:auto;max-width:680px}.keepReading_G_Ux{cursor:pointer;font-size:1.2rem;margin-left:.5rem}.scrollDown_cjGa{bottom:0;cursor:pointer;display:flex;left:42%;margin:2rem auto;position:absolute;width:fit-content}.scrollDownText_HIIX{color:var(--text-color);font-size:1.2rem;font-weight:700;margin:0 1rem}.scrollDownIcon_UZ2V{width:2rem}.scrollDownIcon_UZ2V path{fill:var(--text-color)}.sponsorContainer_zSLg{display:flex;flex-direction:column;gap:1rem;margin:1.5rem auto;max-width:680px}.sponsor_JDQ1{background:var(--background);color:var(--text-color);display:flex;flex-direction:column-reverse;gap:.5rem;justify-content:left;padding:.5rem;width:100%}.sponsor_JDQ1 span{font-size:.9rem;text-align:center}.sponsor_JDQ1 span a{color:var(--text-color);font-size:.9rem}.button{border:none;border-radius:4px;color:var(--black);cursor:pointer;display:inline-block;display:flex;font-size:1.3rem;font-weight:700;margin:.5rem;min-width:12rem;padding:.75rem 1rem;position:relative;transition:none}.button svg{margin-right:.5rem;width:1.8rem}.button.btn-small{font-size:1rem;max-width:12rem;min-width:auto;padding:.5rem .75rem;width:100%}.button.btn-small svg{margin-right:.35rem;width:1.2rem}.button.color-btn:hover{color:rgba(0,0,0,.702);top:2px}.button.color-btn:active{box-shadow:none!important;color:rgba(0,0,0,.502);top:4px}.button.color-btn.btn-pink{box-shadow:0 4px var(--accent-pink-dark)}.button.color-btn.btn-pink:hover{box-shadow:0 2px var(--accent-pink-dark)}.button.color-btn.btn-purple{background:var(--accent-purple);box-shadow:0 4px var(--accent-purple-dark)}.button.color-btn.btn-purple:hover{box-shadow:0 2px var(--accent-purple-dark)}.button.color-btn.btn-blue{background:var(--accent-blue);box-shadow:0 4px var(--accent-blue-dark)}.button.color-btn.btn-blue:hover{box-shadow:0 2px var(--accent-blue-dark)}.button.color-btn.btn-green{background:var(--accent-green);box-shadow:0 4px var(--accent-green-dark)}.button.color-btn.btn-green:hover{box-shadow:0 2px var(--accent-green-dark)}.button.color-btn.btn-yellow{background:var(--accent-yellow);box-shadow:0 4px var(--accent-yellow-dark)}.button.color-btn.btn-yellow:hover{box-shadow:0 2px var(--accent-yellow-dark)}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0,.home-page-features-wrapper .feature .feature-half.assets,.statItem_at6_,.tile_CRAN:hover,main.docs{box-shadow:var(--feature-img-shadow)}main.docs{margin:1rem auto;max-width:1250px;padding:1rem}main.docs h1.docs-title{font-size:5rem;margin:0;text-align:center}main.docs p.see-repo-note{background:rgba(255,255,170,.839);border-radius:3px;display:none;padding:.25rem 1rem;width:fit-content}main.docs p.see-repo-note a,main.docs p.see-repo-note b,main.docs p.see-repo-note span{color:#000;font-size:1rem;font-style:italic}main.docs p.see-repo-note a{color:#000;font-size:1rem;font-weight:700}main.docs div.row1{display:flex;flex-direction:row}main.docs div.docs-links-section{display:flex;flex-wrap:wrap}main.docs div.docs-contents{color:var(--text-color);display:flex;flex-direction:column;justify-content:space-around}main.docs div.docs-contents h3.section-title{margin:1rem .5rem .5rem}main.docs div.docs-contents a{display:flex;flex-direction:column;margin:.5rem .75rem;min-height:5rem;padding:.25rem .5rem;width:calc(33% - 1.5rem)}main.docs div.docs-contents a p.name{font-size:1.4rem;margin:0;text-align:left}main.docs div.docs-contents a span.description{font-size:1rem;text-align:left;white-space:pre-wrap}.color-pink{--feature-color:var(--accent-pink)}.color-blue{--feature-color:var(--accent-blue)}.color-green{--feature-color:var(--accent-green)}.color-yellow{--feature-color:var(--accent-yellow)}.color-white{--feature-color:#fff}html[data-theme=light] .color-pink{--feature-color:var(--accent-pink-dark)}html[data-theme=light] .color-blue{--feature-color:var(--accent-purple-dark)}html[data-theme=light] .color-green{--feature-color:var(--accent-green-dark)}html[data-theme=light] .color-yellow{--feature-color:var(--accent-yellow-dark)}html[data-theme=light] .color-white{--feature-color:#000}.home-page-features-wrapper{display:flex;flex-direction:column;font-size:1.2rem}.home-page-features-wrapper .feature{align-items:center;display:flex;justify-content:center;margin:0;padding:.5rem}.home-page-features-wrapper .feature.align-left{background:var(--background);box-shadow:0 1px 4px #000;flex-direction:row}.home-page-features-wrapper .feature.align-right{background:var(--hero-background);box-shadow:0 -1px 4px rgba(0,0,0,.702);flex-direction:row-reverse}.home-page-features-wrapper .feature .feature-half{margin:1rem;padding:.25rem;width:40%}.home-page-features-wrapper .feature .feature-half.assets{align-items:center;background:var(--feature-color);border-radius:8px;display:flex;height:fit-content;justify-content:center;margin:.5rem}.home-page-features-wrapper .feature .feature-half.assets img.demo,.home-page-features-wrapper .feature .feature-half.assets span.not-demo{background:var(--kinda-transparent);border-radius:8px;box-shadow:var(--feature-img-shadow);height:100%;max-height:36rem;min-height:12rem;object-fit:contain;width:100%}.home-page-features-wrapper .feature .feature-half.assets span.not-demo{align-items:center;display:flex;justify-content:center}.home-page-features-wrapper .feature .feature-title{align-items:flex-end;color:var(--feature-color);display:flex;margin:1rem 0}.home-page-features-wrapper .feature .feature-title svg{margin-right:.5rem;width:2rem}.home-page-features-wrapper .feature div.read-the-docs a.button-link-wrapper{margin:0 1rem 1rem}.home-page-features-wrapper .feature div.read-the-docs small{margin:0 .5rem;opacity:.75}.authorsSection_t3c7{background:var(--hero-background);padding:3rem 1rem}.inner_rVjr{margin:.5rem auto;width:80%}.heading_j3J0{color:var(--text-color);font-size:2rem}.enjoyingWrap_FXPn{display:flex;gap:2rem;margin-bottom:2rem}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0{background:var(--background);border-radius:8px;padding:.5rem 1rem;width:50%}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0 p{margin:0 0 .5rem}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0 .authorInnerWrap_y4gi{align-items:center;display:flex;gap:.75rem}.subHeading_D3Yw{color:var(--text-color);font-size:1.2rem;margin:.25rem 0}.builtBy_GoGF,.name_mmMI{color:var(--mid-grey);font-size:.9rem}.grid_qTnk{display:grid;gap:.75rem;grid-template-columns:repeat(auto-fill,minmax(120px,1fr));margin:0 auto 2rem;max-width:1800px;padding:0}.tile_CRAN{align-items:center;background:var(--background);border-radius:8px;display:flex;flex-direction:column;gap:.4rem;padding:.6rem;position:relative;transition:transform .2s,box-shadow .2s}.ctaSection_oQB2,.statItem_at6_{align-items:center;display:flex}.tile_CRAN:hover{transform:translateY(-2px)}.avatar__HA9{border-radius:50%;height:48px;width:48px}.name_mmMI{max-width:120px;overflow:hidden}.star_JoTu{position:absolute;right:.5rem;top:.5rem}.fallback_O_bA{margin:0 auto;max-width:100%}.fallback_O_bA img{height:auto;max-width:100%}.ctaSection_oQB2{flex-direction:row;gap:.5rem;justify-content:center;margin-top:.25rem}.builtBy_GoGF{margin-bottom:1rem}.statsSection_Rc3S{background:var(--hero-background);padding:.25rem 1rem 3rem}.inner_YtnN{display:flex;flex-wrap:wrap;gap:1.5rem;justify-content:space-around;margin:0 auto;max-width:1000px;width:80%}.statItem_at6_{background:var(--background);border-radius:8px;flex:1 1 180px;flex-direction:column;max-width:240px;padding:1.5rem 2rem;text-align:center}.icon_vhSc{height:3rem;margin-bottom:.25rem;width:3rem}.count_Ad5j,.icon_vhSc{color:var(--stat-color)}.count_Ad5j{font-size:2.8rem;font-weight:400;line-height:1.1}.label_wElN{color:var(--mid-grey);font-size:1rem;margin-top:.25rem}.buttonGroup_M5ko button,.codeBlockContainer_Ckt0{background:var(--prism-background-color);color:var(--prism-color)}iframe#dashy-survey{border:none;height:92vh;width:100%}.codeBlockContainer_Ckt0{border-radius:var(--ifm-code-border-radius);box-shadow:var(--ifm-global-shadow-lw);margin-bottom:var(--ifm-leading)}.codeBlock_bY9V{--ifm-pre-background:var(--prism-background-color);margin:0;padding:0}.codeBlockLines_e6Vv{float:left;font:inherit;min-width:100%;padding:var(--ifm-pre-padding)}.codeBlockLinesWithNumbering_o6Pm{display:table;padding:var(--ifm-pre-padding) 0}:where(:root){--docusaurus-highlighted-code-line-bg:#484d5b}:where([data-theme=dark]){--docusaurus-highlighted-code-line-bg:#646464}.theme-code-block-highlighted-line{background-color:var(--docusaurus-highlighted-code-line-bg);display:block;margin:0 calc(var(--ifm-pre-padding)*-1);padding:0 var(--ifm-pre-padding)}.codeLine_lJS_{counter-increment:line-count;display:table-row}.codeLineNumber_Tfdd{background:var(--ifm-pre-background);display:table-cell;left:0;overflow-wrap:normal;padding:0 var(--ifm-pre-padding);position:sticky;text-align:right;width:1%}.codeLineNumber_Tfdd:before{content:counter(line-count);opacity:.4}.codeLineContent_feaV{padding-right:var(--ifm-pre-padding)}.theme-code-block:hover .copyButtonCopied_Vdqa{opacity:1!important}.copyButtonIcons_IEyt{height:1.125rem;position:relative;width:1.125rem}.copyButtonIcon_TrPX,.copyButtonSuccessIcon_cVMy{fill:currentColor;height:inherit;left:0;opacity:inherit;position:absolute;top:0;transition:all var(--ifm-transition-fast) ease;width:inherit}.copyButtonSuccessIcon_cVMy{color:#00d600;left:50%;opacity:0;top:50%;transform:translate(-50%,-50%) scale(.33)}.copyButtonCopied_Vdqa .copyButtonIcon_TrPX{opacity:0;transform:scale(.33)}.copyButtonCopied_Vdqa .copyButtonSuccessIcon_cVMy{opacity:1;transform:translate(-50%,-50%) scale(1);transition-delay:75ms}.wordWrapButtonIcon_b1P5{height:1.2rem;width:1.2rem}.buttonGroup_M5ko{column-gap:.2rem;display:flex;position:absolute;right:calc(var(--ifm-pre-padding)/2);top:calc(var(--ifm-pre-padding)/2)}.buttonGroup_M5ko button{align-items:center;border:1px solid var(--ifm-color-emphasis-300);border-radius:var(--ifm-global-radius);display:flex;line-height:0;opacity:0;padding:.4rem;transition:opacity var(--ifm-transition-fast) ease-in-out}.buttonGroup_M5ko button:focus-visible,.buttonGroup_M5ko button:hover{opacity:1!important}.theme-code-block:hover .buttonGroup_M5ko button{opacity:.4}.codeBlockContent_QJqH{border-radius:inherit;direction:ltr;position:relative}.codeBlockTitle_OeMC{border-bottom:1px solid var(--ifm-color-emphasis-300);border-top-left-radius:inherit;border-top-right-radius:inherit;font-size:var(--ifm-code-font-size);font-weight:500;padding:.75rem var(--ifm-pre-padding)}.codeBlockTitle_OeMC+.codeBlockContent_QJqH .codeBlock_a8dz{border-top-left-radius:0;border-top-right-radius:0}.details_lb9f{--docusaurus-details-summary-arrow-size:0.38rem;--docusaurus-details-transition:transform 200ms ease;--docusaurus-details-decoration-color:grey}.details_lb9f>summary{cursor:pointer;padding-left:1rem;position:relative}.details_lb9f>summary::-webkit-details-marker{display:none}.details_lb9f>summary:before{border-color:#0000 #0000 #0000 var(--docusaurus-details-decoration-color);border-style:solid;border-width:var(--docusaurus-details-summary-arrow-size);content:"";left:0;position:absolute;top:.45rem;transform:rotate(0);transform-origin:calc(var(--docusaurus-details-summary-arrow-size)/2) 50%;transition:var(--docusaurus-details-transition)}.collapsibleContent_i85q{border-top:1px solid var(--docusaurus-details-decoration-color);margin-top:1rem;padding-top:1rem}.details_b_Ee{--docusaurus-details-decoration-color:var(--ifm-alert-border-color);--docusaurus-details-transition:transform var(--ifm-transition-fast) ease;border:1px solid var(--ifm-alert-border-color);margin:0 0 var(--ifm-spacing-vertical)}:not(.containsTaskList_mC6p>li)>.containsTaskList_mC6p{padding-left:0}.img_ev3q{height:auto}.admonition_xJq3{margin-bottom:1em}.admonitionHeading_Gvgb{font:var(--ifm-heading-font-weight) var(--ifm-h5-font-size)/var(--ifm-heading-line-height) var(--ifm-heading-font-family)}.admonitionHeading_Gvgb:not(:last-child){margin-bottom:.3rem}.admonitionHeading_Gvgb code{text-transform:none}.admonitionIcon_Rf37{display:inline-block;margin-right:.4em;vertical-align:middle}.admonitionIcon_Rf37 svg{display:inline-block;fill:var(--ifm-alert-foreground-color);height:1.6em;width:1.6em}.tableOfContents_bqdL{max-height:calc(100vh - var(--ifm-navbar-height) - 2rem);overflow-y:auto;position:sticky;top:calc(var(--ifm-navbar-height) + 1rem)}.breadcrumbHomeIcon_YNFT{height:1.1rem;position:relative;top:1px;vertical-align:top;width:1.1rem}.breadcrumbsContainer_Z_bl{--ifm-breadcrumb-size-multiplier:0.8;margin-bottom:.8rem}@media (min-width:997px){.collapseSidebarButton_PEFL,.expandButton_TmdG{background-color:var(--docusaurus-collapse-button-bg)}:root{--docusaurus-announcement-bar-height:30px}.announcementBarClose_gvF7,.announcementBarPlaceholder_vyr4{flex-basis:50px}.collapseSidebarButton_PEFL{border:1px solid var(--ifm-toc-border-color);border-radius:0;bottom:0;display:block!important;height:40px;position:sticky}.collapseSidebarButtonIcon_kv0_{margin-top:4px;transform:rotate(180deg)}.expandButtonIcon_i1dp,[dir=rtl] .collapseSidebarButtonIcon_kv0_{transform:rotate(0)}.collapseSidebarButton_PEFL:focus,.collapseSidebarButton_PEFL:hover,.expandButton_TmdG:focus,.expandButton_TmdG:hover{background-color:var(--docusaurus-collapse-button-bg-hover)}.menuHtmlItem_M9Kj{padding:var(--ifm-menu-link-padding-vertical) var(--ifm-menu-link-padding-horizontal)}.menu_SIkG{flex-grow:1;padding:.5rem}@supports (scrollbar-gutter:stable){.menu_SIkG{padding:.5rem 0 .5rem .5rem;scrollbar-gutter:stable}}.menuWithAnnouncementBar_GW3s{margin-bottom:var(--docusaurus-announcement-bar-height)}.sidebar_njMd{display:flex;flex-direction:column;height:100%;padding-top:var(--ifm-navbar-height);width:var(--doc-sidebar-width)}.sidebarWithHideableNavbar_wUlq{padding-top:0}.sidebarHidden_VK0M{opacity:0;visibility:hidden}.sidebarLogo_isFc{align-items:center;color:inherit!important;display:flex!important;margin:0 var(--ifm-navbar-padding-horizontal);max-height:var(--ifm-navbar-height);min-height:var(--ifm-navbar-height);-webkit-text-decoration:none!important;text-decoration:none!important}.sidebarLogo_isFc img{height:2rem;margin-right:.5rem}.expandButton_TmdG{align-items:center;display:flex;height:100%;justify-content:center;position:absolute;right:0;top:0;transition:background-color var(--ifm-transition-fast) ease;width:100%}[dir=rtl] .expandButtonIcon_i1dp{transform:rotate(180deg)}.docSidebarContainer_YfHR{border-right:1px solid var(--ifm-toc-border-color);clip-path:inset(0);display:block;margin-top:calc(var(--ifm-navbar-height)*-1);transition:width var(--ifm-transition-fast) ease;width:var(--doc-sidebar-width);will-change:width}.docSidebarContainerHidden_DPk8{cursor:pointer;width:var(--doc-sidebar-hidden-width)}.sidebarViewport_aRkj{height:100%;max-height:100vh;position:sticky;top:0}.docMainContainer_TBSr{flex-grow:1;max-width:calc(100% - var(--doc-sidebar-width))}.docMainContainerEnhanced_lQrH{max-width:calc(100% - var(--doc-sidebar-hidden-width))}.docItemWrapperEnhanced_JWYK{max-width:calc(var(--ifm-container-width) + var(--doc-sidebar-width))!important}.navbarSearchContainer_Bca1{padding:0 var(--ifm-navbar-item-padding-horizontal)}.lastUpdated_JAkA{text-align:right}.tocMobile_ITEo{display:none}.docItemCol_VOVn{max-width:75%!important}}@media (min-width:997px)and (max-width:1200px){nav.navbar .navbar__item.navbar__link{font-size:1rem}nav.navbar .navbar__item.navbar__link svg{display:none}}@media screen and (min-width:1200px){.sponsorContainer_zSLg{flex-direction:row;justify-content:center;max-width:1000px}.sponsor_JDQ1{flex:1 1 0;min-height:100%}}@media (min-width:1440px){.container{max-width:var(--ifm-container-width-xl)}}@media (max-width:996px){.col{--ifm-col-width:100%;flex-basis:var(--ifm-col-width);margin-left:0}.footer{--ifm-footer-padding-horizontal:0}.colorModeToggle_DEke,.footer__link-separator,.navbar__item,.tableOfContents_bqdL{display:none}.footer__col{margin-bottom:calc(var(--ifm-spacing-vertical)*3)}.footer__link-item{display:block;width:max-content}.hero{padding-left:0;padding-right:0}.navbar>.container,.navbar>.container-fluid{padding:0}.navbar__toggle{display:inherit}.navbar__search-input{width:9rem}.pills--block,.tabs--block{flex-direction:column}.navbarSearchContainer_Bca1{position:absolute;right:var(--ifm-navbar-padding-horizontal)}.docItemContainer_F8PC{padding:0 .3rem}}@media not (max-width:996px){.searchBar_RVTs.searchBarLeft_MXDe .dropdownMenu_qbY6{left:0!important;right:auto!important}}@media only screen and (max-width:996px){.searchQueryColumn_q7nx{max-width:60%!important}.searchContextColumn_oWAF{max-width:40%!important}}@media (max-width:966px){.timeline_WTQQ{padding-left:2rem;width:100%}.commitEntry_JtMn:before,.releaseEntry_ZG2V:before,.tagEntry_ORGt:before{left:-2rem}.commitRow_RIZe{flex-wrap:wrap}.commitMsg_XeKn{max-width:100%}.commitMeta_teNK{margin-left:0}.bodyPreview_iKaD{-webkit-line-clamp:2}header.heroBanner_qw_p h1.heroTitle_CteR{font-size:6rem}header.heroBanner_qw_p h3.heroSubTitle_p8ER{font-size:1.5rem}.scrollDown_cjGa{display:none}.button{width:80%}.ctaSection_oQB2,.enjoyingWrap_FXPn,.home-page-features-wrapper .feature.align-left,.home-page-features-wrapper .feature.align-right{flex-direction:column}.home-page-features-wrapper .feature.align-left .feature-half,.home-page-features-wrapper .feature.align-right .feature-half{width:95%}.enjoyingWrap_FXPn .enjoyingWrapPart_oGX0,.inner_rVjr{width:100%}.grid_qTnk{grid-template-columns:repeat(auto-fill,minmax(100px,1fr))}.inner_YtnN{display:grid;gap:1rem;grid-template-columns:1fr 1fr;width:100%}.statItem_at6_{max-width:none;padding:1.25rem 1rem}.count_Ad5j{font-size:2.2rem}}@media screen and (max-width:966px){header.heroBanner_qw_p{padding:2rem}.buttons_LJn2{flex-direction:column}}@media (max-width:768px){.glitchCode_Pt6F{font-size:6rem}}@media (max-width:600px){.banner_woPo{font-size:.85rem;padding:.4rem .75rem}.closeBtn_fC0A{font-size:1.1rem;right:.5rem}}@media (max-width:576px){.markdown h1:first-child{--ifm-h1-font-size:2rem}.markdown>h2{--ifm-h2-font-size:1.5rem}.markdown>h3{--ifm-h3-font-size:1.25rem}.navbar__search-input:not(:focus){width:2rem}.searchBar_RVTs .dropdownMenu_qbY6{max-width:calc(100vw - var(--ifm-navbar-padding-horizontal)*2);width:var(--search-local-modal-width-sm,340px)}.searchBarContainer_NW3z:not(.focused_OWtg) .searchClearButton_qk4g,.searchHintContainer_Pkmr{display:none}}@media screen and (max-width:576px){.searchQueryColumn_q7nx{max-width:100%!important}.searchContextColumn_oWAF{max-width:100%!important;padding-left:var(--ifm-spacing-horizontal)!important}}@media (max-width:500px){main.docs div.docs-contents a{width:100%}}@media (hover:hover){.backToTopButton_sjWU:hover{background-color:var(--ifm-color-emphasis-300)}}@media (pointer:fine){.thin-scrollbar{scrollbar-width:thin}.thin-scrollbar::-webkit-scrollbar{height:var(--ifm-scrollbar-size);width:var(--ifm-scrollbar-size)}.thin-scrollbar::-webkit-scrollbar-track{background:var(--ifm-scrollbar-track-background-color);border-radius:10px}.thin-scrollbar::-webkit-scrollbar-thumb{background:var(--ifm-scrollbar-thumb-background-color);border-radius:10px}.thin-scrollbar::-webkit-scrollbar-thumb:hover{background:var(--ifm-scrollbar-thumb-hover-background-color)}}@media (prefers-color-scheme:dark){#carbonads{--carbon-bg-primary:#1f1f1f;--carbon-bg-secondary:#262626;--carbon-text-color:#e6e6e6}}@media (prefers-reduced-motion:reduce){:root{--ifm-transition-fast:0ms;--ifm-transition-slow:0ms}.skeleton_NVtD{animation:none;opacity:.6}}@media print{.announcementBar_mb4j,.footer,.menu,.navbar,.noPrint_WFHX,.pagination-nav,.table-of-contents,.tocMobile_ITEo{display:none}.tabs{page-break-inside:avoid}.codeBlockLines_e6Vv{white-space:pre-wrap}} \ No newline at end of file diff --git a/assets/js/044d5aa6.e9cdb0c8.js b/assets/js/044d5aa6.ec5ac879.js similarity index 99% rename from assets/js/044d5aa6.e9cdb0c8.js rename to assets/js/044d5aa6.ec5ac879.js index 5b3a146d..2327e548 100644 --- a/assets/js/044d5aa6.e9cdb0c8.js +++ b/assets/js/044d5aa6.ec5ac879.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5332],{7670(e,n,s){s.r(n),s.d(n,{assets:()=>a,contentTitle:()=>r,default:()=>d,frontMatter:()=>t,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"icons","title":"Icons","description":"Both sections and items can have an icon, which is specified using the icon attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here.","source":"@site/docs/icons.md","sourceDirName":".","slug":"/icons","permalink":"/docs/icons","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/icons.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Troubleshooting","permalink":"/docs/troubleshooting"},"next":{"title":"Widgets","permalink":"/docs/widgets"}}');var o=s(4848),c=s(8453);const t={},r="Icons",a={},l=[{value:"Favicons",id:"favicons",level:2},{value:"Font Awesome",id:"font-awesome",level:2},{value:"Simple Icons",id:"simple-icons",level:2},{value:"Generative Icons",id:"generative-icons",level:2},{value:"Emoji Icons",id:"emoji-icons",level:2},{value:"selfh.st Icons",id:"selfhst-icons",level:2},{value:"Home-Lab Icons",id:"home-lab-icons",level:2},{value:"Material Design Icons",id:"material-design-icons",level:2},{value:"Icons by URL",id:"icons-by-url",level:2},{value:"Local Icons",id:"local-icons",level:2},{value:"Default Icon",id:"default-icon",level:2},{value:"No Icon",id:"no-icon",level:2},{value:"Icon Collections and Resources",id:"icon-collections-and-resources",level:2},{value:"Notes",id:"notes",level:2}];function h(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,c.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(n.header,{children:(0,o.jsx)(n.h1,{id:"icons",children:"Icons"})}),"\n",(0,o.jsxs)(n.p,{children:["Both sections and items can have an icon, which is specified using the ",(0,o.jsx)(n.code,{children:"icon"})," attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here."]}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#favicons",children:"Auto-Fetched Favicons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#font-awesome",children:"Font Awesome Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#simple-icons",children:"Simple Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#generative-icons",children:"Generative Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#emoji-icons",children:"Emoji Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#selfhst-icons",children:"selfh.st Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#home-lab-icons",children:"Home-Lab Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#material-design-icons",children:"Material Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#icons-by-url",children:"Icons by URL"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#default-icon",children:"Using a Default Icon"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#no-icon",children:"No Icon"})}),"\n",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/GTVmZnc/dashy-example-icons.png"}),"\n"]}),"\n"]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"favicons",children:"Favicons"}),"\n",(0,o.jsxs)(n.p,{children:["Dashy can auto-fetch an icon for a given service, using it's favicon. Just set ",(0,o.jsx)(n.code,{children:"icon: favicon"})," to use this feature."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/k6wyhnB/favicon-icons.png"}),"\n",(0,o.jsx)(n.p,{children:"Since different websites host their favicons at different paths, for the best results Dashy can use an API to resolve a websites icon."}),"\n",(0,o.jsxs)(n.p,{children:["The default favicon API is ",(0,o.jsx)(n.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"}),", but you can change this under ",(0,o.jsx)(n.code,{children:"appConfig.faviconApi"}),". If you'd prefer not to use an API, just set this value to ",(0,o.jsx)(n.code,{children:"local"}),". You can also use different APIs for individual items, by setting ",(0,o.jsx)(n.code,{children:"icon: favicon-[api]"}),", e.g. ",(0,o.jsx)(n.code,{children:"favicon-iconhorse"}),"."]}),"\n",(0,o.jsx)(n.p,{children:"The following favicon APIs are supported:"}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"allesedv"})," - ",(0,o.jsx)(n.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"})," is a highly efficient IPv6-enabled service"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"iconhorse"})," - ",(0,o.jsx)(n.a,{href:"https://icon.horse/",children:"Icon.Horse"})," returns quality icons for any site, with caching for speed and fallbacks for sites without an icon"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"faviconkit"})," - ",(0,o.jsx)(n.a,{href:"https://faviconkit.com/",children:"faviconkit.com"})," good quality icons and most sites supported (Note: down as of Nov '21)"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"besticon"})," - ",(0,o.jsx)(n.a,{href:"https://github.com/mat/besticon",children:"BestIcon"})," fetches websites icons from manifest"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"mcapi"})," - ",(0,o.jsx)(n.a,{href:"https://eu.mc-api.net/",children:"MC-API"})," fetches default website favicon, originally a Minecraft util"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"duckduckgo"})," - Returns decent quality website icons, from DuckDuckGo search"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"google"})," - Official Google favicon API service, good support for all sites, but poor quality"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"yandex"})," - Lower quality icons, but useful in some regions where other services are blocked"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"webmasterapi"})," - ",(0,o.jsx)(n.a,{href:"https://www.webmasterapi.com/",children:"WebMasterAPI"})," fetches website favicons via their API"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"local"})," - Set to local to fetch the default icon at /favicon.ico instead of using an API"]}),"\n"]}),"\n",(0,o.jsxs)(n.p,{children:["If for a given service none of the APIs work in your situation, and nor does local, then the best option is to find the path of the services logo or favicon, and set the icon to the URL of the raw image. For example, ",(0,o.jsx)(n.code,{children:"icon: https://monitoring.local/faviconx128.png"}),"- you can find this path using the browser dev tools."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"font-awesome",children:"Font Awesome"}),"\n",(0,o.jsxs)(n.p,{children:["You can use any ",(0,o.jsx)(n.a,{href:"https://fontawesome.com/icons",children:"Font Awesome Icon"})," simply by specifying it's identifier. This is in the format of ",(0,o.jsx)(n.code,{children:"[category] [name]"})," and can be found on the page for that icon on the Font Awesome site. For example: ",(0,o.jsx)(n.code,{children:"fas fa-rocket"}),", ",(0,o.jsx)(n.code,{children:"fab fa-monero"})," or ",(0,o.jsx)(n.code,{children:"fas fa-unicorn"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["Font-Awesome has a wide variety of free icons, but you can also use their pro icons if you have a membership. To do so, you need to specify your license key under: ",(0,o.jsx)(n.code,{children:"appConfig.fontAwesomeKey"}),". This is usually a 10-digit string, for example ",(0,o.jsx)(n.code,{children:"13014ae648"}),"."]}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/tMtwNYZ/fontawesome-icons3.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"simple-icons",children:"Simple Icons"}),"\n",(0,o.jsxs)(n.p,{children:[(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"SimpleIcons.org"})," is a collection of 2000+ high quality, free and open source brand and logo SVG icons. Usage of which is very similar to font-awesome icons. First find the glyph you want to use on the ",(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"website"}),", then just set your icon to the simple icon slug, prefixed with ",(0,o.jsx)(n.code,{children:"si-"}),"."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/MVhkXfC/simple-icons-example.png"}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Simple Icons Example\n items:\n - title: Portainer\n icon: si-portainer\n - title: FreeNAS\n icon: si-freenas\n - title: NextCloud\n icon: si-nextcloud\n - title: Home Assistant\n icon: si-homeassistant\n"})}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"generative-icons",children:"Generative Icons"}),"\n",(0,o.jsxs)(n.p,{children:["To uses a unique and programmatically generated icon for a given service just set ",(0,o.jsx)(n.code,{children:"icon: generative"}),". This is particularly useful when you have a lot of similar services with a different IP or port, and no specific icon. These icons are generated with ",(0,o.jsx)(n.a,{href:"https://api.dicebear.com/",children:"DiceBear"})," (or ",(0,o.jsx)(n.a,{href:"https://evatar.io/",children:"Evatar"})," for fallback), and use a hash of the services domain/ ip for entropy, so each domain will have a unique icon."]}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/b2pC2CL/generative-icons-2.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"emoji-icons",children:"Emoji Icons"}),"\n",(0,o.jsxs)(n.p,{children:["You can use almost any emoji as an icon for items or sections. You can specify the emoji either by pasting it directly, using it's unicode ( e.g. ",(0,o.jsx)(n.code,{children:"'U+1F680'"}),") or shortcode (e.g. ",(0,o.jsx)(n.code,{children:"':rocket:'"}),"). You can find these codes for any emoji using ",(0,o.jsx)(n.a,{href:"https://emojipedia.org/",children:"Emojipedia"})," (near the bottom of emoji each page), or for a quick reference to emoji shortcodes, check out ",(0,o.jsx)(n.a,{href:"https://emojis.ninja/",children:"emojis.ninja"})," by @nomanoff."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/YLwgTf9/emoji-icons-1.png"}),"\n",(0,o.jsxs)(n.p,{children:["For example, these will all render the same rocket (\ud83d\ude80) emoji: ",(0,o.jsx)(n.code,{children:"icon: ':rocket:'"})," or ",(0,o.jsx)(n.code,{children:"icon: 'U+1F680'"})," or ",(0,o.jsx)(n.code,{children:"icon: \ud83d\ude80"})]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"selfhst-icons",children:"selfh.st Icons"}),"\n",(0,o.jsxs)(n.p,{children:["The ",(0,o.jsx)(n.a,{href:"https://selfh.st/",children:"selfh.st"})," project provides a set of icons, originally for self-hosted services, but now expanded to include a wide variety of services. These icons can be used by specifying the icon name (without extension and with all spaces replaced with -) preceded by ",(0,o.jsx)(n.code,{children:"sh-"}),". See ",(0,o.jsx)(n.a,{href:"https://selfh.st/icons/",children:"https://selfh.st/icons/"})," for a full list of all available icons. For example, the Home Assistant icon is ",(0,o.jsx)(n.code,{children:"sh-home-assistant"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["Note: These icons are fetched from the jsdelivr CDN, so if you require offline access, the ",(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})," method may be a better option for you."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/pfy09LH/Screenshot-from-2025-01-08-22-04-21.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"home-lab-icons",children:"Home-Lab Icons"}),"\n",(0,o.jsxs)(n.p,{children:["The ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs/dashboard-icons",children:"dashboard-icons"})," repo by ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs",children:"Homarr Labs"})," provides a comprehensive collection of high-quality icons for commonly self-hosted services. Dashy natively supports these icons, and you can use them just by specifying the icon name (without extension) preceded by ",(0,o.jsx)(n.code,{children:"hl-"}),". SVG is preferred where available, with an automatic fallback to PNG for icons that aren't published as SVG. See ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs/dashboard-icons/tree/main/svg",children:"here"})," for a full list of all available icons. Note that these are fetched and cached straight from jsDelivr, so if you require offline access, the ",(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})," method may be a better option for you."]}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Home Lab Icons Example\n items:\n - title: AdGuard Home\n icon: hl-adguardhome\n - title: Long Horn\n icon: hl-longhorn\n - title: Nagios\n icon: hl-nagios\n - title: Whoogle Search\n icon: hl-whooglesearch\n"})}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/PQzYHmD/homelab-icons-2.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"material-design-icons",children:"Material Design Icons"}),"\n",(0,o.jsxs)(n.p,{children:["Dashy also supports 5000+ ",(0,o.jsx)(n.a,{href:"https://github.com/Templarian/MaterialDesign",children:"material-design-icons"}),". To use these, first find the name/ slug for your icon ",(0,o.jsx)(n.a,{href:"https://dev.materialdesignicons.com/icons",children:"here"}),", and then prefix is with ",(0,o.jsx)(n.code,{children:"mdi-"}),"."]}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Material Design Icons Example\n items:\n - title: Alien Icon\n icon: mdi-alien\n - title: Fire Icon\n icon: mdi-fire\n - title: Dino Icon\n icon: mdi-google-downasaur\n"})}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/fC9B4mq/icons-mdi-example.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"icons-by-url",children:"Icons by URL"}),"\n",(0,o.jsxs)(n.p,{children:["You can also set an icon by passing in a valid URL pointing to the icons location. For example ",(0,o.jsx)(n.code,{children:"icon: https://i.ibb.co/710B3Yc/space-invader-x256.png"}),", this can be in .png, .jpg or .svg format, and hosted anywhere (local or remote) - so long as it's accessible from where you are hosting Dashy. The icon will be automatically scaled to fit, however loading in a lot of large icons may have a negative impact on performance, especially if you visit Dashy from new devices often."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"local-icons",children:"Local Icons"}),"\n",(0,o.jsxs)(n.p,{children:["You may also want to store your icons locally, bundled within Dashy so that there is no reliance on outside services. This can be done by putting the icons within Dashy's ",(0,o.jsx)(n.code,{children:"./user-data/item-icons/"})," directory. If you are using Docker, then the easiest option is to map a volume from your host system, for example: ",(0,o.jsx)(n.code,{children:"-v /local/image/directory:/app/user-data/item-icons/"}),". To reference an icon stored locally, just specify it's name and extension. For example, if my icon was stored in ",(0,o.jsx)(n.code,{children:"/app/user-data/item-icons/maltrail.png"}),", then I would just set ",(0,o.jsx)(n.code,{children:"icon: maltrail.png"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["You can also use sub-folders within the ",(0,o.jsx)(n.code,{children:"item-icons"})," directory to keep things organized. You would then specify an icon with it's folder name slash image name. For example: ",(0,o.jsx)(n.code,{children:"networking/monit.png"})]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"default-icon",children:"Default Icon"}),"\n",(0,o.jsxs)(n.p,{children:["If you'd like to set a default icon, to be applied to any items which don't have an icon already set, then this can be done under ",(0,o.jsx)(n.code,{children:"appConfig.defaultIcon"}),"."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"no-icon",children:"No Icon"}),"\n",(0,o.jsxs)(n.p,{children:["If you don't wish for a given item or section to have an icon, just leave out the ",(0,o.jsx)(n.code,{children:"icon"})," attribute."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"icon-collections-and-resources",children:"Icon Collections and Resources"}),"\n",(0,o.jsxs)(n.p,{children:["The following websites provide good-quality, free icon sets. To use any of these icons, either copy the link to the raw icon (it should end in ",(0,o.jsx)(n.code,{children:".svg"})," or ",(0,o.jsx)(n.code,{children:".png"}),") and paste it as your ",(0,o.jsx)(n.code,{children:"icon"}),", or download and save the icons in ",(0,o.jsx)(n.code,{children:"/user-data/item-icons"})," / mapped Docker volume. Full credit to the authors, please see the licenses for each service for usage and copyright information."]}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://thehomelab.wiki/books/helpful-tools-resources/page/icons-for-self-hosted-dashboards",children:"Icons for Self-Hosted Apps"})," - 350+ high-quality icons for commonly self-hosted services"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://svgbox.net/iconsets/",children:"SVG Box"})," - Cryptocurrency, social media apps and flag icons"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"Simple Icons"})," - Free SVG brand icons, with easy API access"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://github.com/google/material-design-icons/",children:"Material Design Icons"})," - Hundreds of Open source PNG + SVG icons by Google"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://icons8.com/icons",children:"Icons8"})," - Thousands of icons, all with free versions at 64x64"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://www.flaticon.com/",children:"Flat Icon"})," - Wide variety of icon sets, most of which are free to use"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://www.svgrepo.com/",children:"SVG Repo"})," - 300,000+ Vector Icons"]}),"\n"]}),"\n",(0,o.jsxs)(n.p,{children:["If you are a student, then you can get free access to premium icons on ",(0,o.jsx)(n.a,{href:"https://education.github.com/pack/redeem/iconscout-student",children:"Icon Scout"})," or ",(0,o.jsx)(n.a,{href:"https://icons8.com/github-students",children:"Icons8"})," using the ",(0,o.jsx)(n.a,{href:"https://education.github.com/pack",children:"GitHub Student Pack"}),"."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"notes",children:"Notes"}),"\n",(0,o.jsx)(n.p,{children:"If you are using icons from an external source, these will be fetched on initial page load automatically, if and when needed. But combining icons from multiple services may have a negative impact on performance."}),"\n",(0,o.jsx)(n.p,{children:"You can improve load speeds, by downloading your required icons, and serving them locally. Scaling icons to the minimum required dimensions (e.g. 128x128 or 64x64) will also greatly improve application load times."}),"\n",(0,o.jsx)(n.p,{children:"For icons from external sources, please see the Privacy Policies and Licenses for that provider."})]})}function d(e={}){const{wrapper:n}={...(0,c.R)(),...e.components};return n?(0,o.jsx)(n,{...e,children:(0,o.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>r});var i=s(6540);const o={},c=i.createContext(o);function t(e){const n=i.useContext(c);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:t(e.components),i.createElement(c.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5332],{7670(e,n,s){s.r(n),s.d(n,{assets:()=>a,contentTitle:()=>r,default:()=>d,frontMatter:()=>t,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"icons","title":"Icons","description":"Both sections and items can have an icon, which is specified using the icon attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here.","source":"@site/docs/icons.md","sourceDirName":".","slug":"/icons","permalink":"/docs/icons","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/icons.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Troubleshooting","permalink":"/docs/troubleshooting"},"next":{"title":"Widgets","permalink":"/docs/widgets"}}');var o=s(4848),c=s(8453);const t={},r="Icons",a={},l=[{value:"Favicons",id:"favicons",level:2},{value:"Font Awesome",id:"font-awesome",level:2},{value:"Simple Icons",id:"simple-icons",level:2},{value:"Generative Icons",id:"generative-icons",level:2},{value:"Emoji Icons",id:"emoji-icons",level:2},{value:"selfh.st Icons",id:"selfhst-icons",level:2},{value:"Home-Lab Icons",id:"home-lab-icons",level:2},{value:"Material Design Icons",id:"material-design-icons",level:2},{value:"Icons by URL",id:"icons-by-url",level:2},{value:"Local Icons",id:"local-icons",level:2},{value:"Default Icon",id:"default-icon",level:2},{value:"No Icon",id:"no-icon",level:2},{value:"Icon Collections and Resources",id:"icon-collections-and-resources",level:2},{value:"Notes",id:"notes",level:2}];function h(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,c.R)(),...e.components};return(0,o.jsxs)(o.Fragment,{children:[(0,o.jsx)(n.header,{children:(0,o.jsx)(n.h1,{id:"icons",children:"Icons"})}),"\n",(0,o.jsxs)(n.p,{children:["Both sections and items can have an icon, which is specified using the ",(0,o.jsx)(n.code,{children:"icon"})," attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here."]}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#favicons",children:"Auto-Fetched Favicons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#font-awesome",children:"Font Awesome Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#simple-icons",children:"Simple Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#generative-icons",children:"Generative Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#emoji-icons",children:"Emoji Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#selfhst-icons",children:"selfh.st Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#home-lab-icons",children:"Home-Lab Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#material-design-icons",children:"Material Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#icons-by-url",children:"Icons by URL"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#default-icon",children:"Using a Default Icon"})}),"\n"]}),"\n",(0,o.jsxs)(n.li,{children:["\n",(0,o.jsx)(n.p,{children:(0,o.jsx)(n.a,{href:"#no-icon",children:"No Icon"})}),"\n",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/GTVmZnc/dashy-example-icons.png"}),"\n"]}),"\n"]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"favicons",children:"Favicons"}),"\n",(0,o.jsxs)(n.p,{children:["Dashy can auto-fetch an icon for a given service, using it's favicon. Just set ",(0,o.jsx)(n.code,{children:"icon: favicon"})," to use this feature."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/k6wyhnB/favicon-icons.png"}),"\n",(0,o.jsx)(n.p,{children:"Since different websites host their favicons at different paths, for the best results Dashy can use an API to resolve a websites icon."}),"\n",(0,o.jsxs)(n.p,{children:["The default favicon API is ",(0,o.jsx)(n.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"}),", but you can change this under ",(0,o.jsx)(n.code,{children:"appConfig.faviconApi"}),". If you'd prefer not to use an API, just set this value to ",(0,o.jsx)(n.code,{children:"local"}),". You can also use different APIs for individual items, by setting ",(0,o.jsx)(n.code,{children:"icon: favicon-[api]"}),", e.g. ",(0,o.jsx)(n.code,{children:"favicon-iconhorse"}),"."]}),"\n",(0,o.jsx)(n.p,{children:"The following favicon APIs are supported:"}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"allesedv"})," - ",(0,o.jsx)(n.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"})," is a highly efficient IPv6-enabled service"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"iconhorse"})," - ",(0,o.jsx)(n.a,{href:"https://icon.horse/",children:"Icon.Horse"})," returns quality icons for any site, with caching for speed and fallbacks for sites without an icon"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"faviconkit"})," - ",(0,o.jsx)(n.a,{href:"https://faviconkit.com/",children:"faviconkit.com"})," good quality icons and most sites supported (Note: down as of Nov '21)"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"besticon"})," - ",(0,o.jsx)(n.a,{href:"https://github.com/mat/besticon",children:"BestIcon"})," fetches websites icons from manifest"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"mcapi"})," - ",(0,o.jsx)(n.a,{href:"https://eu.mc-api.net/",children:"MC-API"})," fetches default website favicon, originally a Minecraft util"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"duckduckgo"})," - Returns decent quality website icons, from DuckDuckGo search"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"google"})," - Official Google favicon API service, good support for all sites, but poor quality"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"yandex"})," - Lower quality icons, but useful in some regions where other services are blocked"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"webmasterapi"})," - ",(0,o.jsx)(n.a,{href:"https://www.webmasterapi.com/",children:"WebMasterAPI"})," fetches website favicons via their API"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.code,{children:"local"})," - Set to local to fetch the default icon at /favicon.ico instead of using an API"]}),"\n"]}),"\n",(0,o.jsxs)(n.p,{children:["If for a given service none of the APIs work in your situation, and nor does local, then the best option is to find the path of the services logo or favicon, and set the icon to the URL of the raw image. For example, ",(0,o.jsx)(n.code,{children:"icon: https://monitoring.local/faviconx128.png"}),"- you can find this path using the browser dev tools."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"font-awesome",children:"Font Awesome"}),"\n",(0,o.jsxs)(n.p,{children:["You can use any ",(0,o.jsx)(n.a,{href:"https://fontawesome.com/icons",children:"Font Awesome Icon"})," simply by specifying it's identifier. This is in the format of ",(0,o.jsx)(n.code,{children:"[category] [name]"})," and can be found on the page for that icon on the Font Awesome site. For example: ",(0,o.jsx)(n.code,{children:"fas fa-rocket"}),", ",(0,o.jsx)(n.code,{children:"fab fa-monero"})," or ",(0,o.jsx)(n.code,{children:"fas fa-unicorn"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["Font-Awesome has a wide variety of free icons, but you can also use their pro icons if you have a membership. To do so, you need to specify your license key under: ",(0,o.jsx)(n.code,{children:"appConfig.fontAwesomeKey"}),". This is usually a 10-digit string, for example ",(0,o.jsx)(n.code,{children:"13014ae648"}),"."]}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/tMtwNYZ/fontawesome-icons3.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"simple-icons",children:"Simple Icons"}),"\n",(0,o.jsxs)(n.p,{children:[(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"SimpleIcons.org"})," is a collection of 2000+ high quality, free and open source brand and logo SVG icons. Usage of which is very similar to font-awesome icons. First find the glyph you want to use on the ",(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"website"}),", then just set your icon to the simple icon slug, prefixed with ",(0,o.jsx)(n.code,{children:"si-"}),"."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/MVhkXfC/simple-icons-example.png"}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Simple Icons Example\n items:\n - title: Portainer\n icon: si-portainer\n - title: FreeNAS\n icon: si-freenas\n - title: NextCloud\n icon: si-nextcloud\n - title: Home Assistant\n icon: si-homeassistant\n"})}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"generative-icons",children:"Generative Icons"}),"\n",(0,o.jsxs)(n.p,{children:["To uses a unique and programmatically generated icon for a given service just set ",(0,o.jsx)(n.code,{children:"icon: generative"}),". This is particularly useful when you have a lot of similar services with a different IP or port, and no specific icon. These icons are generated with ",(0,o.jsx)(n.a,{href:"https://api.dicebear.com/",children:"DiceBear"})," (or ",(0,o.jsx)(n.a,{href:"https://evatar.io/",children:"Evatar"})," for fallback), and use a hash of the services domain/ ip for entropy, so each domain will have a unique icon."]}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/b2pC2CL/generative-icons-2.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"emoji-icons",children:"Emoji Icons"}),"\n",(0,o.jsxs)(n.p,{children:["You can use almost any emoji as an icon for items or sections. You can specify the emoji either by pasting it directly, using it's unicode ( e.g. ",(0,o.jsx)(n.code,{children:"'U+1F680'"}),") or shortcode (e.g. ",(0,o.jsx)(n.code,{children:"':rocket:'"}),"). You can find these codes for any emoji using ",(0,o.jsx)(n.a,{href:"https://emojipedia.org/",children:"Emojipedia"})," (near the bottom of emoji each page), or for a quick reference to emoji shortcodes, check out ",(0,o.jsx)(n.a,{href:"https://emojis.ninja/",children:"emojis.ninja"})," by @nomanoff."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/YLwgTf9/emoji-icons-1.png"}),"\n",(0,o.jsxs)(n.p,{children:["For example, these will all render the same rocket (\ud83d\ude80) emoji: ",(0,o.jsx)(n.code,{children:"icon: ':rocket:'"})," or ",(0,o.jsx)(n.code,{children:"icon: 'U+1F680'"})," or ",(0,o.jsx)(n.code,{children:"icon: \ud83d\ude80"})]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"selfhst-icons",children:"selfh.st Icons"}),"\n",(0,o.jsxs)(n.p,{children:["The ",(0,o.jsx)(n.a,{href:"https://selfh.st/",children:"selfh.st"})," project provides a set of icons, originally for self-hosted services, but now expanded to include a wide variety of services. These icons can be used by specifying the icon name (without extension and with all spaces replaced with -) preceded by ",(0,o.jsx)(n.code,{children:"sh-"}),". See ",(0,o.jsx)(n.a,{href:"https://selfh.st/icons/",children:"https://selfh.st/icons/"})," for a full list of all available icons. For example, the Home Assistant icon is ",(0,o.jsx)(n.code,{children:"sh-home-assistant"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["Note: These icons are fetched from the jsdelivr CDN, so if you require offline access, the ",(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})," method may be a better option for you."]}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/pfy09LH/Screenshot-from-2025-01-08-22-04-21.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"home-lab-icons",children:"Home-Lab Icons"}),"\n",(0,o.jsxs)(n.p,{children:["The ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs/dashboard-icons",children:"dashboard-icons"})," repo by ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs",children:"Homarr Labs"})," provides a comprehensive collection of high-quality icons for commonly self-hosted services. Dashy natively supports these icons, and you can use them just by specifying the icon name (without extension) preceded by ",(0,o.jsx)(n.code,{children:"hl-"}),". SVG is preferred where available, with an automatic fallback to PNG for icons that aren't published as SVG. See ",(0,o.jsx)(n.a,{href:"https://github.com/homarr-labs/dashboard-icons/tree/main/svg",children:"here"})," for a full list of all available icons. Note that these are fetched and cached straight from jsDelivr, so if you require offline access, the ",(0,o.jsx)(n.a,{href:"#local-icons",children:"Local Icons"})," method may be a better option for you."]}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Home Lab Icons Example\n items:\n - title: AdGuard Home\n icon: hl-adguardhome\n - title: Long Horn\n icon: hl-longhorn\n - title: Nagios\n icon: hl-nagios\n - title: Whoogle Search\n icon: hl-whooglesearch\n"})}),"\n ",(0,o.jsx)(n.img,{width:"580",src:"https://i.ibb.co/PQzYHmD/homelab-icons-2.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"material-design-icons",children:"Material Design Icons"}),"\n",(0,o.jsxs)(n.p,{children:["Dashy also supports 5000+ ",(0,o.jsx)(n.a,{href:"https://github.com/Templarian/MaterialDesign",children:"material-design-icons"}),". To use these, first find the name/ slug for your icon ",(0,o.jsx)(n.a,{href:"https://dev.materialdesignicons.com/icons",children:"here"}),", and then prefix is with ",(0,o.jsx)(n.code,{children:"mdi-"}),"."]}),"\n",(0,o.jsx)(n.p,{children:"For example:"}),"\n",(0,o.jsx)(n.pre,{children:(0,o.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Material Design Icons Example\n items:\n - title: Alien Icon\n icon: mdi-alien\n - title: Fire Icon\n icon: mdi-fire\n - title: Dino Icon\n icon: mdi-google-downasaur\n"})}),"\n ",(0,o.jsx)(n.img,{width:"500",src:"https://i.ibb.co/fC9B4mq/icons-mdi-example.png"}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"icons-by-url",children:"Icons by URL"}),"\n",(0,o.jsxs)(n.p,{children:["You can also set an icon by passing in a valid URL pointing to the icons location. For example ",(0,o.jsx)(n.code,{children:"icon: https://i.ibb.co/710B3Yc/space-invader-x256.png"}),", this can be in .png, .jpg or .svg format, and hosted anywhere (local or remote) - so long as it's accessible from where you are hosting Dashy. The icon will be automatically scaled to fit, however loading in a lot of large icons may have a negative impact on performance, especially if you visit Dashy from new devices often."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"local-icons",children:"Local Icons"}),"\n",(0,o.jsxs)(n.p,{children:["You may also want to store your icons locally, bundled within Dashy so that there is no reliance on outside services. This can be done by putting the icons within Dashy's ",(0,o.jsx)(n.code,{children:"./user-data/item-icons/"})," directory. If you are using Docker, then the easiest option is to map a volume from your host system, for example: ",(0,o.jsx)(n.code,{children:"-v /local/image/directory:/app/user-data/item-icons/"}),". To reference an icon stored locally, just specify it's name and extension. For example, if my icon was stored in ",(0,o.jsx)(n.code,{children:"/app/user-data/item-icons/maltrail.png"}),", then I would just set ",(0,o.jsx)(n.code,{children:"icon: maltrail.png"}),"."]}),"\n",(0,o.jsxs)(n.p,{children:["You can also use sub-folders within the ",(0,o.jsx)(n.code,{children:"item-icons"})," directory to keep things organized. You would then specify an icon with it's folder name slash image name. For example: ",(0,o.jsx)(n.code,{children:"networking/monit.png"})]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"default-icon",children:"Default Icon"}),"\n",(0,o.jsxs)(n.p,{children:["If you'd like to set a default icon, to be applied to any items which don't have an icon already set, then this can be done under ",(0,o.jsx)(n.code,{children:"appConfig.defaultIcon"}),"."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"no-icon",children:"No Icon"}),"\n",(0,o.jsxs)(n.p,{children:["If you don't wish for a given item or section to have an icon, just leave out the ",(0,o.jsx)(n.code,{children:"icon"})," attribute."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"icon-collections-and-resources",children:"Icon Collections and Resources"}),"\n",(0,o.jsxs)(n.p,{children:["The following websites provide good-quality, free icon sets. To use any of these icons, either copy the link to the raw icon (it should end in ",(0,o.jsx)(n.code,{children:".svg"})," or ",(0,o.jsx)(n.code,{children:".png"}),") and paste it as your ",(0,o.jsx)(n.code,{children:"icon"}),", or download and save the icons in ",(0,o.jsx)(n.code,{children:"/user-data/item-icons"})," / mapped Docker volume. Full credit to the authors, please see the licenses for each service for usage and copyright information."]}),"\n",(0,o.jsxs)(n.ul,{children:["\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://thehomelab.wiki/books/helpful-tools-resources/page/icons-for-self-hosted-dashboards",children:"Icons for Self-Hosted Apps"})," - 350+ high-quality icons for commonly self-hosted services"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://svgbox.net/iconsets/",children:"SVG Box"})," - Cryptocurrency, social media apps and flag icons"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://simpleicons.org/",children:"Simple Icons"})," - Free SVG brand icons, with easy API access"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://github.com/google/material-design-icons/",children:"Material Design Icons"})," - Hundreds of Open source PNG + SVG icons by Google"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://icons8.com/icons",children:"Icons8"})," - Thousands of icons, all with free versions at 64x64"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://www.flaticon.com/",children:"Flat Icon"})," - Wide variety of icon sets, most of which are free to use"]}),"\n",(0,o.jsxs)(n.li,{children:[(0,o.jsx)(n.a,{href:"https://www.svgrepo.com/",children:"SVG Repo"})," - 300,000+ Vector Icons"]}),"\n"]}),"\n",(0,o.jsxs)(n.p,{children:["If you are a student, then you can get free access to premium icons on ",(0,o.jsx)(n.a,{href:"https://education.github.com/pack/redeem/iconscout-student",children:"Icon Scout"})," or ",(0,o.jsx)(n.a,{href:"https://icons8.com/github-students",children:"Icons8"})," using the ",(0,o.jsx)(n.a,{href:"https://education.github.com/pack",children:"GitHub Student Pack"}),"."]}),"\n",(0,o.jsx)(n.hr,{}),"\n",(0,o.jsx)(n.h2,{id:"notes",children:"Notes"}),"\n",(0,o.jsx)(n.p,{children:"If you are using icons from an external source, these will be fetched on initial page load automatically, if and when needed. But combining icons from multiple services may have a negative impact on performance."}),"\n",(0,o.jsx)(n.p,{children:"You can improve load speeds, by downloading your required icons, and serving them locally. Scaling icons to the minimum required dimensions (e.g. 128x128 or 64x64) will also greatly improve application load times."}),"\n",(0,o.jsx)(n.p,{children:"For icons from external sources, please see the Privacy Policies and Licenses for that provider."})]})}function d(e={}){const{wrapper:n}={...(0,c.R)(),...e.components};return n?(0,o.jsx)(n,{...e,children:(0,o.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>r});var i=s(6540);const o={},c=i.createContext(o);function t(e){const n=i.useContext(c);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(o):e.components||o:t(e.components),i.createElement(c.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/0bfe2fe2.f922c786.js b/assets/js/0bfe2fe2.23bf9fe6.js similarity index 99% rename from assets/js/0bfe2fe2.f922c786.js rename to assets/js/0bfe2fe2.23bf9fe6.js index e5f72075..fb39db34 100644 --- a/assets/js/0bfe2fe2.f922c786.js +++ b/assets/js/0bfe2fe2.23bf9fe6.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[7151],{5230(e,n,s){s.r(n),s.d(n,{assets:()=>h,contentTitle:()=>c,default:()=>d,frontMatter:()=>a,metadata:()=>i,toc:()=>o});const i=JSON.parse('{"id":"searching","title":"Keyboard Shortcuts","description":"Searching","source":"@site/docs/searching.md","sourceDirName":".","slug":"/searching","permalink":"/docs/searching","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/searching.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Authentication","permalink":"/docs/authentication"},"next":{"title":"Alternate Views & Opening Methods","permalink":"/docs/alternate-views"}}');var r=s(4848),t=s(8453);const a={},c="Keyboard Shortcuts",h={},o=[{value:"Searching",id:"searching",level:2},{value:"Navigating",id:"navigating",level:2},{value:"Launching Apps",id:"launching-apps",level:2},{value:"Tags",id:"tags",level:2},{value:"Custom Hotkeys",id:"custom-hotkeys",level:2},{value:"Web Search",id:"web-search",level:2},{value:"Setting Search Engine",id:"setting-search-engine",level:3},{value:"Using Custom Search Engine",id:"using-custom-search-engine",level:3},{value:"Setting Opening Method",id:"setting-opening-method",level:3},{value:"Using Bangs",id:"using-bangs",level:3},{value:"Disabling Web Search",id:"disabling-web-search",level:3},{value:"Opening URLs Directly",id:"opening-urls-directly",level:3},{value:"Clearing Search",id:"clearing-search",level:2}];function l(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",kbd:"kbd",li:"li",p:"p",pre:"pre",ul:"ul",...(0,t.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"keyboard-shortcuts",children:"Keyboard Shortcuts"})}),"\n",(0,r.jsx)(n.h2,{id:"searching",children:"Searching"}),"\n",(0,r.jsx)(n.p,{children:"One of the primary purposes of Dashy is to allow you to quickly find and launch a given app. To make this as quick as possible, there is no need to touch the mouse, or press a certain key to begin searching - just start typing. Results will be filtered in real-time. No need to worry about case, special characters or small typos, these are taken care of, and your results should appear."}),"\n",(0,r.jsx)(n.h2,{id:"navigating",children:"Navigating"}),"\n",(0,r.jsxs)(n.p,{children:["You can navigate through your items or search results using the keyboard. You can use ",(0,r.jsx)(n.kbd,{children:"Tab"})," to cycle through results, and ",(0,r.jsx)(n.kbd,{children:"Shift"})," + ",(0,r.jsx)(n.kbd,{children:"Tab"})," to go backwards. Or use the arrow keys, ",(0,r.jsx)(n.kbd,{children:"\u2191"}),", ",(0,r.jsx)(n.kbd,{children:"\u2192"}),", ",(0,r.jsx)(n.kbd,{children:"\u2193"})," and ",(0,r.jsx)(n.kbd,{children:"\u2190"}),"."]}),"\n",(0,r.jsx)(n.h2,{id:"launching-apps",children:"Launching Apps"}),"\n",(0,r.jsxs)(n.p,{children:["You can launch a elected app by hitting ",(0,r.jsx)(n.kbd,{children:"Enter"}),". This will open the app using your default opening method, specified in ",(0,r.jsx)(n.code,{children:"target"})," (either ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"}),", ",(0,r.jsx)(n.code,{children:"modal"}),", ",(0,r.jsx)(n.code,{children:"top"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),"). You can also use ",(0,r.jsx)(n.kbd,{children:"Alt"})," + ",(0,r.jsx)(n.kbd,{children:"Enter"})," to open the app in a pop-up modal, or ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"Enter"})," to open it in a new tab. For all available opening methods, just right-click on an item, to bring up the context menu."]}),"\n",(0,r.jsx)(n.h2,{id:"tags",children:"Tags"}),"\n",(0,r.jsxs)(n.p,{children:["By default, items are filtered by the ",(0,r.jsx)(n.code,{children:"title"})," attribute, as well as the hostname (extracted from ",(0,r.jsx)(n.code,{children:"url"}),"), the ",(0,r.jsx)(n.code,{children:"provider"})," and ",(0,r.jsx)(n.code,{children:"description"}),". If you need to find results based on text which isn't included in these attributes, then you can add ",(0,r.jsx)(n.code,{children:"tags"})," to a given item."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" items:\n - title: Plex\n description: Media library\n icon: favicon\n url: https://plex.lab.local\n tags: [ movies, videos, music ]\n - title: FreshRSS\n description: RSS Reader\n icon: favicon\n url: https://freshrss.lab.local\n tags: [ news, updates, blogs ]\n\n"})}),"\n",(0,r.jsx)(n.p,{children:"In the above example, Plex will be visible when searching for 'movies', and FreshRSS with 'news'"}),"\n",(0,r.jsx)(n.h2,{id:"custom-hotkeys",children:"Custom Hotkeys"}),"\n",(0,r.jsxs)(n.p,{children:["For apps that you use regularly, you can set a custom keybinding. Use the ",(0,r.jsx)(n.code,{children:"hotkey"})," parameter on a certain item to specify a numeric key, between ",(0,r.jsx)(n.code,{children:"0 - 9"}),". You can then launch that app, by just pressing that key, which is much quicker than searching for it, if it's an app you use frequently."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- title: Bookstack\n icon: far fa-books\n url: https://bookstack.lab.local/\n hotkey: 2\n- title: Git Tea\n icon: fab fa-git\n url: https://git.lab.local/\n target: workspace\n hotkey: 3\n"})}),"\n",(0,r.jsxs)(n.p,{children:["In the above example, pressing ",(0,r.jsx)(n.kbd,{children:"2"})," will launch Bookstack. Or hitting ",(0,r.jsx)(n.kbd,{children:"3"})," will open Git in the workspace view."]}),"\n",(0,r.jsx)(n.h2,{id:"web-search",children:"Web Search"}),"\n",(0,r.jsxs)(n.p,{children:["It's possible to search the web directly from Dashy, which might be useful if you're using Dashy as your start page. This can be done by typing your query as normal, and then pressing ",(0,r.jsx)(n.kbd,{children:"\u23ce"}),". Web search options are configured under ",(0,r.jsx)(n.code,{children:"appConfig.webSearch"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"setting-search-engine",children:"Setting Search Engine"}),"\n",(0,r.jsxs)(n.p,{children:["Set your default search engine using the ",(0,r.jsx)(n.code,{children:"webSearch.searchEngine"})," property. This defaults to DuckDuckGo. Search engine must be referenced by their key, the following providers are supported:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://duckduckgo.com",children:(0,r.jsx)(n.code,{children:"duckduckgo"})}),", ",(0,r.jsx)(n.a,{href:"https://google.com",children:(0,r.jsx)(n.code,{children:"google"})}),", ",(0,r.jsx)(n.a,{href:"https://whoogle.sdf.org",children:(0,r.jsx)(n.code,{children:"whoogle"})}),", ",(0,r.jsx)(n.a,{href:"https://www.qwant.com",children:(0,r.jsx)(n.code,{children:"qwant"})}),", ",(0,r.jsx)(n.a,{href:"https://www.startpage.com",children:(0,r.jsx)(n.code,{children:"startpage"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.bar",children:(0,r.jsx)(n.code,{children:"searx-bar"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.info",children:(0,r.jsx)(n.code,{children:"searx-info"})})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://searx.tiekoetter.com",children:(0,r.jsx)(n.code,{children:"searx-tiekoetter"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.bissisoft.com",children:(0,r.jsx)(n.code,{children:"searx-bissisoft"})}),", ",(0,r.jsx)(n.a,{href:"https://www.ecosia.org",children:(0,r.jsx)(n.code,{children:"ecosia"})}),", ",(0,r.jsx)(n.a,{href:"https://metager.org/meta",children:(0,r.jsx)(n.code,{children:"metager"})}),", ",(0,r.jsx)(n.a,{href:"https://swisscows.com",children:(0,r.jsx)(n.code,{children:"swisscows"})}),", ",(0,r.jsx)(n.a,{href:"https://www.mojeek.com",children:(0,r.jsx)(n.code,{children:"mojeek"})})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://en.wikipedia.org",children:(0,r.jsx)(n.code,{children:"wikipedia"})}),", ",(0,r.jsx)(n.a,{href:"https://www.wolframalpha.com",children:(0,r.jsx)(n.code,{children:"wolframalpha"})}),", ",(0,r.jsx)(n.a,{href:"https://stackoverflow.com",children:(0,r.jsx)(n.code,{children:"stackoverflow"})}),", ",(0,r.jsx)(n.a,{href:"https://github.com",children:(0,r.jsx)(n.code,{children:"github"})}),", ",(0,r.jsx)(n.a,{href:"https://www.reddit.com",children:(0,r.jsx)(n.code,{children:"reddit"})}),", ",(0,r.jsx)(n.a,{href:"https://youtube.com",children:(0,r.jsx)(n.code,{children:"youtube"})}),", ",(0,r.jsx)(n.a,{href:"https://www.bbc.co.uk",children:(0,r.jsx)(n.code,{children:"bbc"})})]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"using-custom-search-engine",children:"Using Custom Search Engine"}),"\n",(0,r.jsxs)(n.p,{children:["You can also use a custom search engine, that isn't included in the above list (like a self-hosted instance of ",(0,r.jsx)(n.a,{href:"https://github.com/benbusby/whoogle-search",children:"Whoogle"})," or ",(0,r.jsx)(n.a,{href:"https://searx.github.io/searx/",children:"Searx"}),"). Set ",(0,r.jsx)(n.code,{children:"searchEngine: custom"}),", and then specify the URL (plus query params) to you're search engine under ",(0,r.jsx)(n.code,{children:"customSearchEngine"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n searchEngine: custom\n customSearchEngine: 'https://searx.local/search?q='\n"})}),"\n",(0,r.jsx)(n.h3,{id:"setting-opening-method",children:"Setting Opening Method"}),"\n",(0,r.jsxs)(n.p,{children:["In a similar way to opening apps, you can specify where you would like search results to be opened. This is done under the ",(0,r.jsx)(n.code,{children:"openingMethod"})," attribute, and can be set to either ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),". By default results are opened in a new tab."]}),"\n",(0,r.jsx)(n.h3,{id:"using-bangs",children:"Using Bangs"}),"\n",(0,r.jsxs)(n.p,{children:["An insanely useful feature of DDG is ",(0,r.jsx)(n.a,{href:"https://duckduckgo.com/bang",children:"Bangs"}),", where you type a specific character combination at the start of your search query, and it will be redirected the that website, such as '!w Docker' will display the Docker wikipedia page. Dashy has a similar feature, enabling you to define your own custom bangs to redirect search results to a specific app, website or search engine."]}),"\n",(0,r.jsxs)(n.p,{children:["This is done under the ",(0,r.jsx)(n.code,{children:"searchBangs"})," property, with a list of key value pairs. The key is what you will type, and the value is the destination, either as an identifier or a URL with query parameters."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n searchEngine: 'duckduckgo'\n openingMethod: 'newtab'\n searchBangs:\n /r: reddit\n /w: wikipedia\n /s: https://whoogle.local/search?q=\n /a: https://www.amazon.co.uk/s?k=\n ':wolf': wolframalpha\n ':so': stackoverflow\n ':git': github\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Note that bangs begging with ",(0,r.jsx)(n.code,{children:"!"})," or ",(0,r.jsx)(n.code,{children:":"})," must be surrounded them in quotes"]}),"\n",(0,r.jsx)(n.h3,{id:"disabling-web-search",children:"Disabling Web Search"}),"\n",(0,r.jsxs)(n.p,{children:["Web search can be disabled, by setting ",(0,r.jsx)(n.code,{children:"disableWebSearch"}),", for example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch: { disableWebSearch: true }\n"})}),"\n",(0,r.jsx)(n.h3,{id:"opening-urls-directly",children:"Opening URLs Directly"}),"\n",(0,r.jsxs)(n.p,{children:["When enabled, if your search query looks like a URL (e.g. ",(0,r.jsx)(n.code,{children:"github.com"}),", ",(0,r.jsx)(n.code,{children:"https://example.org/path"}),"), pressing ",(0,r.jsx)(n.kbd,{children:"Enter"})," will navigate directly to that URL instead of searching for it."]}),"\n",(0,r.jsxs)(n.p,{children:["Set ",(0,r.jsx)(n.code,{children:"appConfig.webSearch.openUrlsDirectly"})," to ",(0,r.jsx)(n.code,{children:"true"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n openUrlsDirectly: true\n"})}),"\n",(0,r.jsx)(n.h2,{id:"clearing-search",children:"Clearing Search"}),"\n",(0,r.jsxs)(n.p,{children:["You can clear your search term at any time, resting the UI to it's initial state, by pressing ",(0,r.jsx)(n.kbd,{children:"Esc"}),".\nThis can also be used to close any open pop-up modals."]})]})}function d(e={}){const{wrapper:n}={...(0,t.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(l,{...e})}):l(e)}},8453(e,n,s){s.d(n,{R:()=>a,x:()=>c});var i=s(6540);const r={},t=i.createContext(r);function a(e){const n=i.useContext(t);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function c(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:a(e.components),i.createElement(t.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[7151],{5230(e,n,s){s.r(n),s.d(n,{assets:()=>h,contentTitle:()=>c,default:()=>d,frontMatter:()=>a,metadata:()=>i,toc:()=>o});const i=JSON.parse('{"id":"searching","title":"Keyboard Shortcuts","description":"Searching","source":"@site/docs/searching.md","sourceDirName":".","slug":"/searching","permalink":"/docs/searching","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/searching.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Authentication","permalink":"/docs/authentication"},"next":{"title":"Alternate Views & Opening Methods","permalink":"/docs/alternate-views"}}');var r=s(4848),t=s(8453);const a={},c="Keyboard Shortcuts",h={},o=[{value:"Searching",id:"searching",level:2},{value:"Navigating",id:"navigating",level:2},{value:"Launching Apps",id:"launching-apps",level:2},{value:"Tags",id:"tags",level:2},{value:"Custom Hotkeys",id:"custom-hotkeys",level:2},{value:"Web Search",id:"web-search",level:2},{value:"Setting Search Engine",id:"setting-search-engine",level:3},{value:"Using Custom Search Engine",id:"using-custom-search-engine",level:3},{value:"Setting Opening Method",id:"setting-opening-method",level:3},{value:"Using Bangs",id:"using-bangs",level:3},{value:"Disabling Web Search",id:"disabling-web-search",level:3},{value:"Opening URLs Directly",id:"opening-urls-directly",level:3},{value:"Clearing Search",id:"clearing-search",level:2}];function l(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",kbd:"kbd",li:"li",p:"p",pre:"pre",ul:"ul",...(0,t.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"keyboard-shortcuts",children:"Keyboard Shortcuts"})}),"\n",(0,r.jsx)(n.h2,{id:"searching",children:"Searching"}),"\n",(0,r.jsx)(n.p,{children:"One of the primary purposes of Dashy is to allow you to quickly find and launch a given app. To make this as quick as possible, there is no need to touch the mouse, or press a certain key to begin searching - just start typing. Results will be filtered in real-time. No need to worry about case, special characters or small typos, these are taken care of, and your results should appear."}),"\n",(0,r.jsx)(n.h2,{id:"navigating",children:"Navigating"}),"\n",(0,r.jsxs)(n.p,{children:["You can navigate through your items or search results using the keyboard. You can use ",(0,r.jsx)(n.kbd,{children:"Tab"})," to cycle through results, and ",(0,r.jsx)(n.kbd,{children:"Shift"})," + ",(0,r.jsx)(n.kbd,{children:"Tab"})," to go backwards. Or use the arrow keys, ",(0,r.jsx)(n.kbd,{children:"\u2191"}),", ",(0,r.jsx)(n.kbd,{children:"\u2192"}),", ",(0,r.jsx)(n.kbd,{children:"\u2193"})," and ",(0,r.jsx)(n.kbd,{children:"\u2190"}),"."]}),"\n",(0,r.jsx)(n.h2,{id:"launching-apps",children:"Launching Apps"}),"\n",(0,r.jsxs)(n.p,{children:["You can launch a elected app by hitting ",(0,r.jsx)(n.kbd,{children:"Enter"}),". This will open the app using your default opening method, specified in ",(0,r.jsx)(n.code,{children:"target"})," (either ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"}),", ",(0,r.jsx)(n.code,{children:"modal"}),", ",(0,r.jsx)(n.code,{children:"top"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),"). You can also use ",(0,r.jsx)(n.kbd,{children:"Alt"})," + ",(0,r.jsx)(n.kbd,{children:"Enter"})," to open the app in a pop-up modal, or ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"Enter"})," to open it in a new tab. For all available opening methods, just right-click on an item, to bring up the context menu."]}),"\n",(0,r.jsx)(n.h2,{id:"tags",children:"Tags"}),"\n",(0,r.jsxs)(n.p,{children:["By default, items are filtered by the ",(0,r.jsx)(n.code,{children:"title"})," attribute, as well as the hostname (extracted from ",(0,r.jsx)(n.code,{children:"url"}),"), the ",(0,r.jsx)(n.code,{children:"provider"})," and ",(0,r.jsx)(n.code,{children:"description"}),". If you need to find results based on text which isn't included in these attributes, then you can add ",(0,r.jsx)(n.code,{children:"tags"})," to a given item."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" items:\n - title: Plex\n description: Media library\n icon: favicon\n url: https://plex.lab.local\n tags: [ movies, videos, music ]\n - title: FreshRSS\n description: RSS Reader\n icon: favicon\n url: https://freshrss.lab.local\n tags: [ news, updates, blogs ]\n\n"})}),"\n",(0,r.jsx)(n.p,{children:"In the above example, Plex will be visible when searching for 'movies', and FreshRSS with 'news'"}),"\n",(0,r.jsx)(n.h2,{id:"custom-hotkeys",children:"Custom Hotkeys"}),"\n",(0,r.jsxs)(n.p,{children:["For apps that you use regularly, you can set a custom keybinding. Use the ",(0,r.jsx)(n.code,{children:"hotkey"})," parameter on a certain item to specify a numeric key, between ",(0,r.jsx)(n.code,{children:"0 - 9"}),". You can then launch that app, by just pressing that key, which is much quicker than searching for it, if it's an app you use frequently."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- title: Bookstack\n icon: far fa-books\n url: https://bookstack.lab.local/\n hotkey: 2\n- title: Git Tea\n icon: fab fa-git\n url: https://git.lab.local/\n target: workspace\n hotkey: 3\n"})}),"\n",(0,r.jsxs)(n.p,{children:["In the above example, pressing ",(0,r.jsx)(n.kbd,{children:"2"})," will launch Bookstack. Or hitting ",(0,r.jsx)(n.kbd,{children:"3"})," will open Git in the workspace view."]}),"\n",(0,r.jsx)(n.h2,{id:"web-search",children:"Web Search"}),"\n",(0,r.jsxs)(n.p,{children:["It's possible to search the web directly from Dashy, which might be useful if you're using Dashy as your start page. This can be done by typing your query as normal, and then pressing ",(0,r.jsx)(n.kbd,{children:"\u23ce"}),". Web search options are configured under ",(0,r.jsx)(n.code,{children:"appConfig.webSearch"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"setting-search-engine",children:"Setting Search Engine"}),"\n",(0,r.jsxs)(n.p,{children:["Set your default search engine using the ",(0,r.jsx)(n.code,{children:"webSearch.searchEngine"})," property. This defaults to DuckDuckGo. Search engine must be referenced by their key, the following providers are supported:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://duckduckgo.com",children:(0,r.jsx)(n.code,{children:"duckduckgo"})}),", ",(0,r.jsx)(n.a,{href:"https://google.com",children:(0,r.jsx)(n.code,{children:"google"})}),", ",(0,r.jsx)(n.a,{href:"https://whoogle.sdf.org",children:(0,r.jsx)(n.code,{children:"whoogle"})}),", ",(0,r.jsx)(n.a,{href:"https://www.qwant.com",children:(0,r.jsx)(n.code,{children:"qwant"})}),", ",(0,r.jsx)(n.a,{href:"https://www.startpage.com",children:(0,r.jsx)(n.code,{children:"startpage"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.bar",children:(0,r.jsx)(n.code,{children:"searx-bar"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.info",children:(0,r.jsx)(n.code,{children:"searx-info"})})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://searx.tiekoetter.com",children:(0,r.jsx)(n.code,{children:"searx-tiekoetter"})}),", ",(0,r.jsx)(n.a,{href:"https://searx.bissisoft.com",children:(0,r.jsx)(n.code,{children:"searx-bissisoft"})}),", ",(0,r.jsx)(n.a,{href:"https://www.ecosia.org",children:(0,r.jsx)(n.code,{children:"ecosia"})}),", ",(0,r.jsx)(n.a,{href:"https://metager.org/meta",children:(0,r.jsx)(n.code,{children:"metager"})}),", ",(0,r.jsx)(n.a,{href:"https://swisscows.com",children:(0,r.jsx)(n.code,{children:"swisscows"})}),", ",(0,r.jsx)(n.a,{href:"https://www.mojeek.com",children:(0,r.jsx)(n.code,{children:"mojeek"})})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"https://en.wikipedia.org",children:(0,r.jsx)(n.code,{children:"wikipedia"})}),", ",(0,r.jsx)(n.a,{href:"https://www.wolframalpha.com",children:(0,r.jsx)(n.code,{children:"wolframalpha"})}),", ",(0,r.jsx)(n.a,{href:"https://stackoverflow.com",children:(0,r.jsx)(n.code,{children:"stackoverflow"})}),", ",(0,r.jsx)(n.a,{href:"https://github.com",children:(0,r.jsx)(n.code,{children:"github"})}),", ",(0,r.jsx)(n.a,{href:"https://www.reddit.com",children:(0,r.jsx)(n.code,{children:"reddit"})}),", ",(0,r.jsx)(n.a,{href:"https://youtube.com",children:(0,r.jsx)(n.code,{children:"youtube"})}),", ",(0,r.jsx)(n.a,{href:"https://www.bbc.co.uk",children:(0,r.jsx)(n.code,{children:"bbc"})})]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"using-custom-search-engine",children:"Using Custom Search Engine"}),"\n",(0,r.jsxs)(n.p,{children:["You can also use a custom search engine, that isn't included in the above list (like a self-hosted instance of ",(0,r.jsx)(n.a,{href:"https://github.com/benbusby/whoogle-search",children:"Whoogle"})," or ",(0,r.jsx)(n.a,{href:"https://searx.github.io/searx/",children:"Searx"}),"). Set ",(0,r.jsx)(n.code,{children:"searchEngine: custom"}),", and then specify the URL (plus query params) to you're search engine under ",(0,r.jsx)(n.code,{children:"customSearchEngine"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n searchEngine: custom\n customSearchEngine: 'https://searx.local/search?q='\n"})}),"\n",(0,r.jsx)(n.h3,{id:"setting-opening-method",children:"Setting Opening Method"}),"\n",(0,r.jsxs)(n.p,{children:["In a similar way to opening apps, you can specify where you would like search results to be opened. This is done under the ",(0,r.jsx)(n.code,{children:"openingMethod"})," attribute, and can be set to either ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),". By default results are opened in a new tab."]}),"\n",(0,r.jsx)(n.h3,{id:"using-bangs",children:"Using Bangs"}),"\n",(0,r.jsxs)(n.p,{children:["An insanely useful feature of DDG is ",(0,r.jsx)(n.a,{href:"https://duckduckgo.com/bang",children:"Bangs"}),", where you type a specific character combination at the start of your search query, and it will be redirected the that website, such as '!w Docker' will display the Docker wikipedia page. Dashy has a similar feature, enabling you to define your own custom bangs to redirect search results to a specific app, website or search engine."]}),"\n",(0,r.jsxs)(n.p,{children:["This is done under the ",(0,r.jsx)(n.code,{children:"searchBangs"})," property, with a list of key value pairs. The key is what you will type, and the value is the destination, either as an identifier or a URL with query parameters."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n searchEngine: 'duckduckgo'\n openingMethod: 'newtab'\n searchBangs:\n /r: reddit\n /w: wikipedia\n /s: https://whoogle.local/search?q=\n /a: https://www.amazon.co.uk/s?k=\n ':wolf': wolframalpha\n ':so': stackoverflow\n ':git': github\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Note that bangs begging with ",(0,r.jsx)(n.code,{children:"!"})," or ",(0,r.jsx)(n.code,{children:":"})," must be surrounded them in quotes"]}),"\n",(0,r.jsx)(n.h3,{id:"disabling-web-search",children:"Disabling Web Search"}),"\n",(0,r.jsxs)(n.p,{children:["Web search can be disabled, by setting ",(0,r.jsx)(n.code,{children:"disableWebSearch"}),", for example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch: { disableWebSearch: true }\n"})}),"\n",(0,r.jsx)(n.h3,{id:"opening-urls-directly",children:"Opening URLs Directly"}),"\n",(0,r.jsxs)(n.p,{children:["When enabled, if your search query looks like a URL (e.g. ",(0,r.jsx)(n.code,{children:"github.com"}),", ",(0,r.jsx)(n.code,{children:"https://example.org/path"}),"), pressing ",(0,r.jsx)(n.kbd,{children:"Enter"})," will navigate directly to that URL instead of searching for it."]}),"\n",(0,r.jsxs)(n.p,{children:["Set ",(0,r.jsx)(n.code,{children:"appConfig.webSearch.openUrlsDirectly"})," to ",(0,r.jsx)(n.code,{children:"true"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n webSearch:\n openUrlsDirectly: true\n"})}),"\n",(0,r.jsx)(n.h2,{id:"clearing-search",children:"Clearing Search"}),"\n",(0,r.jsxs)(n.p,{children:["You can clear your search term at any time, resting the UI to it's initial state, by pressing ",(0,r.jsx)(n.kbd,{children:"Esc"}),".\nThis can also be used to close any open pop-up modals."]})]})}function d(e={}){const{wrapper:n}={...(0,t.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(l,{...e})}):l(e)}},8453(e,n,s){s.d(n,{R:()=>a,x:()=>c});var i=s(6540);const r={},t=i.createContext(r);function a(e){const n=i.useContext(t);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function c(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:a(e.components),i.createElement(t.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/1745c531.cc638927.js b/assets/js/1745c531.11084348.js similarity index 94% rename from assets/js/1745c531.cc638927.js rename to assets/js/1745c531.11084348.js index b53018e2..f3636445 100644 --- a/assets/js/1745c531.cc638927.js +++ b/assets/js/1745c531.11084348.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[3790],{2092(e,t,s){s.r(t),s.d(t,{assets:()=>d,contentTitle:()=>r,default:()=>h,frontMatter:()=>a,metadata:()=>o,toc:()=>i});const o=JSON.parse('{"id":"showcase/readme","title":"readme","description":"See: Showcase.","source":"@site/docs/showcase/readme.md","sourceDirName":"showcase","slug":"/showcase/","permalink":"/docs/showcase/","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/showcase/readme.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{}}');var n=s(4848),c=s(8453);const a={},r=void 0,d={},i=[];function u(e){const t={a:"a",p:"p",...(0,c.R)(),...e.components};return(0,n.jsxs)(t.p,{children:["See: ",(0,n.jsx)(t.a,{href:"/docs/showcase",children:"Showcase"}),"."]})}function h(e={}){const{wrapper:t}={...(0,c.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(u,{...e})}):u(e)}},8453(e,t,s){s.d(t,{R:()=>a,x:()=>r});var o=s(6540);const n={},c=o.createContext(n);function a(e){const t=o.useContext(c);return o.useMemo(function(){return"function"==typeof e?e(t):{...t,...e}},[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(n):e.components||n:a(e.components),o.createElement(c.Provider,{value:t},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[3790],{2092(e,t,s){s.r(t),s.d(t,{assets:()=>d,contentTitle:()=>r,default:()=>h,frontMatter:()=>a,metadata:()=>o,toc:()=>i});const o=JSON.parse('{"id":"showcase/readme","title":"readme","description":"See: Showcase.","source":"@site/docs/showcase/readme.md","sourceDirName":"showcase","slug":"/showcase/","permalink":"/docs/showcase/","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/showcase/readme.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{}}');var n=s(4848),c=s(8453);const a={},r=void 0,d={},i=[];function u(e){const t={a:"a",p:"p",...(0,c.R)(),...e.components};return(0,n.jsxs)(t.p,{children:["See: ",(0,n.jsx)(t.a,{href:"/docs/showcase",children:"Showcase"}),"."]})}function h(e={}){const{wrapper:t}={...(0,c.R)(),...e.components};return t?(0,n.jsx)(t,{...e,children:(0,n.jsx)(u,{...e})}):u(e)}},8453(e,t,s){s.d(t,{R:()=>a,x:()=>r});var o=s(6540);const n={},c=o.createContext(n);function a(e){const t=o.useContext(c);return o.useMemo(function(){return"function"==typeof e?e(t):{...t,...e}},[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(n):e.components||n:a(e.components),o.createElement(c.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/18ba09e8.96348509.js b/assets/js/18ba09e8.e1d0267a.js similarity index 98% rename from assets/js/18ba09e8.96348509.js rename to assets/js/18ba09e8.e1d0267a.js index 756fa753..cd043cc3 100644 --- a/assets/js/18ba09e8.96348509.js +++ b/assets/js/18ba09e8.e1d0267a.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[6849],{4580(e,s,t){t.r(s),t.d(s,{assets:()=>d,contentTitle:()=>c,default:()=>a,frontMatter:()=>o,metadata:()=>n,toc:()=>h});const n=JSON.parse('{"id":"credits","title":"Credits","description":"Sponsors","source":"@site/docs/credits.md","sourceDirName":".","slug":"/credits","permalink":"/docs/credits","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/credits.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Privacy & Security","permalink":"/docs/privacy"},"next":{"title":"License","permalink":"/docs/license"}}');var i=t(4848),r=t(8453);const o={},c="Credits",d={},h=[{value:"Sponsors",id:"sponsors",level:2},{value:"Contributors",id:"contributors",level:2},{value:"Newest Stargazers",id:"newest-stargazers",level:2},{value:"Dependencies",id:"dependencies",level:2},{value:"Core",id:"core",level:3},{value:"Utilities",id:"utilities",level:3},{value:"Frontend Components",id:"frontend-components",level:3},{value:"You",id:"you",level:2}];function l(e){const s={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",strong:"strong",ul:"ul",...(0,r.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(s.header,{children:(0,i.jsx)(s.h1,{id:"credits",children:"Credits"})}),"\n",(0,i.jsx)(s.h2,{id:"sponsors",children:"Sponsors"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/sponsors/lissy93",alt:"Sponsors"})}),"\n",(0,i.jsx)(s.h2,{id:"contributors",children:"Contributors"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/contributors/lissy93/dashy?perRow=12&limit=96",alt:"Contributors"})}),"\n",(0,i.jsx)(s.h2,{id:"newest-stargazers",children:"Newest Stargazers"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/stargazers",children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/stargazers/lissy93/dashy?perRow=16&fontSize=10&limit=96",alt:"Stargazers"})})}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://github.com/lissy93/dashy/network/members",children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/forkers/lissy93/dashy?perRow=16&fontSize=10&limit=96",alt:"Forkers"})})}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"dependencies",children:"Dependencies"}),"\n",(0,i.jsx)(s.p,{children:"This app definitely wouldn't have been quite so possible without the making use of the following package and components. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them"}),"\n",(0,i.jsx)(s.h3,{id:"core",children:"Core"}),"\n",(0,i.jsxs)(s.p,{children:["At it's core, the application uses ",(0,i.jsx)(s.a,{href:"https://github.com/vuejs/vue",children:(0,i.jsx)(s.strong,{children:"Vue.js"})}),", as well as it's services with ",(0,i.jsx)(s.a,{href:"https://vuex.vuejs.org/",children:(0,i.jsx)(s.strong,{children:"VueX"})})," for state management. Styling is done with ",(0,i.jsx)(s.a,{href:"https://github.com/sass/sass",children:(0,i.jsx)(s.strong,{children:"SCSS"})}),", JavaScript is currently ",(0,i.jsx)(s.a,{href:"https://github.com/babel/babel",children:(0,i.jsx)(s.strong,{children:"Babel"})}),", (but I am in the process of converting to ",(0,i.jsx)(s.a,{href:"https://github.com/Microsoft/TypeScript",children:(0,i.jsx)(s.strong,{children:"TypeScript"})}),"). Linting is done with ",(0,i.jsx)(s.a,{href:"https://github.com/eslint/eslint",children:(0,i.jsx)(s.strong,{children:"ESLint"})})," and ",(0,i.jsx)(s.a,{href:"https://prettier.io/",children:(0,i.jsx)(s.strong,{children:"Prettier"})}),", both following the ",(0,i.jsx)(s.a,{href:"https://github.com/airbnb/javascript",children:(0,i.jsx)(s.strong,{children:"AirBnB Styleguide"})}),". The config is defined in ",(0,i.jsx)(s.a,{href:"https://github.com/yaml/yaml",children:(0,i.jsx)(s.strong,{children:"YAML"})}),", with a simple ",(0,i.jsx)(s.a,{href:"https://github.com/nodejs/node",children:(0,i.jsx)(s.strong,{children:"Node.js"})})," server to serve up the static app and the optional API endpoints, and container deployment is done with ",(0,i.jsx)(s.a,{href:"https://www.docker.com/",children:(0,i.jsx)(s.strong,{children:"Docker"})}),"."]}),"\n",(0,i.jsx)(s.h3,{id:"utilities",children:"Utilities"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/brix/crypto-js",children:(0,i.jsx)(s.code,{children:"crypto-js"})})," - Encryption implementations by @evanvosberg and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/axios/axios",children:(0,i.jsx)(s.code,{children:"axios"})})," - Promise based HTTP client by @mzabriskie and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/ajv-validator/ajv",children:(0,i.jsx)(s.code,{children:"ajv"})})," - JSON schema Validator by @epoberezkin and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/kazupon/vue-i18n",children:(0,i.jsx)(s.code,{children:"i18n"})})," - Internationalization plugin by @kazupon and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/frappe/charts",children:(0,i.jsx)(s.code,{children:"frappe-charts"})})," - Lightweight charting library by @frappe ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n"]}),"\n",(0,i.jsx)(s.h3,{id:"frontend-components",children:"Frontend Components"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/sagalbot/vue-select",children:(0,i.jsx)(s.code,{children:"vue-select"})})," - Dropdown component by @sagalbot ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/euvl/vue-js-modal",children:(0,i.jsx)(s.code,{children:"vue-js-modal"})})," - Modal component by @euvl ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/Akryum/v-tooltip",children:(0,i.jsx)(s.code,{children:"v-tooltip"})})," - Tooltip component by @Akryum ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/jairoblatt/vue-material-tabs",children:(0,i.jsx)(s.code,{children:"vue-material-tabs"})})," - Tab view component by @jairoblatt ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/yansenlei/VJsoneditor",children:(0,i.jsx)(s.code,{children:"VJsoneditor"})})," - Interactive JSON editor component by @yansenlei ",(0,i.jsx)(s.code,{children:"MIT"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:["Forked from ",(0,i.jsx)(s.a,{href:"https://github.com/josdejong/jsoneditor",children:(0,i.jsx)(s.code,{children:"JsonEditor"})})," by @josdejong ",(0,i.jsx)(s.code,{children:"Apache-2.0 License"})]}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/saintplay/vue-swatches",children:(0,i.jsx)(s.code,{children:"vue-swatches"})})," - Color palete picker by @saintplay ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"you",children:"You"}),"\n",(0,i.jsxs)(s.p,{children:["Would you like to be listed here? Whatever your skill set, Dashy needs people like you to help support future development. Check out the ",(0,i.jsx)(s.a,{href:"/docs/contributing",children:"Contributing Page"})," for ways that you can get involved. Huge thank you to everyone who has already contributed! \ud83d\udc96"]})]})}function a(e={}){const{wrapper:s}={...(0,r.R)(),...e.components};return s?(0,i.jsx)(s,{...e,children:(0,i.jsx)(l,{...e})}):l(e)}},8453(e,s,t){t.d(s,{R:()=>o,x:()=>c});var n=t(6540);const i={},r=n.createContext(i);function o(e){const s=n.useContext(r);return n.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function c(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),n.createElement(r.Provider,{value:s},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[6849],{4580(e,s,t){t.r(s),t.d(s,{assets:()=>d,contentTitle:()=>c,default:()=>a,frontMatter:()=>o,metadata:()=>n,toc:()=>h});const n=JSON.parse('{"id":"credits","title":"Credits","description":"Sponsors","source":"@site/docs/credits.md","sourceDirName":".","slug":"/credits","permalink":"/docs/credits","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/credits.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Privacy & Security","permalink":"/docs/privacy"},"next":{"title":"License","permalink":"/docs/license"}}');var i=t(4848),r=t(8453);const o={},c="Credits",d={},h=[{value:"Sponsors",id:"sponsors",level:2},{value:"Contributors",id:"contributors",level:2},{value:"Newest Stargazers",id:"newest-stargazers",level:2},{value:"Dependencies",id:"dependencies",level:2},{value:"Core",id:"core",level:3},{value:"Utilities",id:"utilities",level:3},{value:"Frontend Components",id:"frontend-components",level:3},{value:"You",id:"you",level:2}];function l(e){const s={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",strong:"strong",ul:"ul",...(0,r.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(s.header,{children:(0,i.jsx)(s.h1,{id:"credits",children:"Credits"})}),"\n",(0,i.jsx)(s.h2,{id:"sponsors",children:"Sponsors"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/sponsors/lissy93",alt:"Sponsors"})}),"\n",(0,i.jsx)(s.h2,{id:"contributors",children:"Contributors"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/contributors/lissy93/dashy?perRow=12&limit=96",alt:"Contributors"})}),"\n",(0,i.jsx)(s.h2,{id:"newest-stargazers",children:"Newest Stargazers"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/stargazers",children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/stargazers/lissy93/dashy?perRow=16&fontSize=10&limit=96",alt:"Stargazers"})})}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://github.com/lissy93/dashy/network/members",children:(0,i.jsx)(s.img,{src:"https://readme-contribs.as93.net/forkers/lissy93/dashy?perRow=16&fontSize=10&limit=96",alt:"Forkers"})})}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"dependencies",children:"Dependencies"}),"\n",(0,i.jsx)(s.p,{children:"This app definitely wouldn't have been quite so possible without the making use of the following package and components. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them"}),"\n",(0,i.jsx)(s.h3,{id:"core",children:"Core"}),"\n",(0,i.jsxs)(s.p,{children:["At it's core, the application uses ",(0,i.jsx)(s.a,{href:"https://github.com/vuejs/vue",children:(0,i.jsx)(s.strong,{children:"Vue.js"})}),", as well as it's services with ",(0,i.jsx)(s.a,{href:"https://vuex.vuejs.org/",children:(0,i.jsx)(s.strong,{children:"VueX"})})," for state management. Styling is done with ",(0,i.jsx)(s.a,{href:"https://github.com/sass/sass",children:(0,i.jsx)(s.strong,{children:"SCSS"})}),", JavaScript is currently ",(0,i.jsx)(s.a,{href:"https://github.com/babel/babel",children:(0,i.jsx)(s.strong,{children:"Babel"})}),", (but I am in the process of converting to ",(0,i.jsx)(s.a,{href:"https://github.com/Microsoft/TypeScript",children:(0,i.jsx)(s.strong,{children:"TypeScript"})}),"). Linting is done with ",(0,i.jsx)(s.a,{href:"https://github.com/eslint/eslint",children:(0,i.jsx)(s.strong,{children:"ESLint"})})," and ",(0,i.jsx)(s.a,{href:"https://prettier.io/",children:(0,i.jsx)(s.strong,{children:"Prettier"})}),", both following the ",(0,i.jsx)(s.a,{href:"https://github.com/airbnb/javascript",children:(0,i.jsx)(s.strong,{children:"AirBnB Styleguide"})}),". The config is defined in ",(0,i.jsx)(s.a,{href:"https://github.com/yaml/yaml",children:(0,i.jsx)(s.strong,{children:"YAML"})}),", with a simple ",(0,i.jsx)(s.a,{href:"https://github.com/nodejs/node",children:(0,i.jsx)(s.strong,{children:"Node.js"})})," server to serve up the static app and the optional API endpoints, and container deployment is done with ",(0,i.jsx)(s.a,{href:"https://www.docker.com/",children:(0,i.jsx)(s.strong,{children:"Docker"})}),"."]}),"\n",(0,i.jsx)(s.h3,{id:"utilities",children:"Utilities"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/brix/crypto-js",children:(0,i.jsx)(s.code,{children:"crypto-js"})})," - Encryption implementations by @evanvosberg and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/axios/axios",children:(0,i.jsx)(s.code,{children:"axios"})})," - Promise based HTTP client by @mzabriskie and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/ajv-validator/ajv",children:(0,i.jsx)(s.code,{children:"ajv"})})," - JSON schema Validator by @epoberezkin and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/kazupon/vue-i18n",children:(0,i.jsx)(s.code,{children:"i18n"})})," - Internationalization plugin by @kazupon and community ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/frappe/charts",children:(0,i.jsx)(s.code,{children:"frappe-charts"})})," - Lightweight charting library by @frappe ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n"]}),"\n",(0,i.jsx)(s.h3,{id:"frontend-components",children:"Frontend Components"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/sagalbot/vue-select",children:(0,i.jsx)(s.code,{children:"vue-select"})})," - Dropdown component by @sagalbot ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/euvl/vue-js-modal",children:(0,i.jsx)(s.code,{children:"vue-js-modal"})})," - Modal component by @euvl ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/Akryum/v-tooltip",children:(0,i.jsx)(s.code,{children:"v-tooltip"})})," - Tooltip component by @Akryum ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/jairoblatt/vue-material-tabs",children:(0,i.jsx)(s.code,{children:"vue-material-tabs"})})," - Tab view component by @jairoblatt ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/yansenlei/VJsoneditor",children:(0,i.jsx)(s.code,{children:"VJsoneditor"})})," - Interactive JSON editor component by @yansenlei ",(0,i.jsx)(s.code,{children:"MIT"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:["Forked from ",(0,i.jsx)(s.a,{href:"https://github.com/josdejong/jsoneditor",children:(0,i.jsx)(s.code,{children:"JsonEditor"})})," by @josdejong ",(0,i.jsx)(s.code,{children:"Apache-2.0 License"})]}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://github.com/saintplay/vue-swatches",children:(0,i.jsx)(s.code,{children:"vue-swatches"})})," - Color palete picker by @saintplay ",(0,i.jsx)(s.code,{children:"MIT"})]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"you",children:"You"}),"\n",(0,i.jsxs)(s.p,{children:["Would you like to be listed here? Whatever your skill set, Dashy needs people like you to help support future development. Check out the ",(0,i.jsx)(s.a,{href:"/docs/contributing",children:"Contributing Page"})," for ways that you can get involved. Huge thank you to everyone who has already contributed! \ud83d\udc96"]})]})}function a(e={}){const{wrapper:s}={...(0,r.R)(),...e.components};return s?(0,i.jsx)(s,{...e,children:(0,i.jsx)(l,{...e})}):l(e)}},8453(e,s,t){t.d(s,{R:()=>o,x:()=>c});var n=t(6540);const i={},r=n.createContext(i);function o(e){const s=n.useContext(r);return n.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function c(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),n.createElement(r.Provider,{value:s},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/1f5fb0df.d4dc9dbe.js b/assets/js/1f5fb0df.dc0107fb.js similarity index 99% rename from assets/js/1f5fb0df.d4dc9dbe.js rename to assets/js/1f5fb0df.dc0107fb.js index a2565823..60b03532 100644 --- a/assets/js/1f5fb0df.d4dc9dbe.js +++ b/assets/js/1f5fb0df.dc0107fb.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9695],{6644(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>a,default:()=>h,frontMatter:()=>t,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"management","title":"App Management","description":"The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains.","source":"@site/docs/management.md","sourceDirName":".","slug":"/management","permalink":"/docs/management","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/management.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Configuring","permalink":"/docs/configuring"},"next":{"title":"Troubleshooting","permalink":"/docs/troubleshooting"}}');var r=s(4848),o=s(8453);const t={},a="App Management",c={},l=[{value:"Contents",id:"contents",level:2},{value:"Providing Assets",id:"providing-assets",level:2},{value:"File Ownership and Permissions",id:"file-ownership-and-permissions",level:2},{value:"Running Commands",id:"running-commands",level:2},{value:"Healthchecks",id:"healthchecks",level:2},{value:"HTTP Healthcheck Endpoint",id:"http-healthcheck-endpoint",level:3},{value:"Logs and Performance",id:"logs-and-performance",level:2},{value:"Container Logs",id:"container-logs",level:3},{value:"Container Performance",id:"container-performance",level:3},{value:"Management Apps",id:"management-apps",level:3},{value:"Advanced Logging and Monitoring",id:"advanced-logging-and-monitoring",level:3},{value:"Auto-Starting at System Boot",id:"auto-starting-at-system-boot",level:2},{value:"Updating",id:"updating",level:2},{value:"Updating Docker Container",id:"updating-docker-container",level:3},{value:"Automatic Docker Updates",id:"automatic-docker-updates",level:3},{value:"Updating Dashy from Source",id:"updating-dashy-from-source",level:3},{value:"Backing Up",id:"backing-up",level:2},{value:"Backing Up Containers",id:"backing-up-containers",level:3},{value:"Backing Up Volumes",id:"backing-up-volumes",level:3},{value:"Dashy-Specific Backup",id:"dashy-specific-backup",level:3},{value:"Scheduling",id:"scheduling",level:2},{value:"SSL Certificates",id:"ssl-certificates",level:2},{value:"Auto-SSL",id:"auto-ssl",level:3},{value:"Getting a Self-Signed SSL Certificate",id:"getting-a-self-signed-ssl-certificate",level:3},{value:"Passing a Self-Signed Certificate to Dashy",id:"passing-a-self-signed-certificate-to-dashy",level:3},{value:"Authentication",id:"authentication",level:2},{value:"Network Exposure",id:"network-exposure",level:2},{value:"Managing Containers with Docker Compose",id:"managing-containers-with-docker-compose",level:2},{value:"Passing in Environmental Variables",id:"passing-in-environmental-variables",level:2},{value:"Setting Headers",id:"setting-headers",level:2},{value:"Example Headers",id:"example-headers",level:3},{value:"Caddy",id:"caddy",level:4},{value:"NGINX",id:"nginx",level:4},{value:"Traefik",id:"traefik",level:4},{value:"HAProxy",id:"haproxy",level:4},{value:"Apache",id:"apache",level:4},{value:"Squid",id:"squid",level:4},{value:"Remote Access",id:"remote-access",level:2},{value:"WireGuard",id:"wireguard",level:3},{value:"Example Server Config",id:"example-server-config",level:4},{value:"Example Client Config",id:"example-client-config",level:4},{value:"Reverse SSH Tunnel",id:"reverse-ssh-tunnel",level:3},{value:"TCP Tunnel",id:"tcp-tunnel",level:3},{value:"Custom Domain",id:"custom-domain",level:2},{value:"Using DNS",id:"using-dns",level:3},{value:"Using NGINX",id:"using-nginx",level:3},{value:"Container Security",id:"container-security",level:2},{value:"Keep Docker Up-To-Date",id:"keep-docker-up-to-date",level:3},{value:"Set Resource Quotas",id:"set-resource-quotas",level:3},{value:"Don't Run as Root",id:"dont-run-as-root",level:3},{value:"Specify a User",id:"specify-a-user",level:3},{value:"Limit capabilities",id:"limit-capabilities",level:3},{value:"Prevent new Privileges being Added",id:"prevent-new-privileges-being-added",level:3},{value:"Disable Inter-Container Communication",id:"disable-inter-container-communication",level:3},{value:"Don't Expose the Docker Daemon Socket",id:"dont-expose-the-docker-daemon-socket",level:3},{value:"Use Read-Only Volumes",id:"use-read-only-volumes",level:3},{value:"Set the Logging Level",id:"set-the-logging-level",level:3},{value:"Verify Image before Pulling",id:"verify-image-before-pulling",level:3},{value:"Specify the Tag",id:"specify-the-tag",level:3},{value:"Container Security Scanning",id:"container-security-scanning",level:3},{value:"Registry Security",id:"registry-security",level:3},{value:"Security Modules",id:"security-modules",level:3},{value:"Web Server Configuration",id:"web-server-configuration",level:2},{value:"NGINX",id:"nginx-1",level:3},{value:"Apache",id:"apache-1",level:3},{value:"Caddy",id:"caddy-1",level:3},{value:"Firebase Hosting",id:"firebase-hosting",level:3},{value:"cPanel",id:"cpanel",level:3},{value:"Running a Modified Version of the App",id:"running-a-modified-version-of-the-app",level:2},{value:"Building your Own Container",id:"building-your-own-container",level:2}];function d(e){const n={a:"a",blockquote:"blockquote",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"app-management",children:"App Management"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains."})}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#providing-assets",children:"Providing Assets"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#file-ownership-and-permissions",children:"File Ownership and Permissions"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#running-commands",children:"Running Commands"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthchecks",children:"Healthchecks"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#logs-and-performance",children:"Logs and Performance"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#auto-starting-at-system-boot",children:"Auto-Starting at Boot"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#updating",children:"Updating"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#backing-up",children:"Backing Up"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#scheduling",children:"Scheduling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ssl-certificates",children:"SSL Certificates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#authentication",children:"Authentication"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-exposure",children:"Network Exposure"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#managing-containers-with-docker-compose",children:"Managing with Compose"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#passing-in-environmental-variables",children:"Environmental Variables"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#setting-headers",children:"Setting Headers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#remote-access",children:"Remote Access"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#custom-domain",children:"Custom Domain"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-security",children:"Securing Containers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#web-server-configuration",children:"Web Server Configuration"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#running-a-modified-version-of-the-app",children:"Running a Modified App"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#building-your-own-container",children:"Building your Own Container"})}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"providing-assets",children:"Providing Assets"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy reads everything for your dashboard from a single host directory mounted into the container at ",(0,r.jsx)(n.code,{children:"/app/user-data"}),". This is done with a ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/storage/volumes/",children:"Docker volume"}),", e.g. ",(0,r.jsx)(n.code,{children:"-v /path/to/your/user-data:/app/user-data"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["The directory must contain a ",(0,r.jsx)(n.code,{children:"conf.yml"}),". It can also contain anything else you want served from the web root: sub-config files for additional pages, item icons, favicon, fonts, custom CSS, manifest, and so on. Any file placed there is reachable at ",(0,r.jsx)(n.code,{children:"/"})," in the browser, overriding files of the same name in the bundled ",(0,r.jsx)(n.code,{children:"public/"})," defaults."]}),"\n",(0,r.jsx)(n.p,{children:"Typical contents:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"conf.yml"})," - Main config (required)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"*.yml"})," / ",(0,r.jsx)(n.code,{children:"*.yaml"})," - Sub-config files for ",(0,r.jsx)(n.a,{href:"/docs/pages-and-sections#multi-page-support",children:"multi-page support"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"item-icons/"})," - Local icons referenced by ",(0,r.jsx)(n.code,{children:"icon: ./item-icons/foo.png"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"favicon.ico"}),", ",(0,r.jsx)(n.code,{children:"manifest.json"}),", ",(0,r.jsx)(n.code,{children:"robots.txt"})," - Override the bundled defaults"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"fonts/"}),", ",(0,r.jsx)(n.code,{children:"widget-resources/"})," - Custom fonts or assets used by widgets"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"file-ownership-and-permissions",children:"File Ownership and Permissions"}),"\n",(0,r.jsxs)(n.p,{children:["Inside the container, Dashy runs as a non-root user with uid/gid 1000 (the built-in ",(0,r.jsx)(n.code,{children:"node"})," user from the Node base image). This is fine for the vast majority of installs, since the first user on a default Linux or macOS box is also uid 1000, so a bind-mounted ",(0,r.jsx)(n.code,{children:"user-data"})," directory is read/writable straight away."]}),"\n",(0,r.jsx)(n.p,{children:"It only gets fiddly if your host uid happens to be something else (NAS systems and multi-user servers being the usual culprits). In that case, config saves from the UI will fail with a permission error, because the container's uid 1000 doesn't own your directory."}),"\n",(0,r.jsx)(n.p,{children:"There are two ways to sort it. Pick whichever is less hassle:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Run the container as your own user."})," The cleanest option, since you don't touch host file ownership."]}),"\n",(0,r.jsxs)(n.p,{children:["On ",(0,r.jsx)(n.code,{children:"docker run"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d -p 8080:8080 \\\n --user $(id -u):$(id -g) \\\n -v /path/to/user-data:/app/user-data \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["In compose, uncomment the ",(0,r.jsx)(n.code,{children:"user:"})," line under the service and set it:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'user: "1001:1001" # whatever `id -u` and `id -g` give you\n'})}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Hand the directory to uid 1000."})," Quicker if you don't mind changing host ownership:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"sudo chown -R 1000:1000 /path/to/user-data\n"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["Note that if you run the container as root (e.g. ",(0,r.jsx)(n.code,{children:"--user 0:0"}),"), Dashy will still work, but you lose the security benefit of a non-root container. Don't do that unless you've a good reason."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"running-commands",children:"Running Commands"}),"\n",(0,r.jsxs)(n.p,{children:["If you're running an app in Docker, then commands will need to be passed to the container to be executed. This can be done by preceding each command with ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id]"}),", where container ID can be found by running ",(0,r.jsx)(n.code,{children:"docker ps"}),". For example ",(0,r.jsx)(n.code,{children:"docker exec -it 26c156c467b4 yarn build"}),". You can also enter the container, with ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] /bin/ash"}),", and navigate around it with normal Linux commands."]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy has several commands that can be used for various tasks, you can find a list of these either in the ",(0,r.jsx)(n.a,{href:"/docs/developing#project-commands",children:"Developing Docs"}),", or by looking at the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/package.json#l5",children:(0,r.jsx)(n.code,{children:"package.json"})}),". These can be used by running ",(0,r.jsx)(n.code,{children:"yarn [command-name]"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"healthchecks",children:"Healthchecks"}),"\n",(0,r.jsxs)(n.p,{children:["Healthchecks are configured to periodically check that Dashy is up and running correctly on the specified port. By default, the health script is called every 5 minutes, but this can be modified with the ",(0,r.jsx)(n.code,{children:"--health-interval"})," option. You can check the current container health with: ",(0,r.jsx)(n.code,{children:'docker inspect --format "{{json .State.Health }}" [container-id]'}),", and a summary of health status will show up under ",(0,r.jsx)(n.code,{children:"docker ps"}),". You can also manually request the current application status by running ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] yarn health-check"}),". You can disable healthchecks altogether by adding the ",(0,r.jsx)(n.code,{children:"--no-healthcheck"})," flag to your Docker run command."]}),"\n",(0,r.jsxs)(n.p,{children:["To restart unhealthy containers automatically, check out ",(0,r.jsx)(n.a,{href:"https://hub.docker.com/r/willfarrell/autoheal/",children:"Autoheal"}),". This image watches for unhealthy containers, and automatically triggers a restart. (This is a stand in for Docker's ",(0,r.jsx)(n.code,{children:"--exit-on-unhealthy"})," that was proposed, but ",(0,r.jsx)(n.a,{href:"https://github.com/moby/moby/pull/22719",children:"not merged"}),"). There's also ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/deunhealth",children:"Deunhealth"}),", which is super light-weight, and doesn't require network access."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n --name autoheal \\\n --restart=always \\\n -e AUTOHEAL_CONTAINER_LABEL=all \\\n -v /var/run/docker.sock:/var/run/docker.sock \\\n willfarrell/autoheal\n"})}),"\n",(0,r.jsx)(n.h3,{id:"http-healthcheck-endpoint",children:"HTTP Healthcheck Endpoint"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy also exposes an unauthenticated HTTP liveness endpoint at ",(0,r.jsx)(n.code,{children:"/healthz"}),", which returns a ",(0,r.jsx)(n.code,{children:"200"})," with a small JSON body (",(0,r.jsx)(n.code,{children:"status"}),", ",(0,r.jsx)(n.code,{children:"uptime"}),", ",(0,r.jsx)(n.code,{children:"version"}),"). It bypasses auth and SSL redirection so probes keep working regardless of how Dashy is configured."]}),"\n",(0,r.jsx)(n.p,{children:"Useful when fronting Dashy with a load balancer / reverse proxy that needs an HTTP probe for auto-failover, or when running on Kubernetes:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"# Kubernetes\nlivenessProbe:\n httpGet:\n path: /healthz\n port: 8080\n"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'# Traefik (label on the Dashy service)\n- "traefik.http.services.dashy.loadbalancer.healthcheck.path=/healthz"\n- "traefik.http.services.dashy.loadbalancer.healthcheck.interval=30s"\n'})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-caddyfile",children:"# Caddy (reverse_proxy block)\nreverse_proxy dashy:8080 {\n health_uri /healthz\n health_interval 30s\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For Nginx Proxy Manager, set the ",(0,r.jsx)(n.em,{children:"Forward Hostname"})," health-check path to ",(0,r.jsx)(n.code,{children:"/healthz"})," under the proxy host's ",(0,r.jsx)(n.em,{children:"Custom locations"})," / advanced config."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"logs-and-performance",children:"Logs and Performance"}),"\n",(0,r.jsx)(n.h3,{id:"container-logs",children:"Container Logs"}),"\n",(0,r.jsxs)(n.p,{children:["You can view logs for a given Docker container with ",(0,r.jsx)(n.code,{children:"docker logs [container-id]"}),", add the ",(0,r.jsx)(n.code,{children:"--follow"})," flag to stream the logs. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/logging/",children:"Logging Documentation"}),". There's also ",(0,r.jsx)(n.a,{href:"https://dozzle.dev/",children:"Dozzle"}),", a useful tool, that provides a web interface where you can stream and query logs from all your running containers from a single web app."]}),"\n",(0,r.jsx)(n.h3,{id:"container-performance",children:"Container Performance"}),"\n",(0,r.jsxs)(n.p,{children:["You can check the resource usage for your running Docker containers with ",(0,r.jsx)(n.code,{children:"docker stats"})," or ",(0,r.jsx)(n.code,{children:"docker stats [container-id]"}),". For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/stats/",children:"Stats Documentation"}),". There's also ",(0,r.jsx)(n.a,{href:"https://github.com/google/cadvisor",children:"cAdvisor"}),", a useful web app for viewing and analyzing resource usage and performance of all your running containers."]}),"\n",(0,r.jsx)(n.h3,{id:"management-apps",children:"Management Apps"}),"\n",(0,r.jsxs)(n.p,{children:["You can also view logs, resource usage and other info as well as manage your entire Docker workflow in third-party Docker management apps. For example ",(0,r.jsx)(n.a,{href:"https://github.com/portainer/portainer",children:"Portainer"})," an all-in-one open source management web UI for Docker and Kubernetes, or ",(0,r.jsx)(n.a,{href:"https://github.com/jesseduffield/lazydocker",children:"LazyDocker"})," a terminal UI for Docker container management and monitoring."]}),"\n",(0,r.jsx)(n.h3,{id:"advanced-logging-and-monitoring",children:"Advanced Logging and Monitoring"}),"\n",(0,r.jsxs)(n.p,{children:["Docker supports using ",(0,r.jsx)(n.a,{href:"https://prometheus.io/",children:"Prometheus"})," to collect logs, which can then be visualized using a platform like ",(0,r.jsx)(n.a,{href:"https://grafana.com/",children:"Grafana"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/daemon/prometheus/",children:"this guide"}),". If you need to route your logs to a remote syslog, then consider using ",(0,r.jsx)(n.a,{href:"https://github.com/gliderlabs/logspout",children:"logspout"}),". For enterprise-grade instances, there are managed services, that make monitoring container logs and metrics very easy, such as ",(0,r.jsx)(n.a,{href:"https://sematext.com/blog/docker-container-monitoring-with-sematext/",children:"Sematext"})," with ",(0,r.jsx)(n.a,{href:"https://github.com/sematext/logagent-js",children:"Logagent"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"auto-starting-at-system-boot",children:"Auto-Starting at System Boot"}),"\n",(0,r.jsxs)(n.p,{children:["You can use Docker's ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#restart-policies---restart",children:"restart policies"})," to instruct the container to start after a system reboot, or restart after a crash. Just add the ",(0,r.jsx)(n.code,{children:"--restart=always"})," flag to your Docker compose script or Docker run command. For more information, see the docs on ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/start-containers-automatically/",children:"Starting Containers Automatically"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["For Podman, you can use ",(0,r.jsx)(n.code,{children:"systemd"})," to create a service that launches your container, ",(0,r.jsx)(n.a,{href:"https://podman.io/blogs/2018/09/13/systemd.html",children:"the docs"})," explains things further. A similar approach can be used with Docker, if you need to start containers after a reboot, but before any user interaction."]}),"\n",(0,r.jsxs)(n.p,{children:["To restart the container after something within it has crashed, consider using ",(0,r.jsx)(n.a,{href:"https://github.com/willfarrell/docker-autoheal",children:(0,r.jsx)(n.code,{children:"docker-autoheal"})})," by @willfarrell, a service that monitors and restarts unhealthy containers. For more info, see the ",(0,r.jsx)(n.a,{href:"#healthchecks",children:"Healthchecks"})," section above."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"updating",children:"Updating"}),"\n",(0,r.jsx)(n.p,{children:"Dashy is under active development, so to take advantage of the latest features, you may need to update your instance every now and again."}),"\n",(0,r.jsx)(n.h3,{id:"updating-docker-container",children:"Updating Docker Container"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Pull latest image: ",(0,r.jsx)(n.code,{children:"docker pull lissy93/dashy:latest"})]}),"\n",(0,r.jsxs)(n.li,{children:["Kill off existing container\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Find container ID: ",(0,r.jsx)(n.code,{children:"docker ps"})]}),"\n",(0,r.jsxs)(n.li,{children:["Stop container: ",(0,r.jsx)(n.code,{children:"docker stop [container_id]"})]}),"\n",(0,r.jsxs)(n.li,{children:["Remove container: ",(0,r.jsx)(n.code,{children:"docker rm [container_id]"})]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Spin up new container: ",(0,r.jsx)(n.code,{children:"docker run [params] lissy93/dashy"})]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"automatic-docker-updates",children:"Automatic Docker Updates"}),"\n",(0,r.jsxs)(n.p,{children:["You can automate the above process using ",(0,r.jsx)(n.a,{href:"https://github.com/containrrr/watchtower",children:"Watchtower"}),".\nWatchtower will watch for new versions of a given image on Docker Hub, pull down your new image, gracefully shut down your existing container and restart it with the same options that were used when it was deployed initially."]}),"\n",(0,r.jsx)(n.p,{children:"To get started, spin up the watchtower container:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n --name watchtower \\\n -v /var/run/docker.sock:/var/run/docker.sock \\\n containrrr/watchtower\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more information, see the ",(0,r.jsx)(n.a,{href:"https://containrrr.dev/watchtower/",children:"Watchtower Docs"})]}),"\n",(0,r.jsx)(n.h3,{id:"updating-dashy-from-source",children:"Updating Dashy from Source"}),"\n",(0,r.jsxs)(n.p,{children:["Stop your current instance of Dashy, then navigate into the source directory. Pull down the latest code, with ",(0,r.jsx)(n.code,{children:"git pull origin master"}),", then update dependencies with ",(0,r.jsx)(n.code,{children:"yarn"}),", rebuild with ",(0,r.jsx)(n.code,{children:"yarn build"}),", and start the server again with ",(0,r.jsx)(n.code,{children:"yarn start"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"backing-up",children:"Backing Up"}),"\n",(0,r.jsx)(n.h3,{id:"backing-up-containers",children:"Backing Up Containers"}),"\n",(0,r.jsxs)(n.p,{children:["You can make a backup of any running container really easily, using ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/commit/",children:(0,r.jsx)(n.code,{children:"docker commit"})})," and save it with ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/export/",children:(0,r.jsx)(n.code,{children:"docker export"})}),", to do so:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["First find the container ID, you can do this with ",(0,r.jsx)(n.code,{children:"docker container ls"})]}),"\n",(0,r.jsxs)(n.li,{children:["Now to create the snapshot, just run ",(0,r.jsx)(n.code,{children:"docker commit -p [container-id] my-backup"})]}),"\n",(0,r.jsxs)(n.li,{children:["Finally, to save the backup locally, run ",(0,r.jsx)(n.code,{children:"docker save -o ~/dashy-backup.tar my-backup"})]}),"\n",(0,r.jsxs)(n.li,{children:["If you want to push this to a container registry, run ",(0,r.jsx)(n.code,{children:"docker push my-backup:latest"})]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Note that this will not include any data in docker volumes, and the process here is a bit different. Since these files exist on your host system, if you have an existing backup solution implemented, you can incorporate and volume files within that system."}),"\n",(0,r.jsx)(n.h3,{id:"backing-up-volumes",children:"Backing Up Volumes"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/offen/docker-volume-backup",children:"offen/docker-volume-backup"})," is a useful tool for periodic Docker volume backups, to any S3-compatible storage provider. It's run as a light-weight Docker container, and is easy to setup, and also supports GPG-encryption, email notification, and routing away older backups."]}),"\n",(0,r.jsxs)(n.p,{children:["To get started, create a docker-compose similar to the example below, and then start the container. For more info, check out their ",(0,r.jsx)(n.a,{href:"https://github.com/offen/docker-volume-backup",children:"documentation"}),", which is very clear."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'services:\n backup:\n image: offen/docker-volume-backup:latest\n environment:\n BACKUP_CRON_EXPRESSION: "0 * * * *"\n BACKUP_PRUNING_PREFIX: backup-\n BACKUP_RETENTION_DAYS: 7\n AWS_BUCKET_NAME: backup-bucket\n AWS_ACCESS_KEY_ID: AKIAIOSFODNN7EXAMPLE\n AWS_SECRET_ACCESS_KEY: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY\n volumes:\n - data:/backup/my-app-backup:ro\n - /var/run/docker.sock:/var/run/docker.sock:ro\nvolumes:\n data:\n'})}),"\n",(0,r.jsx)(n.p,{children:"It's worth noting that this process can also be done manually, using the following commands:"}),"\n",(0,r.jsx)(n.p,{children:"Backup:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run --rm -v some_volume:/volume -v /tmp:/backup alpine tar -cjf /backup/some_archive.tar.bz2 -C /volume ./\n"})}),"\n",(0,r.jsx)(n.p,{children:"Restore:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:'docker run --rm -v some_volume:/volume -v /tmp:/backup alpine sh -c "rm -rf /volume/* /volume/..?* /volume/.[!.]* ; tar -C /volume/ -xjf /backup/some_archive.tar.bz2"\n'})}),"\n",(0,r.jsx)(n.h3,{id:"dashy-specific-backup",children:"Dashy-Specific Backup"}),"\n",(0,r.jsxs)(n.p,{children:["All configuration and dashboard settings are stored in your ",(0,r.jsx)(n.code,{children:"user-data/conf.yml"})," file. If you provide additional assets (like icons, fonts, themes, etc), these will also live in the ",(0,r.jsx)(n.code,{children:"user-data"})," directory. So to backup all Dashy data, this is the only directory you need to backup."]}),"\n",(0,r.jsxs)(n.p,{children:["When you save config through the UI, Dashy automatically creates a timestamped backup in ",(0,r.jsx)(n.code,{children:"user-data/config-backups/"})," (configurable via the ",(0,r.jsx)(n.code,{children:"BACKUP_DIR"})," env var). If you break your config, check that directory for a recent copy. Backups can be disabled by setting ",(0,r.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS=true"})," (e.g. on read-only filesystems or where permissions don't allow it)."]}),"\n",(0,r.jsx)(n.p,{children:"Since Dashy is open source, there shouldn't be any need to backup the main container."}),"\n",(0,r.jsxs)(n.p,{children:["Dashy also has a built-in cloud backup feature, which is free for personal users, and will let you make and restore fully encrypted backups of your config directly through the UI. To learn more, see the ",(0,r.jsx)(n.a,{href:"/docs/backup-restore",children:"Cloud Backup Docs"})]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"scheduling",children:"Scheduling"}),"\n",(0,r.jsxs)(n.p,{children:["If you need to periodically schedule the running of a given command on Dashy (or any other container), then a useful tool for doing so it ",(0,r.jsx)(n.a,{href:"https://github.com/mcuadros/ofelia",children:"ofelia"}),". This runs as a Docker container and is really useful for things like backups, logging, updating, notifications, etc. Crons are specified using Go's crontab format, and a useful tool for visualizing this is ",(0,r.jsx)(n.a,{href:"https://crontab.guru/",children:"crontab.guru"}),". This can also be done natively with Alpine: ",(0,r.jsx)(n.code,{children:"docker run -it alpine ls /etc/periodic"}),".\nI recommend combining this with ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"healthchecks"})," for easy monitoring of jobs, and failure notifications."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"ssl-certificates",children:"SSL Certificates"}),"\n",(0,r.jsx)(n.p,{children:"Enabling HTTPS with an SSL certificate is recommended, especially if you are hosting Dashy anywhere other than your home. This will ensure that all traffic is encrypted in transit."}),"\n",(0,r.jsx)(n.h3,{id:"auto-ssl",children:"Auto-SSL"}),"\n",(0,r.jsxs)(n.p,{children:["If you are using ",(0,r.jsx)(n.a,{href:"https://nginxproxymanager.com/",children:"NGINX Proxy Manager"}),', then SSL is supported out of the box. Once you\'ve added your proxy host and web address, then set the scheme to HTTPS, then under the SSL Tab select "Request a new SSL certificate" and follow the on-screen instructions.']}),"\n",(0,r.jsxs)(n.p,{children:["If you're hosting Dashy behind Cloudflare, then they offer ",(0,r.jsx)(n.a,{href:"https://www.cloudflare.com/en-gb/learning/ssl/what-is-an-ssl-certificate/",children:"free and easy SSL"}),"- all you need to do is enable it under the SSL/TLS tab. Or if you are using shared hosting, you may find ",(0,r.jsx)(n.a,{href:"https://www.sitepoint.com/a-guide-to-setting-up-lets-encrypt-ssl-on-shared-hosting/",children:"this tutorial"})," helpful."]}),"\n",(0,r.jsx)(n.h3,{id:"getting-a-self-signed-ssl-certificate",children:"Getting a Self-Signed SSL Certificate"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://letsencrypt.org/docs/",children:"Let's Encrypt"})," is a global Certificate Authority, providing free SSL/TLS Domain Validation certificates in order to enable secure HTTPS access to your website. They have good browser/ OS ",(0,r.jsx)(n.a,{href:"https://letsencrypt.org/docs/certificate-compatibility/",children:"compatibility"})," with their ISRG X1 and DST CA X3 root certificates, support ",(0,r.jsx)(n.a,{href:"https://community.letsencrypt.org/t/acme-v2-production-environment-wildcards/55578",children:"Wildcard issuance"})," done via ACMEv2 using the DNS-01 and have ",(0,r.jsx)(n.a,{href:"https://letsencrypt.org/2020/02/19/multi-perspective-validation.html",children:"Multi-Perspective Validation"}),". Let's Encrypt provide ",(0,r.jsx)(n.a,{href:"https://certbot.eff.org/",children:"CertBot"})," an easy app for generating and setting up an SSL certificate."]}),"\n",(0,r.jsxs)(n.p,{children:["This process can be automated, using something like the ",(0,r.jsx)(n.a,{href:"https://github.com/Valian/docker-nginx-auto-ssl",children:"Docker-NGINX-Auto-SSL Container"})," to generate and renew certificates when needed."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're not so comfortable on the command line, then you can use a tool like ",(0,r.jsx)(n.a,{href:"https://www.sslforfree.com/",children:"SSL For Free"})," or ",(0,r.jsx)(n.a,{href:"https://zerossl.com/",children:"ZeroSSL"})," to generate your cert. They also provide step-by-step setup instructions for most platforms."]}),"\n",(0,r.jsx)(n.h3,{id:"passing-a-self-signed-certificate-to-dashy",children:"Passing a Self-Signed Certificate to Dashy"}),"\n",(0,r.jsxs)(n.p,{children:["Once you've generated your SSL cert, you'll need to pass it to Dashy. This can be done by specifying the paths to your public and private keys using the ",(0,r.jsx)(n.code,{children:"SSL_PRIV_KEY_PATH"})," and ",(0,r.jsx)(n.code,{children:"SSL_PUB_KEY_PATH"})," environmental variables. Or if you're using Docker, then just pass public + private SSL keys in under ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-pub.pem"})," and ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-priv.key"})," respectively, e.g:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/my-private-key.key:/etc/ssl/certs/dashy-priv.key:ro \\\n -v ~/my-public-key.pem:/etc/ssl/certs/dashy-pub.pem:ro \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["By default the SSL port is ",(0,r.jsx)(n.code,{children:"443"})," within a Docker container, or ",(0,r.jsx)(n.code,{children:"4001"})," if running on bare metal, but you can override this with the ",(0,r.jsx)(n.code,{children:"SSL_PORT"})," environmental variable."]}),"\n",(0,r.jsxs)(n.p,{children:["Once everything is setup, you can verify your site is secured using a tool like ",(0,r.jsx)(n.a,{href:"https://www.sslchecker.com/sslchecker",children:"SSL Checker"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"authentication",children:"Authentication"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy natively supports secure authentication using KeyCloak. There is also a Simple Auth feature that doesn't require any additional setup. Usage instructions for both, as well as alternative auth methods, has now moved to the ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})})," page."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"network-exposure",children:"Network Exposure"}),"\n",(0,r.jsx)(n.p,{children:"Dashy is designed to run on your local network, behind your firewall. If you only access it from within your home or over a VPN, the defaults are fine."}),"\n",(0,r.jsxs)(n.p,{children:["If you do need to expose Dashy to the internet, you should put it behind a reverse proxy with its own authentication layer (e.g. Authelia, Authentik, Cloudflare Access, or your proxy's built-in auth). Don't rely solely on Dashy's built-in auth for internet-facing instances - it's a convenience feature for private networks, not a hardened perimeter control. See the ",(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})," for setup options."]}),"\n",(0,r.jsxs)(n.p,{children:["When Dashy runs in server mode (the default Docker setup), it exposes several API endpoints for things like status checks, config saving, system info, and a CORS proxy used by widgets. When authentication is enabled (via ",(0,r.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," or ",(0,r.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"}),"/",(0,r.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," env vars), all of these endpoints require valid credentials. Without auth configured, they are open. That's fine for private networks, but not appropriate for public access."]}),"\n",(0,r.jsxs)(n.p,{children:["The CORS proxy (",(0,r.jsx)(n.code,{children:"/cors-proxy"}),") is worth calling out specifically: it forwards requests from the Dashy server to external URLs, so widgets can reach APIs that don't set CORS headers. On a private network this is harmless, but on an internet-exposed instance without auth, it could be abused as an open proxy. Always enable authentication if your instance is reachable from untrusted networks."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"managing-containers-with-docker-compose",children:"Managing Containers with Docker Compose"}),"\n",(0,r.jsxs)(n.p,{children:["When you have a lot of containers, it quickly becomes hard to manage with ",(0,r.jsx)(n.code,{children:"docker run"})," commands. The solution to this is ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/compose/",children:"docker compose"}),", a handy tool for defining all a containers run settings in a single YAML file, and then spinning up that container with a single short command - ",(0,r.jsx)(n.code,{children:"docker compose up"}),". A good example of which can be seen in ",(0,r.jsx)(n.a,{href:"https://github.com/abhilesh/self-hosted_docker_setups",children:"@abhilesh's docker compose collection"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["You can use Dashy's default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:(0,r.jsx)(n.code,{children:"docker-compose.yml"})})," file as a template, and modify it according to your needs."]}),"\n",(0,r.jsx)(n.p,{children:"An example Docker compose, using the default base image from DockerHub, might look something like this:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n container_name: Dashy\n image: lissy93/dashy:latest\n volumes:\n - ./user-data:/app/user-data\n ports:\n - 4000:8080\n environment:\n - BASE_URL=/my-dashboard\n restart: unless-stopped\n healthcheck:\n test: ['CMD', 'node', '/app/services/healthcheck.js']\n interval: 1m30s\n timeout: 10s\n retries: 3\n start_period: 30s\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"passing-in-environmental-variables",children:"Passing in Environmental Variables"}),"\n",(0,r.jsxs)(n.p,{children:["With Docker, you can define environmental variables under the ",(0,r.jsx)(n.code,{children:"environment"})," section of your Docker compose file. Environmental variables are used to configure high-level settings, usually before the config file has been read. For a list of all supported env vars in Dashy, see ",(0,r.jsx)(n.a,{href:"/docs/developing#environmental-variables",children:"the developing docs"}),", or the default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.env",children:(0,r.jsx)(n.code,{children:".env"})})," file."]}),"\n",(0,r.jsxs)(n.p,{children:["A common use case, is to run Dashy under a sub-page, instead of at the root of a URL (e.g. ",(0,r.jsx)(n.code,{children:"https://my-homelab.local/dashy"})," instead of ",(0,r.jsx)(n.code,{children:"https://dashy.my-homelab.local"}),"). In this use-case, you'd specify the ",(0,r.jsx)(n.code,{children:"BASE_URL"})," variable in your compose file."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"environment:\n - BASE_URL=/dashy\n"})}),"\n",(0,r.jsxs)(n.p,{children:["You can also do the same thing with the docker run command, using the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/run/#set-environment-variables--e---env---env-file",children:(0,r.jsx)(n.code,{children:"--env"})})," flag.\nIf you've got many environmental variables, you might find it useful to put them in a ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/compose/env-file/",children:[(0,r.jsx)(n.code,{children:".env"})," file"]}),". Similarly, for Docker run you can use ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/run/#set-environment-variables--e---env---env-file",children:(0,r.jsx)(n.code,{children:"--env-file"})})," if you'd like to pass in a file containing all your environmental variables."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"setting-headers",children:"Setting Headers"}),"\n",(0,r.jsxs)(n.p,{children:["Any external requests made to a different origin (app/ service under a different domain) will be blocked if the correct headers are not specified. This is known as ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"})," (CORS) and is a security feature built into modern browsers."]}),"\n",(0,r.jsx)(n.p,{children:"If you see a CORS error in your console, this can be easily fixed by setting the correct headers. This is not a bug with Dashy, so please don't raise it as a bug!"}),"\n",(0,r.jsx)(n.h3,{id:"example-headers",children:"Example Headers"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#caddy",children:"Caddy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nginx",children:"NGINX"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#traefik",children:"Tr\xe6f\u026ak"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#haproxy",children:"HAProxy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#apache",children:"Apache"})}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsxs)(n.em,{children:["The following section briefly outlines how you can set headers for common web proxies/ servers. More info can be found in the documentation for the proxy that you are using, or in the ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"MDN Docs"}),"."]})}),"\n",(0,r.jsx)(n.p,{children:"These examples are using:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, but depending on what type of content you are enabling, this will vary. For example, to allow a site to be loaded in an iframe (for the modal or workspace views) you would use ",(0,r.jsx)(n.code,{children:"X-Frame-Options"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:["The domain root (",(0,r.jsx)(n.code,{children:"/"}),"), if your're hosting from a sub-page, replace that with your path."]}),"\n",(0,r.jsxs)(n.li,{children:["A wildcard (",(0,r.jsx)(n.code,{children:"*"}),"), which would allow access from traffic on any domain, this is discouraged, and you should replace it with the URL where you are hosting Dashy. Note that for requests that transport sensitive info, like credentials (e.g. Keycloak login), the wildcard is ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS#requests_with_credentials",children:"disallowed all together"})," and will be blocked."]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"caddy",children:"Caddy"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/header",children:["Caddy ",(0,r.jsx)(n.code,{children:"header"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"headers / {\n Access-Control-Allow-Origin *\n}\n"})}),"\n",(0,r.jsx)(n.h4,{id:"nginx",children:"NGINX"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_headers_module.html",children:["NGINX ",(0,r.jsx)(n.code,{children:"ngx_http_headers_module"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"location / {\n add_header Access-Control-Allow-Origin *;\n}\n"})}),"\n",(0,r.jsx)(n.p,{children:"Note this can also be done through the UI, using NGINX Proxy Manager."}),"\n",(0,r.jsx)(n.h4,{id:"traefik",children:"Traefik"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsx)(n.a,{href:"https://doc.traefik.io/traefik/middlewares/http/headers/#cors-headers",children:"Tr\xe6f\u026ak CORS headers docs"})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'labels:\n - "traefik.http.middlewares.testheader.headers.accesscontrolallowmethods=GET,OPTIONS,PUT"\n - "traefik.http.middlewares.testheader.headers.accesscontrolalloworiginlist=https://foo.bar.org,https://example.org"\n - "traefik.http.middlewares.testheader.headers.accesscontrolmaxage=100"\n - "traefik.http.middlewares.testheader.headers.addvaryheader=true"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"haproxy",children:"HAProxy"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.haproxy.com/documentation/hapee/latest/traffic-routing/rewrites/rewrite-responses/",children:"HAProxy Rewrite Response Docs"})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"/\n http-response add-header Access-Control-Allow-Origin *\n"})}),"\n",(0,r.jsx)(n.h4,{id:"apache",children:"Apache"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://httpd.apache.org/docs/current/mod/mod_headers.html",children:["Apache ",(0,r.jsx)(n.code,{children:"mode_headers"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'Header always set Access-Control-Allow-Origin "*"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"squid",children:"Squid"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"http://www2.gr.squid-cache.org/Doc/config/request_header_access/",children:["Squid ",(0,r.jsx)(n.code,{children:"request_header_access"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"request_header_access Authorization allow all\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"remote-access",children:"Remote Access"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#wireguard",children:"WireGuard"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#reverse-ssh-tunnel",children:"Reverse SSH Tunnel"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tcp-tunnel",children:"TCP Tunnel"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"wireguard",children:"WireGuard"}),"\n",(0,r.jsxs)(n.p,{children:["Using a VPN is one of the easiest ways to provide secure, full access to your local network from remote locations. ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/",children:"WireGuard"})," is a reasonably new open source VPN protocol, that was designed with ease of use, performance and security in mind. Unlike OpenVPN, it doesn't need to recreate the tunnel whenever connection is dropped, and it's also much easier to setup, using shared keys instead."]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Install Wireguard"})," - See the ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/install/",children:"Install Docs"})," for download links + instructions\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["On Debian-based systems, it's ",(0,r.jsx)(n.code,{children:"sudo apt install wireguard"})]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Generate a Private Key"})," - Run ",(0,r.jsx)(n.code,{children:"wg genkey"})," on the Wireguard server, and copy it to somewhere safe for later"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Create Server Config"})," - Open or create a file at ",(0,r.jsx)(n.code,{children:"/etc/wireguard/wg0.conf"})," and under ",(0,r.jsx)(n.code,{children:"[Interface]"})," add the following (see example below):\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Address"})," - as a subnet of all desired IPs"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PrivateKey"})," - that you just generated"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"ListenPort"})," - Default is ",(0,r.jsx)(n.code,{children:"51820"}),", but can be anything"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Get Client App"})," - Download the ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/install/",children:"WG client app"})," for your platform (Linux, Windows, MacOS, Android or iOS are all supported)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Create new Client Tunnel"})," - On your client app, there should be an option to create a new tunnel, when doing so a client private key will be generated (but if not, use the ",(0,r.jsx)(n.code,{children:"wg genkey"})," command again), and keep it somewhere safe. A public key will also be generated, and this will go in our saver config"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Add Clients to Server Config"})," - Head back to your ",(0,r.jsx)(n.code,{children:"wg0.conf"})," file on the server, create a ",(0,r.jsx)(n.code,{children:"[Peer]"})," section, and populate the following info\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"AllowedIPs"})," - List of IP address inside the subnet, the client should have access to"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PublicKey"})," - The public key for the client you just generated"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Start the Server"})," - You can now start the WG server, using: ",(0,r.jsx)(n.code,{children:"wg-quick up wg0"})," on your server"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Finish Client Setup"})," - Head back to your client device, and edit the config file, leave the private key as is, and add the following fields:\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PublicKey"})," - The public key of the server"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Address"})," - This should match the ",(0,r.jsx)(n.code,{children:"AllowedIPs"})," section on the servers config file"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"DNS"})," - The DNS server that'll be used when accessing the network through the VPN"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Endpoint"})," - The hostname or IP + Port where your WG server is running (you may need to forward this in your firewall's settings)"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Done"})," - Your clients should now be able to connect to your WG server :) Depending on your networks firewall rules, you may need to port forward the address of your WG server"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-server-config",children:(0,r.jsx)(n.strong,{children:"Example Server Config"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-ini",children:"# Server file\n[Interface]\n# Which networks does my interface belong to? Notice: /24 and /64\nAddress = 10.5.0.1/24, 2001:470:xxxx:xxxx::1/64\nPrivateKey = xxx\nListenPort = 51820\n\n# Peer 1\n[Peer]\nPublicKey = xxx\n# Which source IPs can I expect from that peer? Notice: /32 and /128\nAllowedIps = 10.5.0.35/32, 2001:470:xxxx:xxxx::746f:786f/128\n\n# Peer 2\n[Peer]\nPublicKey = xxx\n# Which source IPs can I expect from that peer? This one has a LAN which can\n# access hosts/jails without NAT.\n# Peer 2 has a single IP address inside the VPN: it's 10.5.0.25/32\nAllowedIps = 10.5.0.25/32,10.21.10.0/24,10.21.20.0/24,10.21.30.0/24,10.31.0.0/24,2001:470:xxxx:xxxx::ca:571e/128\n"})}),"\n",(0,r.jsx)(n.h4,{id:"example-client-config",children:(0,r.jsx)(n.strong,{children:"Example Client Config"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-ini",children:"[Interface]\n# Which networks does my interface belong to? Notice: /24 and /64\nAddress = 10.5.0.35/24, 2001:470:xxxx:xxxx::746f:786f/64\nPrivateKey = xxx\n\n# Server\n[Peer]\nPublicKey = xxx\n# I want to route everything through the server, both IPv4 and IPv6. All IPs are\n# thus available through the Server, and I can expect packets from any IP to\n# come from that peer.\nAllowedIPs = 0.0.0.0/0, ::0/0\n# Where is the server on the internet? This is a public address. The port\n# (:51820) is the same as ListenPort in the [Interface] of the Server file above\nEndpoint = 1.2.3.4:51820\n# Usually, clients are behind NAT. to keep the connection running, keep alive.\nPersistentKeepalive = 15\n"})}),"\n",(0,r.jsxs)(n.p,{children:["A useful tool for getting WG setup is ",(0,r.jsx)(n.a,{href:"https://github.com/trailofbits/algo",children:"Algo"}),". It includes scripts and docs which cover almost all devices, platforms and clients, and has best practices implemented, and security features enabled. All of this is better explained in ",(0,r.jsx)(n.a,{href:"https://blog.trailofbits.com/2016/12/12/meet-algo-the-vpn-that-works/",children:"this blog post"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"reverse-ssh-tunnel",children:"Reverse SSH Tunnel"}),"\n",(0,r.jsxs)(n.p,{children:["SSH (or ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Secure_Shell",children:"Secure Shell"}),") is a secure tunnel that allows you to connect to a remote host. Unlike the VPN methods, an SSH connection does not require an intermediary, and will not be affected by your IP changing. However it only allows you to access a single service at a time. SSH was really designed for terminal access, but because of the latter mentioned benefits it's useful to setup, as a fallback option."]}),"\n",(0,r.jsx)(n.p,{children:"Directly SSH'ing into your home, would require you to open a port (usually 22), which would be terrible for security, and is not recommended. However a reverse SSH connection is initiated from inside your network. Once the connection is established, the port is redirected, allowing you to use the established connection to SSH into your home network."}),"\n",(0,r.jsxs)(n.p,{children:["The issue you've probably spotted, is that most public, corporate, and institutional networks will block SSH connections. To overcome this, you'd have to establish a server outside of your homelab that your homelab's device could SSH into to establish the reverse SSH connection. You can then connect to that remote server (the ",(0,r.jsx)(n.em,{children:"mothership"}),"), which in turn connects to your home network."]}),"\n",(0,r.jsxs)(n.p,{children:["Now all of this is starting to sound like quite a lot of work, but this is where services like ",(0,r.jsx)(n.a,{href:"https://remote.it/",children:"remot3.it"})," come in. They maintain the intermediary mothership server, and create the tunnel service for you. It's free for personal use, secure and easy. There are several similar services, such as ",(0,r.jsx)(n.a,{href:"https://remoteiot.com/",children:"RemoteIoT"}),", or you could create your own on a cloud VPS (see ",(0,r.jsx)(n.a,{href:"https://gist.github.com/nileshtrivedi/4c615e8d3c1bf053b0d31176b9e69e42",children:"this tutorial"})," for more info on that)."]}),"\n",(0,r.jsxs)(n.p,{children:["Before getting started, you'll need to head over to ",(0,r.jsx)(n.a,{href:"https://app.remote.it/auth/#/sign-up",children:"Remote.it"})," and create an account."]}),"\n",(0,r.jsx)(n.p,{children:"Then setup your local device:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["If you haven't already done so, you'll need to enable and configure SSH.\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["This is out-of-scope of this article, but I've explained it in detail in ",(0,r.jsx)(n.a,{href:"https://notes.aliciasykes.com/22798/my-server-setup#configure-ssh",children:"this post"}),"."]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Download the Remote.it install script from their ",(0,r.jsx)(n.a,{href:"https://github.com/remoteit/installer",children:"GitHub"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"curl -LkO https://raw.githubusercontent.com/remoteit/installer/master/scripts/auto-install.sh"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Make it executable, with ",(0,r.jsx)(n.code,{children:"chmod +x ./auto-install.sh"}),", and then run it with ",(0,r.jsx)(n.code,{children:"sudo ./auto-install.sh"})]}),"\n",(0,r.jsxs)(n.li,{children:["Finally, configure your device, by running ",(0,r.jsx)(n.code,{children:"sudo connectd_installer"})," and following the on-screen instructions"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"And when you're ready to connect to it:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Login to ",(0,r.jsx)(n.a,{href:"https://app.remote.it/",children:"app.remote.it"}),", and select the name of your device"]}),"\n",(0,r.jsx)(n.li,{children:"You should see a list of running services, click SSH"}),"\n",(0,r.jsx)(n.li,{children:"You'll then be presented with some SSH credentials that you can now use to securely connect to your home, via the Remote.it servers"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Done :)"}),"\n",(0,r.jsx)(n.h3,{id:"tcp-tunnel",children:"TCP Tunnel"}),"\n",(0,r.jsxs)(n.p,{children:["If you're running Dashy on your local network, behind a firewall, but need to temporarily share it with someone external, this can be achieved quickly and securely using ",(0,r.jsx)(n.a,{href:"https://ngrok.com/",children:"Ngrok"}),". It's basically a super slick, encrypted TCP tunnel that provides an internet-accessible address that anyone use to access your local service, from anywhere."]}),"\n",(0,r.jsxs)(n.p,{children:["To get started, ",(0,r.jsx)(n.a,{href:"https://ngrok.com/download",children:"Download"})," and install Ngrok for your system, then just run ",(0,r.jsx)(n.code,{children:"ngrok http [port]"})," (replace the port with the http port where Dashy is running, e.g. 8080). When ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-local-https",children:"using https"}),", specify the full local url/ ip including the protocol."]}),"\n",(0,r.jsxs)(n.p,{children:["Some Ngrok features require you to be authenticated, you can ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/signup",children:"create a free account"})," and generate a token in ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/auth/your-authtoken",children:"your dashboard"}),", then run ",(0,r.jsx)(n.code,{children:"ngrok authtoken [token]"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["It's recommended to use authentication for any publicly accessible service. Dashy has an ",(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Auth"})," feature built in, but an even easier method it to use the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-auth",children:(0,r.jsx)(n.code,{children:"-auth"})})," switch. E.g. ",(0,r.jsx)(n.code,{children:'ngrok http -auth="username:password123" 8080'})]}),"\n",(0,r.jsxs)(n.p,{children:["By default, your web app is assigned a randomly generated ngrok domain, but you can also use your own custom domain. Under the ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/endpoints/domains",children:"Domains Tab"})," of your Ngrok dashboard, add your domain, and follow the CNAME instructions. You can now use your domain, with the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-custom-domains",children:(0,r.jsx)(n.code,{children:"-hostname"})})," switch, e.g. ",(0,r.jsx)(n.code,{children:"ngrok http -region=us -hostname=dashy.example.com 8080"}),". If you don't have your own domain name, you can instead use a custom sub-domain (e.g. ",(0,r.jsx)(n.code,{children:"alicia-dashy.ngrok.io"}),"), using the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#custom-subdomain-names",children:(0,r.jsx)(n.code,{children:"-subdomain"})})," switch."]}),"\n",(0,r.jsxs)(n.p,{children:["To integrate this into your docker-compose, take a look at the ",(0,r.jsx)(n.a,{href:"https://github.com/gtriggiano/ngrok-tunnel",children:"gtriggiano/ngrok-tunnel"})," container."]}),"\n",(0,r.jsxs)(n.p,{children:["There's so much more you can do with Ngrok, such as exposing a directory as a file browser, using websockets, relaying requests, rewriting headers, inspecting traffic, TLS and TCP tunnels and lots more. All or which is explained in ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs",children:"the Documentation"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["It's worth noting that Ngrok isn't the only option here, other options include: ",(0,r.jsx)(n.a,{href:"https://github.com/fatedier/frp",children:"FRP"}),", ",(0,r.jsx)(n.a,{href:"https://inlets.dev",children:"Inlets"}),", ",(0,r.jsx)(n.a,{href:"https://localtunnel.me/",children:"Local Tunnel"}),", ",(0,r.jsx)(n.a,{href:"https://tailscale.com/",children:"TailScale"}),", etc. Check out ",(0,r.jsx)(n.a,{href:"https://github.com/anderspitman/awesome-tunneling",children:"Awesome Tunneling"})," for a list of alternatives."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"custom-domain",children:"Custom Domain"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#using-dns",children:"Using DNS"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#using-nginx",children:"Using NGINX"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"using-dns",children:"Using DNS"}),"\n",(0,r.jsxs)(n.p,{children:["For locally running services, a domain can be set up directly in the DNS records. This method is really quick and easy, and doesn't require you to purchase an actual domain. Just update your networks DNS resolver, to point your desired URL to the local IP where Dashy (or any other app) is running. For example, a line in your hosts file might look something like: ",(0,r.jsx)(n.code,{children:"192.168.0.2 dashy.homelab.local"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're using Pi-Hole, a similar thing can be done in the ",(0,r.jsx)(n.code,{children:"/etc/dnsmasq.d/03-custom-dns.conf"})," file, add a line like: ",(0,r.jsx)(n.code,{children:"address=/dashy.example.com/192.168.2.0"})," for each of your services."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're running OPNSense/ PfSense, then this can be done through the UI with Unbound, it's explained nicely in ",(0,r.jsx)(n.a,{href:"https://homenetworkguy.com/how-to/use-custom-domain-name-in-internal-network/",children:"this article"}),", by Dustin Casto."]}),"\n",(0,r.jsx)(n.h3,{id:"using-nginx",children:"Using NGINX"}),"\n",(0,r.jsx)(n.p,{children:"If you're using NGINX, then you can use your own domain name, with a config similar to the below example."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"upstream dashy {\n server 127.0.0.1:32400;\n}\n\nserver {\n listen 8080;\n server_name dashy.mydomain.com;\n\n # Setup SSL\n ssl_certificate /var/www/mydomain/sslcert.pem;\n ssl_certificate_key /var/www/mydomain/sslkey.pem;\n ssl_protocols TLSv1.2 TLSv1.3;\n ssl_ciphers 'EECDH+AESGCM:EDH+AESGCM:AES256+EECDH:AES256+EDH';\n ssl_session_timeout 5m;\n ssl_prefer_server_ciphers on;\n\n location / {\n proxy_pass http://dashy;\n proxy_redirect off;\n proxy_buffering off;\n proxy_set_header host $host;\n proxy_set_header X-Real-IP $remote_addr;\n proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;\n proxy_next_upstream error timeout invalid_header http_500 http_502 http_503 http_504;\n }\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Similarly, a basic ",(0,r.jsx)(n.code,{children:"Caddyfile"})," might look like:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"dashy.example.com {\n reverse_proxy / nginx:8080\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more info, ",(0,r.jsx)(n.a,{href:"https://thehomelab.wiki/books/dns-reverse-proxy/page/create-domain-records-to-point-to-your-home-server-on-cloudflare-using-nginx-progy-manager",children:"this guide"})," on Setting up Domains with NGINX Proxy Manager and CloudFlare may be useful."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"container-security",children:"Container Security"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#keep-docker-up-to-date",children:"Keep Docker Up-To-Date"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#set-resource-quotas",children:"Set Resource Quotas"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dont-run-as-root",children:"Don't Run as Root"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#specify-a-user",children:"Specify a User"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#limit-capabilities",children:"Limit Capabilities"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#prevent-new-privileges-being-added",children:"Prevent new Privileges being Added"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disable-inter-container-communication",children:"Disable Inter-Container Communication"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dont-expose-the-docker-daemon-socket",children:"Don't Expose the Docker Daemon Socket"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#use-read-only-volumes",children:"Use Read-Only Volumes"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#set-the-logging-level",children:"Set the Logging Level"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#verify-image-before-pulling",children:"Verify Image before Pulling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#specify-the-tag",children:"Specify the Tag"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-security-scanning",children:"Container Security Scanning"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#registry-security",children:"Registry Security"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#security-modules",children:"Security Modules"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"keep-docker-up-to-date",children:"Keep Docker Up-To-Date"}),"\n",(0,r.jsxs)(n.p,{children:["To prevent known container escape vulnerabilities, which typically end in escalating to root/administrator privileges, patching Docker Engine and Docker Machine is crucial. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/install/",children:"Docker Installation Docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"set-resource-quotas",children:"Set Resource Quotas"}),"\n",(0,r.jsxs)(n.p,{children:["Docker enables you to limit resource consumption (CPU, memory, disk) on a per-container basis. This not only enhances system performance, but also prevents a compromised container from consuming a large amount of resources, in order to disrupt service or perform malicious activities. To learn more, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Resource Constraints Docs"})]}),"\n",(0,r.jsxs)(n.p,{children:["For example, to run Dashy with max of 1GB ram, and max of 50% of 1 CP core:\n",(0,r.jsx)(n.code,{children:'docker run -d -p 8080:8080 --cpus=".5" --memory="1024m" lissy93/dashy:latest'})]}),"\n",(0,r.jsx)(n.h3,{id:"dont-run-as-root",children:"Don't Run as Root"}),"\n",(0,r.jsxs)(n.p,{children:["Running Docker commands with ",(0,r.jsx)(n.code,{children:"sudo"})," gives the container more host-level access than it needs. You should run Docker as a non-root host user instead."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're facing permission issues on Debian-based systems when running Docker commands without ",(0,r.jsx)(n.code,{children:"sudo"}),", you may need to add your user to the Docker group. First create the group: ",(0,r.jsx)(n.code,{children:"sudo groupadd docker"}),", then add your (non-root) user: ",(0,r.jsx)(n.code,{children:"sudo usermod \u2212aG docker [my-username]"}),", finally ",(0,r.jsx)(n.code,{children:"newgrp docker"})," to refresh."]}),"\n",(0,r.jsx)(n.h3,{id:"specify-a-user",children:"Specify a User"}),"\n",(0,r.jsxs)(n.p,{children:["For containers in general, running as an unprivileged user is one of the best ways to prevent privilege escalation attacks. You can specify a user with the ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/engine/reference/run/#user",children:[(0,r.jsx)(n.code,{children:"--user"})," param"]}),", using the user ID (",(0,r.jsx)(n.code,{children:"UID"}),") from ",(0,r.jsx)(n.code,{children:"id -u"})," and group ID (",(0,r.jsx)(n.code,{children:"GID"}),") from ",(0,r.jsx)(n.code,{children:"id -g"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Note for Dashy:"})," If you use features that write to disk (saving config through the UI), the process needs write access to ",(0,r.jsx)(n.code,{children:"/app/user-data/"}),". Since the default image creates these directories as root, running with ",(0,r.jsx)(n.code,{children:"--user"})," will cause those features to fail with permission errors unless you also fix ownership of the mounted volumes. If you only use Dashy in read-only mode, running as a non-root user works fine:"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.code,{children:"docker run --user 1000:1000 -p 8080:8080 lissy93/dashy"})}),"\n",(0,r.jsx)(n.p,{children:"Or with Docker Compose, using an environmental variable:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy\n user: ${CURRENT_UID}\n ports: [ 4000:8080 ]\n"})}),"\n",(0,r.jsxs)(n.p,{children:["And then to set the variable, and start the container, run: ",(0,r.jsx)(n.code,{children:"CURRENT_UID=$(id -u):$(id -g) docker-compose up"})]}),"\n",(0,r.jsx)(n.h3,{id:"limit-capabilities",children:"Limit capabilities"}),"\n",(0,r.jsxs)(n.p,{children:["Docker containers run with a subset of ",(0,r.jsx)(n.a,{href:"https://man7.org/linux/man-pages/man7/capabilities.7.html",children:"Linux Kernal's Capabilities"})," by default. It's good practice to drop privilege permissions that are not needed for any given container."]}),"\n",(0,r.jsxs)(n.p,{children:["With Docker run, you can use the ",(0,r.jsx)(n.code,{children:"--cap-drop"})," flag to remove capabilities, you can also use ",(0,r.jsx)(n.code,{children:"--cap-drop=all"})," and then define just the required permissions using the ",(0,r.jsx)(n.code,{children:"--cap-add"})," option. For a list of available capabilities, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#runtime-privilege-and-linux-capabilities",children:"Privilege Capabilities Docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Note that dropping privileges and capabilities on runtime is not fool-proof, and often any leftover privileges can be used to re-escalate, see ",(0,r.jsx)(n.a,{href:"https://wiki.sei.cmu.edu/confluence/display/c/POS36-C.+Observe+correct+revocation+order+while+relinquishing+privileges",children:"POS36-C"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Here's an example using docker-compose, removing privileges that are not required for Dashy to run:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy\n ports: [ 4000:8080 ]\n cap_drop:\n - ALL\n cap_add:\n - CHOWN\n - SETGID\n - SETUID\n - DAC_OVERRIDE\n - NET_BIND_SERVICE\n"})}),"\n",(0,r.jsx)(n.h3,{id:"prevent-new-privileges-being-added",children:"Prevent new Privileges being Added"}),"\n",(0,r.jsxs)(n.p,{children:["To prevent processes inside the container from getting additional privileges, pass in the ",(0,r.jsx)(n.code,{children:"--security-opt=no-new-privileges:true"})," option to the Docker run command (see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#security-configuration",children:"docs"}),")."]}),"\n",(0,r.jsxs)(n.p,{children:["Run Command:\n",(0,r.jsx)(n.code,{children:"docker run --security-opt=no-new-privileges:true -p 8080:8080 lissy93/dashy"})]}),"\n",(0,r.jsx)(n.p,{children:"Docker Compose"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"security_opt:\n- no-new-privileges:true\n"})}),"\n",(0,r.jsx)(n.h3,{id:"disable-inter-container-communication",children:"Disable Inter-Container Communication"}),"\n",(0,r.jsxs)(n.p,{children:["By default Docker containers can talk to each other (using ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/config/containers/container-networking/",children:[(0,r.jsx)(n.code,{children:"docker0"})," bridged network"]}),"). If you don't need this capability, then it should be disabled. This can be done with the ",(0,r.jsx)(n.code,{children:"--icc=false"})," in your run command. You can learn more about how to facilitate secure communication between containers in the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/compose/networking/",children:"Compose Networking docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"dont-expose-the-docker-daemon-socket",children:"Don't Expose the Docker Daemon Socket"}),"\n",(0,r.jsxs)(n.p,{children:["Docker socket ",(0,r.jsx)(n.code,{children:"/var/run/docker.sock"})," is the UNIX socket that Docker is listening to. This is the primary entry point for the Docker API. The owner of this socket is root. Giving someone access to it is equivalent to giving unrestricted root access to your host."]}),"\n",(0,r.jsxs)(n.p,{children:["You should ",(0,r.jsx)(n.strong,{children:"not"})," enable TCP Docker daemon socket (",(0,r.jsx)(n.code,{children:"-H tcp://0.0.0.0:XXX"}),"), as doing so exposes un-encrypted and unauthenticated direct access to the Docker daemon, and if the host is connected to the internet, the daemon on your computer can be used by anyone from the public internet- which is bad. If you need TCP, you should ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/dockerd/#daemon-socket-option",children:"see the docs"})," to understand how to do this more securely.\nSimilarly, never expose ",(0,r.jsx)(n.code,{children:"/var/run/docker.sock"})," to other containers as a volume, as it can be exploited."]}),"\n",(0,r.jsx)(n.h3,{id:"use-read-only-volumes",children:"Use Read-Only Volumes"}),"\n",(0,r.jsxs)(n.p,{children:["You can specify that a volume should be read-only by appending ",(0,r.jsx)(n.code,{children:":ro"})," to the ",(0,r.jsx)(n.code,{children:"-v"})," switch. If you don't need the in-app config editor, mount your ",(0,r.jsx)(n.code,{children:"user-data"})," read-only:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/dashy-data:/app/user-data:ro \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If you do want config changes from the UI to persist back to disk, leave the mount writable. You can also use ",(0,r.jsx)(n.code,{children:"--read-only"})," to make the whole container filesystem read-only, but in that case UI-driven config edits will not be saved."]}),"\n",(0,r.jsx)(n.h3,{id:"set-the-logging-level",children:"Set the Logging Level"}),"\n",(0,r.jsxs)(n.p,{children:["Logging is important, as it enables you to review events in the future, and in the case of a compromise this will let get an idea of what may have happened. The default log level is ",(0,r.jsx)(n.code,{children:"INFO"}),", and this is also the recommendation, use ",(0,r.jsx)(n.code,{children:"--log-level info"})," to ensure this is set."]}),"\n",(0,r.jsx)(n.h3,{id:"verify-image-before-pulling",children:"Verify Image before Pulling"}),"\n",(0,r.jsx)(n.p,{children:"Only use trusted images, from verified/ official sources. If an app is open source, it is more likely to be safe, as anyone can verify the code. There are also tools available for scanning containers,"}),"\n",(0,r.jsx)(n.p,{children:"Unless otherwise configured, containers can communicate among each other, so running one bad image may lead to other areas of your setup being compromised. Docker images typically contain both original code, as well as up-stream packages, and even if that image has come from a trusted source, the up-stream packages it includes may not have."}),"\n",(0,r.jsx)(n.h3,{id:"specify-the-tag",children:"Specify the Tag"}),"\n",(0,r.jsxs)(n.p,{children:["Using fixed tags (as opposed to ",(0,r.jsx)(n.code,{children:":latest"})," ) will ensure immutability, meaning the base image will not change between builds. Note that for Dashy, the app is being actively developed, new features, bug fixes and general improvements are merged each week, and if you use a fixed version you will not enjoy these benefits. So it's up to you weather you would prefer a stable and reproducible environment, or the latest features and enhancements."]}),"\n",(0,r.jsx)(n.h3,{id:"container-security-scanning",children:"Container Security Scanning"}),"\n",(0,r.jsxs)(n.p,{children:["It's helpful to be aware of any potential security issues in any of the Docker images you are using. You can run a quick scan using Snyk on any image to output known vulnerabilities using ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/scan/",children:"Docker scan"}),", e.g: ",(0,r.jsx)(n.code,{children:"docker scan lissy93/dashy:latest"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["A similar product is ",(0,r.jsx)(n.a,{href:"https://github.com/aquasecurity/trivy",children:"Trivy"}),", which is free an open source. First install it (with your package manager), then to scan an image, just run: ",(0,r.jsx)(n.code,{children:"trivy image lissy93/dashy:latest"})]}),"\n",(0,r.jsxs)(n.p,{children:["For larger systems, RedHat ",(0,r.jsx)(n.a,{href:"https://www.redhat.com/en/topics/containers/what-is-clair",children:"Clair"})," is an app for parsing image contents and reporting on any found vulnerabilities. You run it locally in a container, and configure it with YAML. It can be integrated with Red Hat Quay, to show results on a dashboard. Most of these use static analysis to find potential issues, and scan included packages for any known security vulnerabilities."]}),"\n",(0,r.jsx)(n.h3,{id:"registry-security",children:"Registry Security"}),"\n",(0,r.jsxs)(n.p,{children:["Although over-kill for most users, you could run your own registry locally which would give you ultimate control over all images, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/registry/deploying/",children:"Deploying a Registry Docs"})," for more info. Another option is ",(0,r.jsx)(n.a,{href:"https://docker-docs.netlify.app/ee/dtr/",children:"Docker Trusted Registry"}),", it's great for enterprise applications, it sits behind your firewall, running on a swarm managed by Docker Universal Control Plane, and lets you securely store and manage your Docker images, mitigating the risk of breaches from the internet."]}),"\n",(0,r.jsx)(n.h3,{id:"security-modules",children:"Security Modules"}),"\n",(0,r.jsx)(n.p,{children:"Docker supports several modules that let you write your own security profiles."}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://www.apparmor.net/",children:"AppArmor"}),"is a kernel module that proactively protects the operating system and applications from external or internal threats, by enabling you to restrict programs' capabilities with per-program profiles. You can specify either a security policy by name, or by file path with the ",(0,r.jsx)(n.code,{children:"apparmor"})," flag in docker run. Learn more about writing profiles, ",(0,r.jsx)(n.a,{href:"https://gitlab.com/apparmor/apparmor/-/wikis/QuickProfileLanguage",children:"here"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Seccomp",children:"Seccomp"})," (Secure Computing Mode) is a sandboxing facility in the Linux kernel that acts like a firewall for system calls (syscalls). It uses Berkeley Packet Filter (BPF) rules to filter syscalls and control how they are handled. These filters can significantly limit a containers access to the Docker Host's Linux kernel - especially for simple containers/applications. It requires a Linux-based Docker host, with secomp enabled, and you can check for this by running ",(0,r.jsx)(n.code,{children:"docker info | grep seccomp"}),". A great resource for learning more about this is ",(0,r.jsx)(n.a,{href:"https://training.play-with-docker.com/security-seccomp/",children:"DockerLabs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"web-server-configuration",children:"Web Server Configuration"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"The following section only applies if you are not using Docker, and would like to use your own web server"})}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy ships with a pre-configured Node.js server, in ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/server.js",children:(0,r.jsx)(n.code,{children:"server.js"})})," which serves up the contents of the ",(0,r.jsx)(n.code,{children:"./dist"})," directory on a given port. You can start the server by running ",(0,r.jsx)(n.code,{children:"node server"}),". Note that the app must have been build (run ",(0,r.jsx)(n.code,{children:"yarn build"}),"), and you need ",(0,r.jsx)(n.a,{href:"https://nodejs.org",children:"Node.js"})," installed."]}),"\n",(0,r.jsxs)(n.p,{children:["If you wish to run Dashy from a sub page (e.g. ",(0,r.jsx)(n.code,{children:"example.com/dashy"}),"), then just set the ",(0,r.jsx)(n.code,{children:"BASE_URL"})," environmental variable to that page name (in this example, ",(0,r.jsx)(n.code,{children:"/dashy"}),"), before building the app, and the path to all assets will then resolve to the new path, instead of ",(0,r.jsx)(n.code,{children:"./"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"However, since Dashy is just a static web application, it can be served with whatever server you like. The following section outlines how you can configure a web server."}),"\n",(0,r.jsxs)(n.p,{children:["Note, that if you choose not to use ",(0,r.jsx)(n.code,{children:"server.js"})," to serve up the app, you will loose access to the following features:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Loading page, while the app is building"}),"\n",(0,r.jsx)(n.li,{children:"Writing config file to disk from the UI"}),"\n",(0,r.jsx)(n.li,{children:"Website status indicators, and ping checks"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Example Configs"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nginx",children:"NGINX"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#apache",children:"Apache"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#caddy",children:"Caddy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#firebase-hosting",children:"Firebase"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpanel",children:"cPanel"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"nginx-1",children:"NGINX"}),"\n",(0,r.jsxs)(n.p,{children:["Create a new file in ",(0,r.jsx)(n.code,{children:"/etc/nginx/sites-enabled/dashy"})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"server {\n\tlisten 8080;\n\tlisten [::]:8080;\n\n\troot /var/www/dashy/html;\n\tindex index.html;\n\n\tserver_name your-domain.com www.your-domain.com;\n\n\tlocation / {\n\t\ttry_files $uri $uri/ =404;\n\t}\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["To use HTML5 history mode (the default - controlled via the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE"})," build-time env var), replace the inside of the location block with: ",(0,r.jsx)(n.code,{children:"try_files $uri $uri/ /index.html;"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Then upload the build contents of Dashy's dist directory to that location.\nFor example: ",(0,r.jsx)(n.code,{children:"scp -r ./dist/* [username]@[server_ip]:/var/www/dashy/html"})]}),"\n",(0,r.jsx)(n.h3,{id:"apache-1",children:"Apache"}),"\n",(0,r.jsxs)(n.p,{children:["Copy Dashy's dist folder to your apache server, ",(0,r.jsx)(n.code,{children:"sudo cp -r ./dashy/dist /var/www/html/dashy"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["In your Apache config, ",(0,r.jsx)(n.code,{children:"/etc/apche2/apache2.conf"})," add:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"\n\tOptions Indexes FollowSymLinks\n\tAllowOverride All\n\tRequire all granted\n\n\n\n RewriteEngine On\n RewriteBase /\n RewriteRule ^index\\.html$ - [L]\n RewriteCond %{REQUEST_FILENAME} !-f\n RewriteCond %{REQUEST_FILENAME} !-d\n RewriteRule . /index.html [L]\n\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Add a ",(0,r.jsx)(n.code,{children:".htaccess"})," file within ",(0,r.jsx)(n.code,{children:"/var/www/html/dashy/.htaccess"}),", and add:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Options -MultiViews\nRewriteEngine On\nRewriteCond %{REQUEST_FILENAME} !-f\nRewriteRule ^ index.html [QSA,L]\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then restart Apache, with ",(0,r.jsx)(n.code,{children:"sudo systemctl restart apache2"})]}),"\n",(0,r.jsx)(n.h3,{id:"caddy-1",children:"Caddy"}),"\n",(0,r.jsx)(n.p,{children:"Caddy v2"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"try_files {path} /\n"})}),"\n",(0,r.jsx)(n.p,{children:"Caddy v1"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"rewrite {\n regexp .*\n to {path} /\n}\n"})}),"\n",(0,r.jsx)(n.h3,{id:"firebase-hosting",children:"Firebase Hosting"}),"\n",(0,r.jsxs)(n.p,{children:["Create a file names ",(0,r.jsx)(n.code,{children:"firebase.json"}),", and populate it with something similar to:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'{\n "hosting": {\n "public": "dist",\n "rewrites": [\n {\n "source": "**",\n "destination": "/index.html"\n }\n ]\n }\n}\n'})}),"\n",(0,r.jsx)(n.h3,{id:"cpanel",children:"cPanel"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsx)(n.li,{children:"Login to your WHM"}),"\n",(0,r.jsx)(n.li,{children:"Open 'Feature Manager' on the left sidebar"}),"\n",(0,r.jsx)(n.li,{children:"Under 'Manage feature list', click 'Edit'"}),"\n",(0,r.jsx)(n.li,{children:"Find 'Application manager' in the list, enable it and hit 'Save'"}),"\n",(0,r.jsx)(n.li,{children:"Log into your users cPanel account, and under 'Software' find 'Application Manager'"}),"\n",(0,r.jsx)(n.li,{children:"Click 'Register Application', fill in the form using the path that Dashy is located, and choose a domain, and hit 'Save'"}),"\n",(0,r.jsx)(n.li,{children:"The application should now show up in the list, click 'Ensure dependencies', and move the toggle switch to 'Enabled'"}),"\n",(0,r.jsx)(n.li,{children:"If you need to change the port, click 'Add environmental variable', give it the name 'PORT', choose a port number and press 'Save'."}),"\n",(0,r.jsx)(n.li,{children:"Dashy should now be running at your selected path an on a given port"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"running-a-modified-version-of-the-app",children:"Running a Modified Version of the App"}),"\n",(0,r.jsx)(n.p,{children:"If you'd like to make any code changes to the app, and deploy your modified version, this section briefly explains how."}),"\n",(0,r.jsxs)(n.p,{children:["The first step is to fork the project on GitHub, and clone it to your local system. Next, install the dependencies (",(0,r.jsx)(n.code,{children:"yarn"}),"), and start the development server (",(0,r.jsx)(n.code,{children:"yarn dev"}),") and visit ",(0,r.jsx)(n.code,{children:"localhost:8080"})," in your browser. You can then make changes to the codebase, and see the live app update in real-time. Once you've finished, running ",(0,r.jsx)(n.code,{children:"yarn build"})," will build the app for production, and output the assets into ",(0,r.jsx)(n.code,{children:"./dist"})," which can then be deployed using a web server, CDN or the built-in Node server with ",(0,r.jsx)(n.code,{children:"yarn start"}),". For more info on all of this, take a look at the ",(0,r.jsx)(n.a,{href:"/docs/developing",children:"Developing Docs"}),". To build your own Docker container from the modified app, see ",(0,r.jsx)(n.a,{href:"#building-your-own-container",children:"Building your Own Container"})]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"building-your-own-container",children:"Building your Own Container"}),"\n",(0,r.jsx)(n.p,{children:"Similar to above, you'll first need to fork and clone Dashy to your local system, and then install dependencies."}),"\n",(0,r.jsxs)(n.p,{children:["Then, either use Dashy's default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/Dockerfile",children:(0,r.jsx)(n.code,{children:"Dockerfile"})})," as is, or modify it according to your needs."]}),"\n",(0,r.jsxs)(n.p,{children:["To build and deploy locally, first build the app with: ",(0,r.jsx)(n.code,{children:"docker build -t dashy ."}),", and then start the app with ",(0,r.jsx)(n.code,{children:"docker run -p 8080:8080 --name my-dashboard dashy"}),". Or modify the ",(0,r.jsx)(n.code,{children:"docker-compose.yml"})," file, replacing ",(0,r.jsx)(n.code,{children:"image: lissy93/dashy"})," with ",(0,r.jsx)(n.code,{children:"build: ."})," and run ",(0,r.jsx)(n.code,{children:"docker compose up"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Your container should now be running, and will appear in the list when you run ",(0,r.jsx)(n.code,{children:"docker container ls \u2013a"}),". If you'd like to enter the container, run ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] /bin/ash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["You may wish to upload your image to a container registry for easier access. Note that if you choose to do this on a public registry, please name your container something other than just 'dashy', to avoid confusion with the official image.\nYou can push your build image, by running: ",(0,r.jsx)(n.code,{children:"docker push ghcr.io/OWNER/IMAGE_NAME:latest"}),". You will first need to authenticate, this can be done by running ",(0,r.jsx)(n.code,{children:"echo $CR_PAT | docker login ghcr.io -u USERNAME --password-stdin"}),", where ",(0,r.jsx)(n.code,{children:"CR_PAT"})," is an environmental variable containing a token generated from your GitHub account. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/packages/working-with-a-github-packages-registry/working-with-the-container-registry",children:"Container Registry Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>a});var i=s(6540);const r={},o=i.createContext(r);function t(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9695],{6644(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>a,default:()=>h,frontMatter:()=>t,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"management","title":"App Management","description":"The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains.","source":"@site/docs/management.md","sourceDirName":".","slug":"/management","permalink":"/docs/management","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/management.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Configuring","permalink":"/docs/configuring"},"next":{"title":"Troubleshooting","permalink":"/docs/troubleshooting"}}');var r=s(4848),o=s(8453);const t={},a="App Management",c={},l=[{value:"Contents",id:"contents",level:2},{value:"Providing Assets",id:"providing-assets",level:2},{value:"File Ownership and Permissions",id:"file-ownership-and-permissions",level:2},{value:"Running Commands",id:"running-commands",level:2},{value:"Healthchecks",id:"healthchecks",level:2},{value:"HTTP Healthcheck Endpoint",id:"http-healthcheck-endpoint",level:3},{value:"Logs and Performance",id:"logs-and-performance",level:2},{value:"Container Logs",id:"container-logs",level:3},{value:"Container Performance",id:"container-performance",level:3},{value:"Management Apps",id:"management-apps",level:3},{value:"Advanced Logging and Monitoring",id:"advanced-logging-and-monitoring",level:3},{value:"Auto-Starting at System Boot",id:"auto-starting-at-system-boot",level:2},{value:"Updating",id:"updating",level:2},{value:"Updating Docker Container",id:"updating-docker-container",level:3},{value:"Automatic Docker Updates",id:"automatic-docker-updates",level:3},{value:"Updating Dashy from Source",id:"updating-dashy-from-source",level:3},{value:"Backing Up",id:"backing-up",level:2},{value:"Backing Up Containers",id:"backing-up-containers",level:3},{value:"Backing Up Volumes",id:"backing-up-volumes",level:3},{value:"Dashy-Specific Backup",id:"dashy-specific-backup",level:3},{value:"Scheduling",id:"scheduling",level:2},{value:"SSL Certificates",id:"ssl-certificates",level:2},{value:"Auto-SSL",id:"auto-ssl",level:3},{value:"Getting a Self-Signed SSL Certificate",id:"getting-a-self-signed-ssl-certificate",level:3},{value:"Passing a Self-Signed Certificate to Dashy",id:"passing-a-self-signed-certificate-to-dashy",level:3},{value:"Authentication",id:"authentication",level:2},{value:"Network Exposure",id:"network-exposure",level:2},{value:"Managing Containers with Docker Compose",id:"managing-containers-with-docker-compose",level:2},{value:"Passing in Environmental Variables",id:"passing-in-environmental-variables",level:2},{value:"Setting Headers",id:"setting-headers",level:2},{value:"Example Headers",id:"example-headers",level:3},{value:"Caddy",id:"caddy",level:4},{value:"NGINX",id:"nginx",level:4},{value:"Traefik",id:"traefik",level:4},{value:"HAProxy",id:"haproxy",level:4},{value:"Apache",id:"apache",level:4},{value:"Squid",id:"squid",level:4},{value:"Remote Access",id:"remote-access",level:2},{value:"WireGuard",id:"wireguard",level:3},{value:"Example Server Config",id:"example-server-config",level:4},{value:"Example Client Config",id:"example-client-config",level:4},{value:"Reverse SSH Tunnel",id:"reverse-ssh-tunnel",level:3},{value:"TCP Tunnel",id:"tcp-tunnel",level:3},{value:"Custom Domain",id:"custom-domain",level:2},{value:"Using DNS",id:"using-dns",level:3},{value:"Using NGINX",id:"using-nginx",level:3},{value:"Container Security",id:"container-security",level:2},{value:"Keep Docker Up-To-Date",id:"keep-docker-up-to-date",level:3},{value:"Set Resource Quotas",id:"set-resource-quotas",level:3},{value:"Don't Run as Root",id:"dont-run-as-root",level:3},{value:"Specify a User",id:"specify-a-user",level:3},{value:"Limit capabilities",id:"limit-capabilities",level:3},{value:"Prevent new Privileges being Added",id:"prevent-new-privileges-being-added",level:3},{value:"Disable Inter-Container Communication",id:"disable-inter-container-communication",level:3},{value:"Don't Expose the Docker Daemon Socket",id:"dont-expose-the-docker-daemon-socket",level:3},{value:"Use Read-Only Volumes",id:"use-read-only-volumes",level:3},{value:"Set the Logging Level",id:"set-the-logging-level",level:3},{value:"Verify Image before Pulling",id:"verify-image-before-pulling",level:3},{value:"Specify the Tag",id:"specify-the-tag",level:3},{value:"Container Security Scanning",id:"container-security-scanning",level:3},{value:"Registry Security",id:"registry-security",level:3},{value:"Security Modules",id:"security-modules",level:3},{value:"Web Server Configuration",id:"web-server-configuration",level:2},{value:"NGINX",id:"nginx-1",level:3},{value:"Apache",id:"apache-1",level:3},{value:"Caddy",id:"caddy-1",level:3},{value:"Firebase Hosting",id:"firebase-hosting",level:3},{value:"cPanel",id:"cpanel",level:3},{value:"Running a Modified Version of the App",id:"running-a-modified-version-of-the-app",level:2},{value:"Building your Own Container",id:"building-your-own-container",level:2}];function d(e){const n={a:"a",blockquote:"blockquote",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"app-management",children:"App Management"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains."})}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#providing-assets",children:"Providing Assets"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#file-ownership-and-permissions",children:"File Ownership and Permissions"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#running-commands",children:"Running Commands"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthchecks",children:"Healthchecks"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#logs-and-performance",children:"Logs and Performance"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#auto-starting-at-system-boot",children:"Auto-Starting at Boot"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#updating",children:"Updating"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#backing-up",children:"Backing Up"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#scheduling",children:"Scheduling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ssl-certificates",children:"SSL Certificates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#authentication",children:"Authentication"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-exposure",children:"Network Exposure"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#managing-containers-with-docker-compose",children:"Managing with Compose"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#passing-in-environmental-variables",children:"Environmental Variables"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#setting-headers",children:"Setting Headers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#remote-access",children:"Remote Access"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#custom-domain",children:"Custom Domain"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-security",children:"Securing Containers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#web-server-configuration",children:"Web Server Configuration"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#running-a-modified-version-of-the-app",children:"Running a Modified App"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#building-your-own-container",children:"Building your Own Container"})}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"providing-assets",children:"Providing Assets"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy reads everything for your dashboard from a single host directory mounted into the container at ",(0,r.jsx)(n.code,{children:"/app/user-data"}),". This is done with a ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/storage/volumes/",children:"Docker volume"}),", e.g. ",(0,r.jsx)(n.code,{children:"-v /path/to/your/user-data:/app/user-data"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["The directory must contain a ",(0,r.jsx)(n.code,{children:"conf.yml"}),". It can also contain anything else you want served from the web root: sub-config files for additional pages, item icons, favicon, fonts, custom CSS, manifest, and so on. Any file placed there is reachable at ",(0,r.jsx)(n.code,{children:"/"})," in the browser, overriding files of the same name in the bundled ",(0,r.jsx)(n.code,{children:"public/"})," defaults."]}),"\n",(0,r.jsx)(n.p,{children:"Typical contents:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"conf.yml"})," - Main config (required)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"*.yml"})," / ",(0,r.jsx)(n.code,{children:"*.yaml"})," - Sub-config files for ",(0,r.jsx)(n.a,{href:"/docs/pages-and-sections#multi-page-support",children:"multi-page support"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"item-icons/"})," - Local icons referenced by ",(0,r.jsx)(n.code,{children:"icon: ./item-icons/foo.png"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"favicon.ico"}),", ",(0,r.jsx)(n.code,{children:"manifest.json"}),", ",(0,r.jsx)(n.code,{children:"robots.txt"})," - Override the bundled defaults"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"fonts/"}),", ",(0,r.jsx)(n.code,{children:"widget-resources/"})," - Custom fonts or assets used by widgets"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"file-ownership-and-permissions",children:"File Ownership and Permissions"}),"\n",(0,r.jsxs)(n.p,{children:["Inside the container, Dashy runs as a non-root user with uid/gid 1000 (the built-in ",(0,r.jsx)(n.code,{children:"node"})," user from the Node base image). This is fine for the vast majority of installs, since the first user on a default Linux or macOS box is also uid 1000, so a bind-mounted ",(0,r.jsx)(n.code,{children:"user-data"})," directory is read/writable straight away."]}),"\n",(0,r.jsx)(n.p,{children:"It only gets fiddly if your host uid happens to be something else (NAS systems and multi-user servers being the usual culprits). In that case, config saves from the UI will fail with a permission error, because the container's uid 1000 doesn't own your directory."}),"\n",(0,r.jsx)(n.p,{children:"There are two ways to sort it. Pick whichever is less hassle:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Run the container as your own user."})," The cleanest option, since you don't touch host file ownership."]}),"\n",(0,r.jsxs)(n.p,{children:["On ",(0,r.jsx)(n.code,{children:"docker run"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d -p 8080:8080 \\\n --user $(id -u):$(id -g) \\\n -v /path/to/user-data:/app/user-data \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["In compose, uncomment the ",(0,r.jsx)(n.code,{children:"user:"})," line under the service and set it:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'user: "1001:1001" # whatever `id -u` and `id -g` give you\n'})}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Hand the directory to uid 1000."})," Quicker if you don't mind changing host ownership:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"sudo chown -R 1000:1000 /path/to/user-data\n"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["Note that if you run the container as root (e.g. ",(0,r.jsx)(n.code,{children:"--user 0:0"}),"), Dashy will still work, but you lose the security benefit of a non-root container. Don't do that unless you've a good reason."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"running-commands",children:"Running Commands"}),"\n",(0,r.jsxs)(n.p,{children:["If you're running an app in Docker, then commands will need to be passed to the container to be executed. This can be done by preceding each command with ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id]"}),", where container ID can be found by running ",(0,r.jsx)(n.code,{children:"docker ps"}),". For example ",(0,r.jsx)(n.code,{children:"docker exec -it 26c156c467b4 yarn build"}),". You can also enter the container, with ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] /bin/ash"}),", and navigate around it with normal Linux commands."]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy has several commands that can be used for various tasks, you can find a list of these either in the ",(0,r.jsx)(n.a,{href:"/docs/developing#project-commands",children:"Developing Docs"}),", or by looking at the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/package.json#l5",children:(0,r.jsx)(n.code,{children:"package.json"})}),". These can be used by running ",(0,r.jsx)(n.code,{children:"yarn [command-name]"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"healthchecks",children:"Healthchecks"}),"\n",(0,r.jsxs)(n.p,{children:["Healthchecks are configured to periodically check that Dashy is up and running correctly on the specified port. By default, the health script is called every 5 minutes, but this can be modified with the ",(0,r.jsx)(n.code,{children:"--health-interval"})," option. You can check the current container health with: ",(0,r.jsx)(n.code,{children:'docker inspect --format "{{json .State.Health }}" [container-id]'}),", and a summary of health status will show up under ",(0,r.jsx)(n.code,{children:"docker ps"}),". You can also manually request the current application status by running ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] yarn health-check"}),". You can disable healthchecks altogether by adding the ",(0,r.jsx)(n.code,{children:"--no-healthcheck"})," flag to your Docker run command."]}),"\n",(0,r.jsxs)(n.p,{children:["To restart unhealthy containers automatically, check out ",(0,r.jsx)(n.a,{href:"https://hub.docker.com/r/willfarrell/autoheal/",children:"Autoheal"}),". This image watches for unhealthy containers, and automatically triggers a restart. (This is a stand in for Docker's ",(0,r.jsx)(n.code,{children:"--exit-on-unhealthy"})," that was proposed, but ",(0,r.jsx)(n.a,{href:"https://github.com/moby/moby/pull/22719",children:"not merged"}),"). There's also ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/deunhealth",children:"Deunhealth"}),", which is super light-weight, and doesn't require network access."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n --name autoheal \\\n --restart=always \\\n -e AUTOHEAL_CONTAINER_LABEL=all \\\n -v /var/run/docker.sock:/var/run/docker.sock \\\n willfarrell/autoheal\n"})}),"\n",(0,r.jsx)(n.h3,{id:"http-healthcheck-endpoint",children:"HTTP Healthcheck Endpoint"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy also exposes an unauthenticated HTTP liveness endpoint at ",(0,r.jsx)(n.code,{children:"/healthz"}),", which returns a ",(0,r.jsx)(n.code,{children:"200"})," with a small JSON body (",(0,r.jsx)(n.code,{children:"status"}),", ",(0,r.jsx)(n.code,{children:"uptime"}),", ",(0,r.jsx)(n.code,{children:"version"}),"). It bypasses auth and SSL redirection so probes keep working regardless of how Dashy is configured."]}),"\n",(0,r.jsx)(n.p,{children:"Useful when fronting Dashy with a load balancer / reverse proxy that needs an HTTP probe for auto-failover, or when running on Kubernetes:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"# Kubernetes\nlivenessProbe:\n httpGet:\n path: /healthz\n port: 8080\n"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'# Traefik (label on the Dashy service)\n- "traefik.http.services.dashy.loadbalancer.healthcheck.path=/healthz"\n- "traefik.http.services.dashy.loadbalancer.healthcheck.interval=30s"\n'})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-caddyfile",children:"# Caddy (reverse_proxy block)\nreverse_proxy dashy:8080 {\n health_uri /healthz\n health_interval 30s\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For Nginx Proxy Manager, set the ",(0,r.jsx)(n.em,{children:"Forward Hostname"})," health-check path to ",(0,r.jsx)(n.code,{children:"/healthz"})," under the proxy host's ",(0,r.jsx)(n.em,{children:"Custom locations"})," / advanced config."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"logs-and-performance",children:"Logs and Performance"}),"\n",(0,r.jsx)(n.h3,{id:"container-logs",children:"Container Logs"}),"\n",(0,r.jsxs)(n.p,{children:["You can view logs for a given Docker container with ",(0,r.jsx)(n.code,{children:"docker logs [container-id]"}),", add the ",(0,r.jsx)(n.code,{children:"--follow"})," flag to stream the logs. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/logging/",children:"Logging Documentation"}),". There's also ",(0,r.jsx)(n.a,{href:"https://dozzle.dev/",children:"Dozzle"}),", a useful tool, that provides a web interface where you can stream and query logs from all your running containers from a single web app."]}),"\n",(0,r.jsx)(n.h3,{id:"container-performance",children:"Container Performance"}),"\n",(0,r.jsxs)(n.p,{children:["You can check the resource usage for your running Docker containers with ",(0,r.jsx)(n.code,{children:"docker stats"})," or ",(0,r.jsx)(n.code,{children:"docker stats [container-id]"}),". For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/stats/",children:"Stats Documentation"}),". There's also ",(0,r.jsx)(n.a,{href:"https://github.com/google/cadvisor",children:"cAdvisor"}),", a useful web app for viewing and analyzing resource usage and performance of all your running containers."]}),"\n",(0,r.jsx)(n.h3,{id:"management-apps",children:"Management Apps"}),"\n",(0,r.jsxs)(n.p,{children:["You can also view logs, resource usage and other info as well as manage your entire Docker workflow in third-party Docker management apps. For example ",(0,r.jsx)(n.a,{href:"https://github.com/portainer/portainer",children:"Portainer"})," an all-in-one open source management web UI for Docker and Kubernetes, or ",(0,r.jsx)(n.a,{href:"https://github.com/jesseduffield/lazydocker",children:"LazyDocker"})," a terminal UI for Docker container management and monitoring."]}),"\n",(0,r.jsx)(n.h3,{id:"advanced-logging-and-monitoring",children:"Advanced Logging and Monitoring"}),"\n",(0,r.jsxs)(n.p,{children:["Docker supports using ",(0,r.jsx)(n.a,{href:"https://prometheus.io/",children:"Prometheus"})," to collect logs, which can then be visualized using a platform like ",(0,r.jsx)(n.a,{href:"https://grafana.com/",children:"Grafana"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/daemon/prometheus/",children:"this guide"}),". If you need to route your logs to a remote syslog, then consider using ",(0,r.jsx)(n.a,{href:"https://github.com/gliderlabs/logspout",children:"logspout"}),". For enterprise-grade instances, there are managed services, that make monitoring container logs and metrics very easy, such as ",(0,r.jsx)(n.a,{href:"https://sematext.com/blog/docker-container-monitoring-with-sematext/",children:"Sematext"})," with ",(0,r.jsx)(n.a,{href:"https://github.com/sematext/logagent-js",children:"Logagent"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"auto-starting-at-system-boot",children:"Auto-Starting at System Boot"}),"\n",(0,r.jsxs)(n.p,{children:["You can use Docker's ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#restart-policies---restart",children:"restart policies"})," to instruct the container to start after a system reboot, or restart after a crash. Just add the ",(0,r.jsx)(n.code,{children:"--restart=always"})," flag to your Docker compose script or Docker run command. For more information, see the docs on ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/start-containers-automatically/",children:"Starting Containers Automatically"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["For Podman, you can use ",(0,r.jsx)(n.code,{children:"systemd"})," to create a service that launches your container, ",(0,r.jsx)(n.a,{href:"https://podman.io/blogs/2018/09/13/systemd.html",children:"the docs"})," explains things further. A similar approach can be used with Docker, if you need to start containers after a reboot, but before any user interaction."]}),"\n",(0,r.jsxs)(n.p,{children:["To restart the container after something within it has crashed, consider using ",(0,r.jsx)(n.a,{href:"https://github.com/willfarrell/docker-autoheal",children:(0,r.jsx)(n.code,{children:"docker-autoheal"})})," by @willfarrell, a service that monitors and restarts unhealthy containers. For more info, see the ",(0,r.jsx)(n.a,{href:"#healthchecks",children:"Healthchecks"})," section above."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"updating",children:"Updating"}),"\n",(0,r.jsx)(n.p,{children:"Dashy is under active development, so to take advantage of the latest features, you may need to update your instance every now and again."}),"\n",(0,r.jsx)(n.h3,{id:"updating-docker-container",children:"Updating Docker Container"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Pull latest image: ",(0,r.jsx)(n.code,{children:"docker pull lissy93/dashy:latest"})]}),"\n",(0,r.jsxs)(n.li,{children:["Kill off existing container\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Find container ID: ",(0,r.jsx)(n.code,{children:"docker ps"})]}),"\n",(0,r.jsxs)(n.li,{children:["Stop container: ",(0,r.jsx)(n.code,{children:"docker stop [container_id]"})]}),"\n",(0,r.jsxs)(n.li,{children:["Remove container: ",(0,r.jsx)(n.code,{children:"docker rm [container_id]"})]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Spin up new container: ",(0,r.jsx)(n.code,{children:"docker run [params] lissy93/dashy"})]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"automatic-docker-updates",children:"Automatic Docker Updates"}),"\n",(0,r.jsxs)(n.p,{children:["You can automate the above process using ",(0,r.jsx)(n.a,{href:"https://github.com/containrrr/watchtower",children:"Watchtower"}),".\nWatchtower will watch for new versions of a given image on Docker Hub, pull down your new image, gracefully shut down your existing container and restart it with the same options that were used when it was deployed initially."]}),"\n",(0,r.jsx)(n.p,{children:"To get started, spin up the watchtower container:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n --name watchtower \\\n -v /var/run/docker.sock:/var/run/docker.sock \\\n containrrr/watchtower\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more information, see the ",(0,r.jsx)(n.a,{href:"https://containrrr.dev/watchtower/",children:"Watchtower Docs"})]}),"\n",(0,r.jsx)(n.h3,{id:"updating-dashy-from-source",children:"Updating Dashy from Source"}),"\n",(0,r.jsxs)(n.p,{children:["Stop your current instance of Dashy, then navigate into the source directory. Pull down the latest code, with ",(0,r.jsx)(n.code,{children:"git pull origin master"}),", then update dependencies with ",(0,r.jsx)(n.code,{children:"yarn"}),", rebuild with ",(0,r.jsx)(n.code,{children:"yarn build"}),", and start the server again with ",(0,r.jsx)(n.code,{children:"yarn start"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"backing-up",children:"Backing Up"}),"\n",(0,r.jsx)(n.h3,{id:"backing-up-containers",children:"Backing Up Containers"}),"\n",(0,r.jsxs)(n.p,{children:["You can make a backup of any running container really easily, using ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/commit/",children:(0,r.jsx)(n.code,{children:"docker commit"})})," and save it with ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/export/",children:(0,r.jsx)(n.code,{children:"docker export"})}),", to do so:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["First find the container ID, you can do this with ",(0,r.jsx)(n.code,{children:"docker container ls"})]}),"\n",(0,r.jsxs)(n.li,{children:["Now to create the snapshot, just run ",(0,r.jsx)(n.code,{children:"docker commit -p [container-id] my-backup"})]}),"\n",(0,r.jsxs)(n.li,{children:["Finally, to save the backup locally, run ",(0,r.jsx)(n.code,{children:"docker save -o ~/dashy-backup.tar my-backup"})]}),"\n",(0,r.jsxs)(n.li,{children:["If you want to push this to a container registry, run ",(0,r.jsx)(n.code,{children:"docker push my-backup:latest"})]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Note that this will not include any data in docker volumes, and the process here is a bit different. Since these files exist on your host system, if you have an existing backup solution implemented, you can incorporate and volume files within that system."}),"\n",(0,r.jsx)(n.h3,{id:"backing-up-volumes",children:"Backing Up Volumes"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/offen/docker-volume-backup",children:"offen/docker-volume-backup"})," is a useful tool for periodic Docker volume backups, to any S3-compatible storage provider. It's run as a light-weight Docker container, and is easy to setup, and also supports GPG-encryption, email notification, and routing away older backups."]}),"\n",(0,r.jsxs)(n.p,{children:["To get started, create a docker-compose similar to the example below, and then start the container. For more info, check out their ",(0,r.jsx)(n.a,{href:"https://github.com/offen/docker-volume-backup",children:"documentation"}),", which is very clear."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'services:\n backup:\n image: offen/docker-volume-backup:latest\n environment:\n BACKUP_CRON_EXPRESSION: "0 * * * *"\n BACKUP_PRUNING_PREFIX: backup-\n BACKUP_RETENTION_DAYS: 7\n AWS_BUCKET_NAME: backup-bucket\n AWS_ACCESS_KEY_ID: AKIAIOSFODNN7EXAMPLE\n AWS_SECRET_ACCESS_KEY: wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY\n volumes:\n - data:/backup/my-app-backup:ro\n - /var/run/docker.sock:/var/run/docker.sock:ro\nvolumes:\n data:\n'})}),"\n",(0,r.jsx)(n.p,{children:"It's worth noting that this process can also be done manually, using the following commands:"}),"\n",(0,r.jsx)(n.p,{children:"Backup:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run --rm -v some_volume:/volume -v /tmp:/backup alpine tar -cjf /backup/some_archive.tar.bz2 -C /volume ./\n"})}),"\n",(0,r.jsx)(n.p,{children:"Restore:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:'docker run --rm -v some_volume:/volume -v /tmp:/backup alpine sh -c "rm -rf /volume/* /volume/..?* /volume/.[!.]* ; tar -C /volume/ -xjf /backup/some_archive.tar.bz2"\n'})}),"\n",(0,r.jsx)(n.h3,{id:"dashy-specific-backup",children:"Dashy-Specific Backup"}),"\n",(0,r.jsxs)(n.p,{children:["All configuration and dashboard settings are stored in your ",(0,r.jsx)(n.code,{children:"user-data/conf.yml"})," file. If you provide additional assets (like icons, fonts, themes, etc), these will also live in the ",(0,r.jsx)(n.code,{children:"user-data"})," directory. So to backup all Dashy data, this is the only directory you need to backup."]}),"\n",(0,r.jsxs)(n.p,{children:["When you save config through the UI, Dashy automatically creates a timestamped backup in ",(0,r.jsx)(n.code,{children:"user-data/config-backups/"})," (configurable via the ",(0,r.jsx)(n.code,{children:"BACKUP_DIR"})," env var). If you break your config, check that directory for a recent copy. Backups can be disabled by setting ",(0,r.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS=true"})," (e.g. on read-only filesystems or where permissions don't allow it)."]}),"\n",(0,r.jsx)(n.p,{children:"Since Dashy is open source, there shouldn't be any need to backup the main container."}),"\n",(0,r.jsxs)(n.p,{children:["Dashy also has a built-in cloud backup feature, which is free for personal users, and will let you make and restore fully encrypted backups of your config directly through the UI. To learn more, see the ",(0,r.jsx)(n.a,{href:"/docs/backup-restore",children:"Cloud Backup Docs"})]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"scheduling",children:"Scheduling"}),"\n",(0,r.jsxs)(n.p,{children:["If you need to periodically schedule the running of a given command on Dashy (or any other container), then a useful tool for doing so it ",(0,r.jsx)(n.a,{href:"https://github.com/mcuadros/ofelia",children:"ofelia"}),". This runs as a Docker container and is really useful for things like backups, logging, updating, notifications, etc. Crons are specified using Go's crontab format, and a useful tool for visualizing this is ",(0,r.jsx)(n.a,{href:"https://crontab.guru/",children:"crontab.guru"}),". This can also be done natively with Alpine: ",(0,r.jsx)(n.code,{children:"docker run -it alpine ls /etc/periodic"}),".\nI recommend combining this with ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"healthchecks"})," for easy monitoring of jobs, and failure notifications."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"ssl-certificates",children:"SSL Certificates"}),"\n",(0,r.jsx)(n.p,{children:"Enabling HTTPS with an SSL certificate is recommended, especially if you are hosting Dashy anywhere other than your home. This will ensure that all traffic is encrypted in transit."}),"\n",(0,r.jsx)(n.h3,{id:"auto-ssl",children:"Auto-SSL"}),"\n",(0,r.jsxs)(n.p,{children:["If you are using ",(0,r.jsx)(n.a,{href:"https://nginxproxymanager.com/",children:"NGINX Proxy Manager"}),', then SSL is supported out of the box. Once you\'ve added your proxy host and web address, then set the scheme to HTTPS, then under the SSL Tab select "Request a new SSL certificate" and follow the on-screen instructions.']}),"\n",(0,r.jsxs)(n.p,{children:["If you're hosting Dashy behind Cloudflare, then they offer ",(0,r.jsx)(n.a,{href:"https://www.cloudflare.com/en-gb/learning/ssl/what-is-an-ssl-certificate/",children:"free and easy SSL"}),"- all you need to do is enable it under the SSL/TLS tab. Or if you are using shared hosting, you may find ",(0,r.jsx)(n.a,{href:"https://www.sitepoint.com/a-guide-to-setting-up-lets-encrypt-ssl-on-shared-hosting/",children:"this tutorial"})," helpful."]}),"\n",(0,r.jsx)(n.h3,{id:"getting-a-self-signed-ssl-certificate",children:"Getting a Self-Signed SSL Certificate"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://letsencrypt.org/docs/",children:"Let's Encrypt"})," is a global Certificate Authority, providing free SSL/TLS Domain Validation certificates in order to enable secure HTTPS access to your website. They have good browser/ OS ",(0,r.jsx)(n.a,{href:"https://letsencrypt.org/docs/certificate-compatibility/",children:"compatibility"})," with their ISRG X1 and DST CA X3 root certificates, support ",(0,r.jsx)(n.a,{href:"https://community.letsencrypt.org/t/acme-v2-production-environment-wildcards/55578",children:"Wildcard issuance"})," done via ACMEv2 using the DNS-01 and have ",(0,r.jsx)(n.a,{href:"https://letsencrypt.org/2020/02/19/multi-perspective-validation.html",children:"Multi-Perspective Validation"}),". Let's Encrypt provide ",(0,r.jsx)(n.a,{href:"https://certbot.eff.org/",children:"CertBot"})," an easy app for generating and setting up an SSL certificate."]}),"\n",(0,r.jsxs)(n.p,{children:["This process can be automated, using something like the ",(0,r.jsx)(n.a,{href:"https://github.com/Valian/docker-nginx-auto-ssl",children:"Docker-NGINX-Auto-SSL Container"})," to generate and renew certificates when needed."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're not so comfortable on the command line, then you can use a tool like ",(0,r.jsx)(n.a,{href:"https://www.sslforfree.com/",children:"SSL For Free"})," or ",(0,r.jsx)(n.a,{href:"https://zerossl.com/",children:"ZeroSSL"})," to generate your cert. They also provide step-by-step setup instructions for most platforms."]}),"\n",(0,r.jsx)(n.h3,{id:"passing-a-self-signed-certificate-to-dashy",children:"Passing a Self-Signed Certificate to Dashy"}),"\n",(0,r.jsxs)(n.p,{children:["Once you've generated your SSL cert, you'll need to pass it to Dashy. This can be done by specifying the paths to your public and private keys using the ",(0,r.jsx)(n.code,{children:"SSL_PRIV_KEY_PATH"})," and ",(0,r.jsx)(n.code,{children:"SSL_PUB_KEY_PATH"})," environmental variables. Or if you're using Docker, then just pass public + private SSL keys in under ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-pub.pem"})," and ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-priv.key"})," respectively, e.g:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/my-private-key.key:/etc/ssl/certs/dashy-priv.key:ro \\\n -v ~/my-public-key.pem:/etc/ssl/certs/dashy-pub.pem:ro \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["By default the SSL port is ",(0,r.jsx)(n.code,{children:"443"})," within a Docker container, or ",(0,r.jsx)(n.code,{children:"4001"})," if running on bare metal, but you can override this with the ",(0,r.jsx)(n.code,{children:"SSL_PORT"})," environmental variable."]}),"\n",(0,r.jsxs)(n.p,{children:["Once everything is setup, you can verify your site is secured using a tool like ",(0,r.jsx)(n.a,{href:"https://www.sslchecker.com/sslchecker",children:"SSL Checker"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"authentication",children:"Authentication"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy natively supports secure authentication using KeyCloak. There is also a Simple Auth feature that doesn't require any additional setup. Usage instructions for both, as well as alternative auth methods, has now moved to the ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})})," page."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"network-exposure",children:"Network Exposure"}),"\n",(0,r.jsx)(n.p,{children:"Dashy is designed to run on your local network, behind your firewall. If you only access it from within your home or over a VPN, the defaults are fine."}),"\n",(0,r.jsxs)(n.p,{children:["If you do need to expose Dashy to the internet, you should put it behind a reverse proxy with its own authentication layer (e.g. Authelia, Authentik, Cloudflare Access, or your proxy's built-in auth). Don't rely solely on Dashy's built-in auth for internet-facing instances - it's a convenience feature for private networks, not a hardened perimeter control. See the ",(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})," for setup options."]}),"\n",(0,r.jsxs)(n.p,{children:["When Dashy runs in server mode (the default Docker setup), it exposes several API endpoints for things like status checks, config saving, system info, and a CORS proxy used by widgets. When authentication is enabled (via ",(0,r.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," or ",(0,r.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"}),"/",(0,r.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," env vars), all of these endpoints require valid credentials. Without auth configured, they are open. That's fine for private networks, but not appropriate for public access."]}),"\n",(0,r.jsxs)(n.p,{children:["The CORS proxy (",(0,r.jsx)(n.code,{children:"/cors-proxy"}),") is worth calling out specifically: it forwards requests from the Dashy server to external URLs, so widgets can reach APIs that don't set CORS headers. On a private network this is harmless, but on an internet-exposed instance without auth, it could be abused as an open proxy. Always enable authentication if your instance is reachable from untrusted networks."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"managing-containers-with-docker-compose",children:"Managing Containers with Docker Compose"}),"\n",(0,r.jsxs)(n.p,{children:["When you have a lot of containers, it quickly becomes hard to manage with ",(0,r.jsx)(n.code,{children:"docker run"})," commands. The solution to this is ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/compose/",children:"docker compose"}),", a handy tool for defining all a containers run settings in a single YAML file, and then spinning up that container with a single short command - ",(0,r.jsx)(n.code,{children:"docker compose up"}),". A good example of which can be seen in ",(0,r.jsx)(n.a,{href:"https://github.com/abhilesh/self-hosted_docker_setups",children:"@abhilesh's docker compose collection"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["You can use Dashy's default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:(0,r.jsx)(n.code,{children:"docker-compose.yml"})})," file as a template, and modify it according to your needs."]}),"\n",(0,r.jsx)(n.p,{children:"An example Docker compose, using the default base image from DockerHub, might look something like this:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n container_name: Dashy\n image: lissy93/dashy:latest\n volumes:\n - ./user-data:/app/user-data\n ports:\n - 4000:8080\n environment:\n - BASE_URL=/my-dashboard\n restart: unless-stopped\n healthcheck:\n test: ['CMD', 'node', '/app/services/healthcheck.js']\n interval: 1m30s\n timeout: 10s\n retries: 3\n start_period: 30s\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"passing-in-environmental-variables",children:"Passing in Environmental Variables"}),"\n",(0,r.jsxs)(n.p,{children:["With Docker, you can define environmental variables under the ",(0,r.jsx)(n.code,{children:"environment"})," section of your Docker compose file. Environmental variables are used to configure high-level settings, usually before the config file has been read. For a list of all supported env vars in Dashy, see ",(0,r.jsx)(n.a,{href:"/docs/developing#environmental-variables",children:"the developing docs"}),", or the default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.env",children:(0,r.jsx)(n.code,{children:".env"})})," file."]}),"\n",(0,r.jsxs)(n.p,{children:["A common use case, is to run Dashy under a sub-page, instead of at the root of a URL (e.g. ",(0,r.jsx)(n.code,{children:"https://my-homelab.local/dashy"})," instead of ",(0,r.jsx)(n.code,{children:"https://dashy.my-homelab.local"}),"). In this use-case, you'd specify the ",(0,r.jsx)(n.code,{children:"BASE_URL"})," variable in your compose file."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"environment:\n - BASE_URL=/dashy\n"})}),"\n",(0,r.jsxs)(n.p,{children:["You can also do the same thing with the docker run command, using the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/run/#set-environment-variables--e---env---env-file",children:(0,r.jsx)(n.code,{children:"--env"})})," flag.\nIf you've got many environmental variables, you might find it useful to put them in a ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/compose/env-file/",children:[(0,r.jsx)(n.code,{children:".env"})," file"]}),". Similarly, for Docker run you can use ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/run/#set-environment-variables--e---env---env-file",children:(0,r.jsx)(n.code,{children:"--env-file"})})," if you'd like to pass in a file containing all your environmental variables."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"setting-headers",children:"Setting Headers"}),"\n",(0,r.jsxs)(n.p,{children:["Any external requests made to a different origin (app/ service under a different domain) will be blocked if the correct headers are not specified. This is known as ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"})," (CORS) and is a security feature built into modern browsers."]}),"\n",(0,r.jsx)(n.p,{children:"If you see a CORS error in your console, this can be easily fixed by setting the correct headers. This is not a bug with Dashy, so please don't raise it as a bug!"}),"\n",(0,r.jsx)(n.h3,{id:"example-headers",children:"Example Headers"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#caddy",children:"Caddy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nginx",children:"NGINX"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#traefik",children:"Tr\xe6f\u026ak"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#haproxy",children:"HAProxy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#apache",children:"Apache"})}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsxs)(n.em,{children:["The following section briefly outlines how you can set headers for common web proxies/ servers. More info can be found in the documentation for the proxy that you are using, or in the ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"MDN Docs"}),"."]})}),"\n",(0,r.jsx)(n.p,{children:"These examples are using:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, but depending on what type of content you are enabling, this will vary. For example, to allow a site to be loaded in an iframe (for the modal or workspace views) you would use ",(0,r.jsx)(n.code,{children:"X-Frame-Options"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:["The domain root (",(0,r.jsx)(n.code,{children:"/"}),"), if your're hosting from a sub-page, replace that with your path."]}),"\n",(0,r.jsxs)(n.li,{children:["A wildcard (",(0,r.jsx)(n.code,{children:"*"}),"), which would allow access from traffic on any domain, this is discouraged, and you should replace it with the URL where you are hosting Dashy. Note that for requests that transport sensitive info, like credentials (e.g. Keycloak login), the wildcard is ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS#requests_with_credentials",children:"disallowed all together"})," and will be blocked."]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"caddy",children:"Caddy"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/header",children:["Caddy ",(0,r.jsx)(n.code,{children:"header"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"headers / {\n Access-Control-Allow-Origin *\n}\n"})}),"\n",(0,r.jsx)(n.h4,{id:"nginx",children:"NGINX"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_headers_module.html",children:["NGINX ",(0,r.jsx)(n.code,{children:"ngx_http_headers_module"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"location / {\n add_header Access-Control-Allow-Origin *;\n}\n"})}),"\n",(0,r.jsx)(n.p,{children:"Note this can also be done through the UI, using NGINX Proxy Manager."}),"\n",(0,r.jsx)(n.h4,{id:"traefik",children:"Traefik"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsx)(n.a,{href:"https://doc.traefik.io/traefik/middlewares/http/headers/#cors-headers",children:"Tr\xe6f\u026ak CORS headers docs"})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'labels:\n - "traefik.http.middlewares.testheader.headers.accesscontrolallowmethods=GET,OPTIONS,PUT"\n - "traefik.http.middlewares.testheader.headers.accesscontrolalloworiginlist=https://foo.bar.org,https://example.org"\n - "traefik.http.middlewares.testheader.headers.accesscontrolmaxage=100"\n - "traefik.http.middlewares.testheader.headers.addvaryheader=true"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"haproxy",children:"HAProxy"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.haproxy.com/documentation/hapee/latest/traffic-routing/rewrites/rewrite-responses/",children:"HAProxy Rewrite Response Docs"})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"/\n http-response add-header Access-Control-Allow-Origin *\n"})}),"\n",(0,r.jsx)(n.h4,{id:"apache",children:"Apache"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"https://httpd.apache.org/docs/current/mod/mod_headers.html",children:["Apache ",(0,r.jsx)(n.code,{children:"mode_headers"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'Header always set Access-Control-Allow-Origin "*"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"squid",children:"Squid"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:["See ",(0,r.jsxs)(n.a,{href:"http://www2.gr.squid-cache.org/Doc/config/request_header_access/",children:["Squid ",(0,r.jsx)(n.code,{children:"request_header_access"})," docs"]})," for more info."]}),"\n"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"request_header_access Authorization allow all\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"remote-access",children:"Remote Access"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#wireguard",children:"WireGuard"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#reverse-ssh-tunnel",children:"Reverse SSH Tunnel"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tcp-tunnel",children:"TCP Tunnel"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"wireguard",children:"WireGuard"}),"\n",(0,r.jsxs)(n.p,{children:["Using a VPN is one of the easiest ways to provide secure, full access to your local network from remote locations. ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/",children:"WireGuard"})," is a reasonably new open source VPN protocol, that was designed with ease of use, performance and security in mind. Unlike OpenVPN, it doesn't need to recreate the tunnel whenever connection is dropped, and it's also much easier to setup, using shared keys instead."]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Install Wireguard"})," - See the ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/install/",children:"Install Docs"})," for download links + instructions\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["On Debian-based systems, it's ",(0,r.jsx)(n.code,{children:"sudo apt install wireguard"})]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Generate a Private Key"})," - Run ",(0,r.jsx)(n.code,{children:"wg genkey"})," on the Wireguard server, and copy it to somewhere safe for later"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Create Server Config"})," - Open or create a file at ",(0,r.jsx)(n.code,{children:"/etc/wireguard/wg0.conf"})," and under ",(0,r.jsx)(n.code,{children:"[Interface]"})," add the following (see example below):\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Address"})," - as a subnet of all desired IPs"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PrivateKey"})," - that you just generated"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"ListenPort"})," - Default is ",(0,r.jsx)(n.code,{children:"51820"}),", but can be anything"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Get Client App"})," - Download the ",(0,r.jsx)(n.a,{href:"https://www.wireguard.com/install/",children:"WG client app"})," for your platform (Linux, Windows, MacOS, Android or iOS are all supported)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Create new Client Tunnel"})," - On your client app, there should be an option to create a new tunnel, when doing so a client private key will be generated (but if not, use the ",(0,r.jsx)(n.code,{children:"wg genkey"})," command again), and keep it somewhere safe. A public key will also be generated, and this will go in our saver config"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Add Clients to Server Config"})," - Head back to your ",(0,r.jsx)(n.code,{children:"wg0.conf"})," file on the server, create a ",(0,r.jsx)(n.code,{children:"[Peer]"})," section, and populate the following info\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"AllowedIPs"})," - List of IP address inside the subnet, the client should have access to"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PublicKey"})," - The public key for the client you just generated"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Start the Server"})," - You can now start the WG server, using: ",(0,r.jsx)(n.code,{children:"wg-quick up wg0"})," on your server"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Finish Client Setup"})," - Head back to your client device, and edit the config file, leave the private key as is, and add the following fields:\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"PublicKey"})," - The public key of the server"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Address"})," - This should match the ",(0,r.jsx)(n.code,{children:"AllowedIPs"})," section on the servers config file"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"DNS"})," - The DNS server that'll be used when accessing the network through the VPN"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"Endpoint"})," - The hostname or IP + Port where your WG server is running (you may need to forward this in your firewall's settings)"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Done"})," - Your clients should now be able to connect to your WG server :) Depending on your networks firewall rules, you may need to port forward the address of your WG server"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-server-config",children:(0,r.jsx)(n.strong,{children:"Example Server Config"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-ini",children:"# Server file\n[Interface]\n# Which networks does my interface belong to? Notice: /24 and /64\nAddress = 10.5.0.1/24, 2001:470:xxxx:xxxx::1/64\nPrivateKey = xxx\nListenPort = 51820\n\n# Peer 1\n[Peer]\nPublicKey = xxx\n# Which source IPs can I expect from that peer? Notice: /32 and /128\nAllowedIps = 10.5.0.35/32, 2001:470:xxxx:xxxx::746f:786f/128\n\n# Peer 2\n[Peer]\nPublicKey = xxx\n# Which source IPs can I expect from that peer? This one has a LAN which can\n# access hosts/jails without NAT.\n# Peer 2 has a single IP address inside the VPN: it's 10.5.0.25/32\nAllowedIps = 10.5.0.25/32,10.21.10.0/24,10.21.20.0/24,10.21.30.0/24,10.31.0.0/24,2001:470:xxxx:xxxx::ca:571e/128\n"})}),"\n",(0,r.jsx)(n.h4,{id:"example-client-config",children:(0,r.jsx)(n.strong,{children:"Example Client Config"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-ini",children:"[Interface]\n# Which networks does my interface belong to? Notice: /24 and /64\nAddress = 10.5.0.35/24, 2001:470:xxxx:xxxx::746f:786f/64\nPrivateKey = xxx\n\n# Server\n[Peer]\nPublicKey = xxx\n# I want to route everything through the server, both IPv4 and IPv6. All IPs are\n# thus available through the Server, and I can expect packets from any IP to\n# come from that peer.\nAllowedIPs = 0.0.0.0/0, ::0/0\n# Where is the server on the internet? This is a public address. The port\n# (:51820) is the same as ListenPort in the [Interface] of the Server file above\nEndpoint = 1.2.3.4:51820\n# Usually, clients are behind NAT. to keep the connection running, keep alive.\nPersistentKeepalive = 15\n"})}),"\n",(0,r.jsxs)(n.p,{children:["A useful tool for getting WG setup is ",(0,r.jsx)(n.a,{href:"https://github.com/trailofbits/algo",children:"Algo"}),". It includes scripts and docs which cover almost all devices, platforms and clients, and has best practices implemented, and security features enabled. All of this is better explained in ",(0,r.jsx)(n.a,{href:"https://blog.trailofbits.com/2016/12/12/meet-algo-the-vpn-that-works/",children:"this blog post"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"reverse-ssh-tunnel",children:"Reverse SSH Tunnel"}),"\n",(0,r.jsxs)(n.p,{children:["SSH (or ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Secure_Shell",children:"Secure Shell"}),") is a secure tunnel that allows you to connect to a remote host. Unlike the VPN methods, an SSH connection does not require an intermediary, and will not be affected by your IP changing. However it only allows you to access a single service at a time. SSH was really designed for terminal access, but because of the latter mentioned benefits it's useful to setup, as a fallback option."]}),"\n",(0,r.jsx)(n.p,{children:"Directly SSH'ing into your home, would require you to open a port (usually 22), which would be terrible for security, and is not recommended. However a reverse SSH connection is initiated from inside your network. Once the connection is established, the port is redirected, allowing you to use the established connection to SSH into your home network."}),"\n",(0,r.jsxs)(n.p,{children:["The issue you've probably spotted, is that most public, corporate, and institutional networks will block SSH connections. To overcome this, you'd have to establish a server outside of your homelab that your homelab's device could SSH into to establish the reverse SSH connection. You can then connect to that remote server (the ",(0,r.jsx)(n.em,{children:"mothership"}),"), which in turn connects to your home network."]}),"\n",(0,r.jsxs)(n.p,{children:["Now all of this is starting to sound like quite a lot of work, but this is where services like ",(0,r.jsx)(n.a,{href:"https://remote.it/",children:"remot3.it"})," come in. They maintain the intermediary mothership server, and create the tunnel service for you. It's free for personal use, secure and easy. There are several similar services, such as ",(0,r.jsx)(n.a,{href:"https://remoteiot.com/",children:"RemoteIoT"}),", or you could create your own on a cloud VPS (see ",(0,r.jsx)(n.a,{href:"https://gist.github.com/nileshtrivedi/4c615e8d3c1bf053b0d31176b9e69e42",children:"this tutorial"})," for more info on that)."]}),"\n",(0,r.jsxs)(n.p,{children:["Before getting started, you'll need to head over to ",(0,r.jsx)(n.a,{href:"https://app.remote.it/auth/#/sign-up",children:"Remote.it"})," and create an account."]}),"\n",(0,r.jsx)(n.p,{children:"Then setup your local device:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["If you haven't already done so, you'll need to enable and configure SSH.\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["This is out-of-scope of this article, but I've explained it in detail in ",(0,r.jsx)(n.a,{href:"https://notes.aliciasykes.com/22798/my-server-setup#configure-ssh",children:"this post"}),"."]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Download the Remote.it install script from their ",(0,r.jsx)(n.a,{href:"https://github.com/remoteit/installer",children:"GitHub"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"curl -LkO https://raw.githubusercontent.com/remoteit/installer/master/scripts/auto-install.sh"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:["Make it executable, with ",(0,r.jsx)(n.code,{children:"chmod +x ./auto-install.sh"}),", and then run it with ",(0,r.jsx)(n.code,{children:"sudo ./auto-install.sh"})]}),"\n",(0,r.jsxs)(n.li,{children:["Finally, configure your device, by running ",(0,r.jsx)(n.code,{children:"sudo connectd_installer"})," and following the on-screen instructions"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"And when you're ready to connect to it:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Login to ",(0,r.jsx)(n.a,{href:"https://app.remote.it/",children:"app.remote.it"}),", and select the name of your device"]}),"\n",(0,r.jsx)(n.li,{children:"You should see a list of running services, click SSH"}),"\n",(0,r.jsx)(n.li,{children:"You'll then be presented with some SSH credentials that you can now use to securely connect to your home, via the Remote.it servers"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Done :)"}),"\n",(0,r.jsx)(n.h3,{id:"tcp-tunnel",children:"TCP Tunnel"}),"\n",(0,r.jsxs)(n.p,{children:["If you're running Dashy on your local network, behind a firewall, but need to temporarily share it with someone external, this can be achieved quickly and securely using ",(0,r.jsx)(n.a,{href:"https://ngrok.com/",children:"Ngrok"}),". It's basically a super slick, encrypted TCP tunnel that provides an internet-accessible address that anyone use to access your local service, from anywhere."]}),"\n",(0,r.jsxs)(n.p,{children:["To get started, ",(0,r.jsx)(n.a,{href:"https://ngrok.com/download",children:"Download"})," and install Ngrok for your system, then just run ",(0,r.jsx)(n.code,{children:"ngrok http [port]"})," (replace the port with the http port where Dashy is running, e.g. 8080). When ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-local-https",children:"using https"}),", specify the full local url/ ip including the protocol."]}),"\n",(0,r.jsxs)(n.p,{children:["Some Ngrok features require you to be authenticated, you can ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/signup",children:"create a free account"})," and generate a token in ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/auth/your-authtoken",children:"your dashboard"}),", then run ",(0,r.jsx)(n.code,{children:"ngrok authtoken [token]"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["It's recommended to use authentication for any publicly accessible service. Dashy has an ",(0,r.jsx)(n.a,{href:"/docs/authentication",children:"Auth"})," feature built in, but an even easier method it to use the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-auth",children:(0,r.jsx)(n.code,{children:"-auth"})})," switch. E.g. ",(0,r.jsx)(n.code,{children:'ngrok http -auth="username:password123" 8080'})]}),"\n",(0,r.jsxs)(n.p,{children:["By default, your web app is assigned a randomly generated ngrok domain, but you can also use your own custom domain. Under the ",(0,r.jsx)(n.a,{href:"https://dashboard.ngrok.com/endpoints/domains",children:"Domains Tab"})," of your Ngrok dashboard, add your domain, and follow the CNAME instructions. You can now use your domain, with the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-custom-domains",children:(0,r.jsx)(n.code,{children:"-hostname"})})," switch, e.g. ",(0,r.jsx)(n.code,{children:"ngrok http -region=us -hostname=dashy.example.com 8080"}),". If you don't have your own domain name, you can instead use a custom sub-domain (e.g. ",(0,r.jsx)(n.code,{children:"alicia-dashy.ngrok.io"}),"), using the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#custom-subdomain-names",children:(0,r.jsx)(n.code,{children:"-subdomain"})})," switch."]}),"\n",(0,r.jsxs)(n.p,{children:["To integrate this into your docker-compose, take a look at the ",(0,r.jsx)(n.a,{href:"https://github.com/gtriggiano/ngrok-tunnel",children:"gtriggiano/ngrok-tunnel"})," container."]}),"\n",(0,r.jsxs)(n.p,{children:["There's so much more you can do with Ngrok, such as exposing a directory as a file browser, using websockets, relaying requests, rewriting headers, inspecting traffic, TLS and TCP tunnels and lots more. All or which is explained in ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs",children:"the Documentation"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["It's worth noting that Ngrok isn't the only option here, other options include: ",(0,r.jsx)(n.a,{href:"https://github.com/fatedier/frp",children:"FRP"}),", ",(0,r.jsx)(n.a,{href:"https://inlets.dev",children:"Inlets"}),", ",(0,r.jsx)(n.a,{href:"https://localtunnel.me/",children:"Local Tunnel"}),", ",(0,r.jsx)(n.a,{href:"https://tailscale.com/",children:"TailScale"}),", etc. Check out ",(0,r.jsx)(n.a,{href:"https://github.com/anderspitman/awesome-tunneling",children:"Awesome Tunneling"})," for a list of alternatives."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"custom-domain",children:"Custom Domain"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#using-dns",children:"Using DNS"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#using-nginx",children:"Using NGINX"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"using-dns",children:"Using DNS"}),"\n",(0,r.jsxs)(n.p,{children:["For locally running services, a domain can be set up directly in the DNS records. This method is really quick and easy, and doesn't require you to purchase an actual domain. Just update your networks DNS resolver, to point your desired URL to the local IP where Dashy (or any other app) is running. For example, a line in your hosts file might look something like: ",(0,r.jsx)(n.code,{children:"192.168.0.2 dashy.homelab.local"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're using Pi-Hole, a similar thing can be done in the ",(0,r.jsx)(n.code,{children:"/etc/dnsmasq.d/03-custom-dns.conf"})," file, add a line like: ",(0,r.jsx)(n.code,{children:"address=/dashy.example.com/192.168.2.0"})," for each of your services."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're running OPNSense/ PfSense, then this can be done through the UI with Unbound, it's explained nicely in ",(0,r.jsx)(n.a,{href:"https://homenetworkguy.com/how-to/use-custom-domain-name-in-internal-network/",children:"this article"}),", by Dustin Casto."]}),"\n",(0,r.jsx)(n.h3,{id:"using-nginx",children:"Using NGINX"}),"\n",(0,r.jsx)(n.p,{children:"If you're using NGINX, then you can use your own domain name, with a config similar to the below example."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"upstream dashy {\n server 127.0.0.1:32400;\n}\n\nserver {\n listen 8080;\n server_name dashy.mydomain.com;\n\n # Setup SSL\n ssl_certificate /var/www/mydomain/sslcert.pem;\n ssl_certificate_key /var/www/mydomain/sslkey.pem;\n ssl_protocols TLSv1.2 TLSv1.3;\n ssl_ciphers 'EECDH+AESGCM:EDH+AESGCM:AES256+EECDH:AES256+EDH';\n ssl_session_timeout 5m;\n ssl_prefer_server_ciphers on;\n\n location / {\n proxy_pass http://dashy;\n proxy_redirect off;\n proxy_buffering off;\n proxy_set_header host $host;\n proxy_set_header X-Real-IP $remote_addr;\n proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;\n proxy_next_upstream error timeout invalid_header http_500 http_502 http_503 http_504;\n }\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Similarly, a basic ",(0,r.jsx)(n.code,{children:"Caddyfile"})," might look like:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"dashy.example.com {\n reverse_proxy / nginx:8080\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more info, ",(0,r.jsx)(n.a,{href:"https://thehomelab.wiki/books/dns-reverse-proxy/page/create-domain-records-to-point-to-your-home-server-on-cloudflare-using-nginx-progy-manager",children:"this guide"})," on Setting up Domains with NGINX Proxy Manager and CloudFlare may be useful."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"container-security",children:"Container Security"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#keep-docker-up-to-date",children:"Keep Docker Up-To-Date"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#set-resource-quotas",children:"Set Resource Quotas"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dont-run-as-root",children:"Don't Run as Root"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#specify-a-user",children:"Specify a User"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#limit-capabilities",children:"Limit Capabilities"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#prevent-new-privileges-being-added",children:"Prevent new Privileges being Added"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disable-inter-container-communication",children:"Disable Inter-Container Communication"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dont-expose-the-docker-daemon-socket",children:"Don't Expose the Docker Daemon Socket"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#use-read-only-volumes",children:"Use Read-Only Volumes"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#set-the-logging-level",children:"Set the Logging Level"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#verify-image-before-pulling",children:"Verify Image before Pulling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#specify-the-tag",children:"Specify the Tag"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-security-scanning",children:"Container Security Scanning"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#registry-security",children:"Registry Security"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#security-modules",children:"Security Modules"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"keep-docker-up-to-date",children:"Keep Docker Up-To-Date"}),"\n",(0,r.jsxs)(n.p,{children:["To prevent known container escape vulnerabilities, which typically end in escalating to root/administrator privileges, patching Docker Engine and Docker Machine is crucial. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/install/",children:"Docker Installation Docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"set-resource-quotas",children:"Set Resource Quotas"}),"\n",(0,r.jsxs)(n.p,{children:["Docker enables you to limit resource consumption (CPU, memory, disk) on a per-container basis. This not only enhances system performance, but also prevents a compromised container from consuming a large amount of resources, in order to disrupt service or perform malicious activities. To learn more, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Resource Constraints Docs"})]}),"\n",(0,r.jsxs)(n.p,{children:["For example, to run Dashy with max of 1GB ram, and max of 50% of 1 CP core:\n",(0,r.jsx)(n.code,{children:'docker run -d -p 8080:8080 --cpus=".5" --memory="1024m" lissy93/dashy:latest'})]}),"\n",(0,r.jsx)(n.h3,{id:"dont-run-as-root",children:"Don't Run as Root"}),"\n",(0,r.jsxs)(n.p,{children:["Running Docker commands with ",(0,r.jsx)(n.code,{children:"sudo"})," gives the container more host-level access than it needs. You should run Docker as a non-root host user instead."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're facing permission issues on Debian-based systems when running Docker commands without ",(0,r.jsx)(n.code,{children:"sudo"}),", you may need to add your user to the Docker group. First create the group: ",(0,r.jsx)(n.code,{children:"sudo groupadd docker"}),", then add your (non-root) user: ",(0,r.jsx)(n.code,{children:"sudo usermod \u2212aG docker [my-username]"}),", finally ",(0,r.jsx)(n.code,{children:"newgrp docker"})," to refresh."]}),"\n",(0,r.jsx)(n.h3,{id:"specify-a-user",children:"Specify a User"}),"\n",(0,r.jsxs)(n.p,{children:["For containers in general, running as an unprivileged user is one of the best ways to prevent privilege escalation attacks. You can specify a user with the ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/engine/reference/run/#user",children:[(0,r.jsx)(n.code,{children:"--user"})," param"]}),", using the user ID (",(0,r.jsx)(n.code,{children:"UID"}),") from ",(0,r.jsx)(n.code,{children:"id -u"})," and group ID (",(0,r.jsx)(n.code,{children:"GID"}),") from ",(0,r.jsx)(n.code,{children:"id -g"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.strong,{children:"Note for Dashy:"})," If you use features that write to disk (saving config through the UI), the process needs write access to ",(0,r.jsx)(n.code,{children:"/app/user-data/"}),". Since the default image creates these directories as root, running with ",(0,r.jsx)(n.code,{children:"--user"})," will cause those features to fail with permission errors unless you also fix ownership of the mounted volumes. If you only use Dashy in read-only mode, running as a non-root user works fine:"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.code,{children:"docker run --user 1000:1000 -p 8080:8080 lissy93/dashy"})}),"\n",(0,r.jsx)(n.p,{children:"Or with Docker Compose, using an environmental variable:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy\n user: ${CURRENT_UID}\n ports: [ 4000:8080 ]\n"})}),"\n",(0,r.jsxs)(n.p,{children:["And then to set the variable, and start the container, run: ",(0,r.jsx)(n.code,{children:"CURRENT_UID=$(id -u):$(id -g) docker-compose up"})]}),"\n",(0,r.jsx)(n.h3,{id:"limit-capabilities",children:"Limit capabilities"}),"\n",(0,r.jsxs)(n.p,{children:["Docker containers run with a subset of ",(0,r.jsx)(n.a,{href:"https://man7.org/linux/man-pages/man7/capabilities.7.html",children:"Linux Kernal's Capabilities"})," by default. It's good practice to drop privilege permissions that are not needed for any given container."]}),"\n",(0,r.jsxs)(n.p,{children:["With Docker run, you can use the ",(0,r.jsx)(n.code,{children:"--cap-drop"})," flag to remove capabilities, you can also use ",(0,r.jsx)(n.code,{children:"--cap-drop=all"})," and then define just the required permissions using the ",(0,r.jsx)(n.code,{children:"--cap-add"})," option. For a list of available capabilities, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#runtime-privilege-and-linux-capabilities",children:"Privilege Capabilities Docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Note that dropping privileges and capabilities on runtime is not fool-proof, and often any leftover privileges can be used to re-escalate, see ",(0,r.jsx)(n.a,{href:"https://wiki.sei.cmu.edu/confluence/display/c/POS36-C.+Observe+correct+revocation+order+while+relinquishing+privileges",children:"POS36-C"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Here's an example using docker-compose, removing privileges that are not required for Dashy to run:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy\n ports: [ 4000:8080 ]\n cap_drop:\n - ALL\n cap_add:\n - CHOWN\n - SETGID\n - SETUID\n - DAC_OVERRIDE\n - NET_BIND_SERVICE\n"})}),"\n",(0,r.jsx)(n.h3,{id:"prevent-new-privileges-being-added",children:"Prevent new Privileges being Added"}),"\n",(0,r.jsxs)(n.p,{children:["To prevent processes inside the container from getting additional privileges, pass in the ",(0,r.jsx)(n.code,{children:"--security-opt=no-new-privileges:true"})," option to the Docker run command (see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/run/#security-configuration",children:"docs"}),")."]}),"\n",(0,r.jsxs)(n.p,{children:["Run Command:\n",(0,r.jsx)(n.code,{children:"docker run --security-opt=no-new-privileges:true -p 8080:8080 lissy93/dashy"})]}),"\n",(0,r.jsx)(n.p,{children:"Docker Compose"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"security_opt:\n- no-new-privileges:true\n"})}),"\n",(0,r.jsx)(n.h3,{id:"disable-inter-container-communication",children:"Disable Inter-Container Communication"}),"\n",(0,r.jsxs)(n.p,{children:["By default Docker containers can talk to each other (using ",(0,r.jsxs)(n.a,{href:"https://docs.docker.com/config/containers/container-networking/",children:[(0,r.jsx)(n.code,{children:"docker0"})," bridged network"]}),"). If you don't need this capability, then it should be disabled. This can be done with the ",(0,r.jsx)(n.code,{children:"--icc=false"})," in your run command. You can learn more about how to facilitate secure communication between containers in the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/compose/networking/",children:"Compose Networking docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"dont-expose-the-docker-daemon-socket",children:"Don't Expose the Docker Daemon Socket"}),"\n",(0,r.jsxs)(n.p,{children:["Docker socket ",(0,r.jsx)(n.code,{children:"/var/run/docker.sock"})," is the UNIX socket that Docker is listening to. This is the primary entry point for the Docker API. The owner of this socket is root. Giving someone access to it is equivalent to giving unrestricted root access to your host."]}),"\n",(0,r.jsxs)(n.p,{children:["You should ",(0,r.jsx)(n.strong,{children:"not"})," enable TCP Docker daemon socket (",(0,r.jsx)(n.code,{children:"-H tcp://0.0.0.0:XXX"}),"), as doing so exposes un-encrypted and unauthenticated direct access to the Docker daemon, and if the host is connected to the internet, the daemon on your computer can be used by anyone from the public internet- which is bad. If you need TCP, you should ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/reference/commandline/dockerd/#daemon-socket-option",children:"see the docs"})," to understand how to do this more securely.\nSimilarly, never expose ",(0,r.jsx)(n.code,{children:"/var/run/docker.sock"})," to other containers as a volume, as it can be exploited."]}),"\n",(0,r.jsx)(n.h3,{id:"use-read-only-volumes",children:"Use Read-Only Volumes"}),"\n",(0,r.jsxs)(n.p,{children:["You can specify that a volume should be read-only by appending ",(0,r.jsx)(n.code,{children:":ro"})," to the ",(0,r.jsx)(n.code,{children:"-v"})," switch. If you don't need the in-app config editor, mount your ",(0,r.jsx)(n.code,{children:"user-data"})," read-only:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/dashy-data:/app/user-data:ro \\\n lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If you do want config changes from the UI to persist back to disk, leave the mount writable. You can also use ",(0,r.jsx)(n.code,{children:"--read-only"})," to make the whole container filesystem read-only, but in that case UI-driven config edits will not be saved."]}),"\n",(0,r.jsx)(n.h3,{id:"set-the-logging-level",children:"Set the Logging Level"}),"\n",(0,r.jsxs)(n.p,{children:["Logging is important, as it enables you to review events in the future, and in the case of a compromise this will let get an idea of what may have happened. The default log level is ",(0,r.jsx)(n.code,{children:"INFO"}),", and this is also the recommendation, use ",(0,r.jsx)(n.code,{children:"--log-level info"})," to ensure this is set."]}),"\n",(0,r.jsx)(n.h3,{id:"verify-image-before-pulling",children:"Verify Image before Pulling"}),"\n",(0,r.jsx)(n.p,{children:"Only use trusted images, from verified/ official sources. If an app is open source, it is more likely to be safe, as anyone can verify the code. There are also tools available for scanning containers,"}),"\n",(0,r.jsx)(n.p,{children:"Unless otherwise configured, containers can communicate among each other, so running one bad image may lead to other areas of your setup being compromised. Docker images typically contain both original code, as well as up-stream packages, and even if that image has come from a trusted source, the up-stream packages it includes may not have."}),"\n",(0,r.jsx)(n.h3,{id:"specify-the-tag",children:"Specify the Tag"}),"\n",(0,r.jsxs)(n.p,{children:["Using fixed tags (as opposed to ",(0,r.jsx)(n.code,{children:":latest"})," ) will ensure immutability, meaning the base image will not change between builds. Note that for Dashy, the app is being actively developed, new features, bug fixes and general improvements are merged each week, and if you use a fixed version you will not enjoy these benefits. So it's up to you weather you would prefer a stable and reproducible environment, or the latest features and enhancements."]}),"\n",(0,r.jsx)(n.h3,{id:"container-security-scanning",children:"Container Security Scanning"}),"\n",(0,r.jsxs)(n.p,{children:["It's helpful to be aware of any potential security issues in any of the Docker images you are using. You can run a quick scan using Snyk on any image to output known vulnerabilities using ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/engine/scan/",children:"Docker scan"}),", e.g: ",(0,r.jsx)(n.code,{children:"docker scan lissy93/dashy:latest"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["A similar product is ",(0,r.jsx)(n.a,{href:"https://github.com/aquasecurity/trivy",children:"Trivy"}),", which is free an open source. First install it (with your package manager), then to scan an image, just run: ",(0,r.jsx)(n.code,{children:"trivy image lissy93/dashy:latest"})]}),"\n",(0,r.jsxs)(n.p,{children:["For larger systems, RedHat ",(0,r.jsx)(n.a,{href:"https://www.redhat.com/en/topics/containers/what-is-clair",children:"Clair"})," is an app for parsing image contents and reporting on any found vulnerabilities. You run it locally in a container, and configure it with YAML. It can be integrated with Red Hat Quay, to show results on a dashboard. Most of these use static analysis to find potential issues, and scan included packages for any known security vulnerabilities."]}),"\n",(0,r.jsx)(n.h3,{id:"registry-security",children:"Registry Security"}),"\n",(0,r.jsxs)(n.p,{children:["Although over-kill for most users, you could run your own registry locally which would give you ultimate control over all images, see the ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/registry/deploying/",children:"Deploying a Registry Docs"})," for more info. Another option is ",(0,r.jsx)(n.a,{href:"https://docker-docs.netlify.app/ee/dtr/",children:"Docker Trusted Registry"}),", it's great for enterprise applications, it sits behind your firewall, running on a swarm managed by Docker Universal Control Plane, and lets you securely store and manage your Docker images, mitigating the risk of breaches from the internet."]}),"\n",(0,r.jsx)(n.h3,{id:"security-modules",children:"Security Modules"}),"\n",(0,r.jsx)(n.p,{children:"Docker supports several modules that let you write your own security profiles."}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://www.apparmor.net/",children:"AppArmor"}),"is a kernel module that proactively protects the operating system and applications from external or internal threats, by enabling you to restrict programs' capabilities with per-program profiles. You can specify either a security policy by name, or by file path with the ",(0,r.jsx)(n.code,{children:"apparmor"})," flag in docker run. Learn more about writing profiles, ",(0,r.jsx)(n.a,{href:"https://gitlab.com/apparmor/apparmor/-/wikis/QuickProfileLanguage",children:"here"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Seccomp",children:"Seccomp"})," (Secure Computing Mode) is a sandboxing facility in the Linux kernel that acts like a firewall for system calls (syscalls). It uses Berkeley Packet Filter (BPF) rules to filter syscalls and control how they are handled. These filters can significantly limit a containers access to the Docker Host's Linux kernel - especially for simple containers/applications. It requires a Linux-based Docker host, with secomp enabled, and you can check for this by running ",(0,r.jsx)(n.code,{children:"docker info | grep seccomp"}),". A great resource for learning more about this is ",(0,r.jsx)(n.a,{href:"https://training.play-with-docker.com/security-seccomp/",children:"DockerLabs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"web-server-configuration",children:"Web Server Configuration"}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"The following section only applies if you are not using Docker, and would like to use your own web server"})}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy ships with a pre-configured Node.js server, in ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/server.js",children:(0,r.jsx)(n.code,{children:"server.js"})})," which serves up the contents of the ",(0,r.jsx)(n.code,{children:"./dist"})," directory on a given port. You can start the server by running ",(0,r.jsx)(n.code,{children:"node server"}),". Note that the app must have been build (run ",(0,r.jsx)(n.code,{children:"yarn build"}),"), and you need ",(0,r.jsx)(n.a,{href:"https://nodejs.org",children:"Node.js"})," installed."]}),"\n",(0,r.jsxs)(n.p,{children:["If you wish to run Dashy from a sub page (e.g. ",(0,r.jsx)(n.code,{children:"example.com/dashy"}),"), then just set the ",(0,r.jsx)(n.code,{children:"BASE_URL"})," environmental variable to that page name (in this example, ",(0,r.jsx)(n.code,{children:"/dashy"}),"), before building the app, and the path to all assets will then resolve to the new path, instead of ",(0,r.jsx)(n.code,{children:"./"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"However, since Dashy is just a static web application, it can be served with whatever server you like. The following section outlines how you can configure a web server."}),"\n",(0,r.jsxs)(n.p,{children:["Note, that if you choose not to use ",(0,r.jsx)(n.code,{children:"server.js"})," to serve up the app, you will loose access to the following features:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Loading page, while the app is building"}),"\n",(0,r.jsx)(n.li,{children:"Writing config file to disk from the UI"}),"\n",(0,r.jsx)(n.li,{children:"Website status indicators, and ping checks"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Example Configs"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nginx",children:"NGINX"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#apache",children:"Apache"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#caddy",children:"Caddy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#firebase-hosting",children:"Firebase"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpanel",children:"cPanel"})}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"nginx-1",children:"NGINX"}),"\n",(0,r.jsxs)(n.p,{children:["Create a new file in ",(0,r.jsx)(n.code,{children:"/etc/nginx/sites-enabled/dashy"})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"server {\n\tlisten 8080;\n\tlisten [::]:8080;\n\n\troot /var/www/dashy/html;\n\tindex index.html;\n\n\tserver_name your-domain.com www.your-domain.com;\n\n\tlocation / {\n\t\ttry_files $uri $uri/ =404;\n\t}\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["To use HTML5 history mode (the default - controlled via the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE"})," build-time env var), replace the inside of the location block with: ",(0,r.jsx)(n.code,{children:"try_files $uri $uri/ /index.html;"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Then upload the build contents of Dashy's dist directory to that location.\nFor example: ",(0,r.jsx)(n.code,{children:"scp -r ./dist/* [username]@[server_ip]:/var/www/dashy/html"})]}),"\n",(0,r.jsx)(n.h3,{id:"apache-1",children:"Apache"}),"\n",(0,r.jsxs)(n.p,{children:["Copy Dashy's dist folder to your apache server, ",(0,r.jsx)(n.code,{children:"sudo cp -r ./dashy/dist /var/www/html/dashy"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["In your Apache config, ",(0,r.jsx)(n.code,{children:"/etc/apche2/apache2.conf"})," add:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"\n\tOptions Indexes FollowSymLinks\n\tAllowOverride All\n\tRequire all granted\n\n\n\n RewriteEngine On\n RewriteBase /\n RewriteRule ^index\\.html$ - [L]\n RewriteCond %{REQUEST_FILENAME} !-f\n RewriteCond %{REQUEST_FILENAME} !-d\n RewriteRule . /index.html [L]\n\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Add a ",(0,r.jsx)(n.code,{children:".htaccess"})," file within ",(0,r.jsx)(n.code,{children:"/var/www/html/dashy/.htaccess"}),", and add:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Options -MultiViews\nRewriteEngine On\nRewriteCond %{REQUEST_FILENAME} !-f\nRewriteRule ^ index.html [QSA,L]\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then restart Apache, with ",(0,r.jsx)(n.code,{children:"sudo systemctl restart apache2"})]}),"\n",(0,r.jsx)(n.h3,{id:"caddy-1",children:"Caddy"}),"\n",(0,r.jsx)(n.p,{children:"Caddy v2"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"try_files {path} /\n"})}),"\n",(0,r.jsx)(n.p,{children:"Caddy v1"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"rewrite {\n regexp .*\n to {path} /\n}\n"})}),"\n",(0,r.jsx)(n.h3,{id:"firebase-hosting",children:"Firebase Hosting"}),"\n",(0,r.jsxs)(n.p,{children:["Create a file names ",(0,r.jsx)(n.code,{children:"firebase.json"}),", and populate it with something similar to:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'{\n "hosting": {\n "public": "dist",\n "rewrites": [\n {\n "source": "**",\n "destination": "/index.html"\n }\n ]\n }\n}\n'})}),"\n",(0,r.jsx)(n.h3,{id:"cpanel",children:"cPanel"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsx)(n.li,{children:"Login to your WHM"}),"\n",(0,r.jsx)(n.li,{children:"Open 'Feature Manager' on the left sidebar"}),"\n",(0,r.jsx)(n.li,{children:"Under 'Manage feature list', click 'Edit'"}),"\n",(0,r.jsx)(n.li,{children:"Find 'Application manager' in the list, enable it and hit 'Save'"}),"\n",(0,r.jsx)(n.li,{children:"Log into your users cPanel account, and under 'Software' find 'Application Manager'"}),"\n",(0,r.jsx)(n.li,{children:"Click 'Register Application', fill in the form using the path that Dashy is located, and choose a domain, and hit 'Save'"}),"\n",(0,r.jsx)(n.li,{children:"The application should now show up in the list, click 'Ensure dependencies', and move the toggle switch to 'Enabled'"}),"\n",(0,r.jsx)(n.li,{children:"If you need to change the port, click 'Add environmental variable', give it the name 'PORT', choose a port number and press 'Save'."}),"\n",(0,r.jsx)(n.li,{children:"Dashy should now be running at your selected path an on a given port"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"running-a-modified-version-of-the-app",children:"Running a Modified Version of the App"}),"\n",(0,r.jsx)(n.p,{children:"If you'd like to make any code changes to the app, and deploy your modified version, this section briefly explains how."}),"\n",(0,r.jsxs)(n.p,{children:["The first step is to fork the project on GitHub, and clone it to your local system. Next, install the dependencies (",(0,r.jsx)(n.code,{children:"yarn"}),"), and start the development server (",(0,r.jsx)(n.code,{children:"yarn dev"}),") and visit ",(0,r.jsx)(n.code,{children:"localhost:8080"})," in your browser. You can then make changes to the codebase, and see the live app update in real-time. Once you've finished, running ",(0,r.jsx)(n.code,{children:"yarn build"})," will build the app for production, and output the assets into ",(0,r.jsx)(n.code,{children:"./dist"})," which can then be deployed using a web server, CDN or the built-in Node server with ",(0,r.jsx)(n.code,{children:"yarn start"}),". For more info on all of this, take a look at the ",(0,r.jsx)(n.a,{href:"/docs/developing",children:"Developing Docs"}),". To build your own Docker container from the modified app, see ",(0,r.jsx)(n.a,{href:"#building-your-own-container",children:"Building your Own Container"})]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"building-your-own-container",children:"Building your Own Container"}),"\n",(0,r.jsx)(n.p,{children:"Similar to above, you'll first need to fork and clone Dashy to your local system, and then install dependencies."}),"\n",(0,r.jsxs)(n.p,{children:["Then, either use Dashy's default ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/Dockerfile",children:(0,r.jsx)(n.code,{children:"Dockerfile"})})," as is, or modify it according to your needs."]}),"\n",(0,r.jsxs)(n.p,{children:["To build and deploy locally, first build the app with: ",(0,r.jsx)(n.code,{children:"docker build -t dashy ."}),", and then start the app with ",(0,r.jsx)(n.code,{children:"docker run -p 8080:8080 --name my-dashboard dashy"}),". Or modify the ",(0,r.jsx)(n.code,{children:"docker-compose.yml"})," file, replacing ",(0,r.jsx)(n.code,{children:"image: lissy93/dashy"})," with ",(0,r.jsx)(n.code,{children:"build: ."})," and run ",(0,r.jsx)(n.code,{children:"docker compose up"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Your container should now be running, and will appear in the list when you run ",(0,r.jsx)(n.code,{children:"docker container ls \u2013a"}),". If you'd like to enter the container, run ",(0,r.jsx)(n.code,{children:"docker exec -it [container-id] /bin/ash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["You may wish to upload your image to a container registry for easier access. Note that if you choose to do this on a public registry, please name your container something other than just 'dashy', to avoid confusion with the official image.\nYou can push your build image, by running: ",(0,r.jsx)(n.code,{children:"docker push ghcr.io/OWNER/IMAGE_NAME:latest"}),". You will first need to authenticate, this can be done by running ",(0,r.jsx)(n.code,{children:"echo $CR_PAT | docker login ghcr.io -u USERNAME --password-stdin"}),", where ",(0,r.jsx)(n.code,{children:"CR_PAT"})," is an environmental variable containing a token generated from your GitHub account. For more info, see the ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/packages/working-with-a-github-packages-registry/working-with-the-container-registry",children:"Container Registry Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,r.jsx)(n.hr,{})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>a});var i=s(6540);const r={},o=i.createContext(r);function t(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/2140bf91.8a811e48.js b/assets/js/2140bf91.0fab015d.js similarity index 99% rename from assets/js/2140bf91.8a811e48.js rename to assets/js/2140bf91.0fab015d.js index 8c926bd8..d08eb5a7 100644 --- a/assets/js/2140bf91.8a811e48.js +++ b/assets/js/2140bf91.0fab015d.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[146],{9327(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>r,default:()=>h,frontMatter:()=>a,metadata:()=>i,toc:()=>c});const i=JSON.parse('{"id":"pages-and-sections","title":"Pages and Sections","description":"Page Metadata","source":"@site/docs/pages-and-sections.md","sourceDirName":".","slug":"/pages-and-sections","permalink":"/docs/pages-and-sections","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/pages-and-sections.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Cloud Backup and Restore","permalink":"/docs/backup-restore"},"next":{"title":"*Dashy Showcase* \ud83c\udf1f","permalink":"/docs/showcase"}}');var t=s(4848),o=s(8453);const a={},r="Pages and Sections",l={},c=[{value:"Page Metadata",id:"page-metadata",level:2},{value:"Multi-Page Support",id:"multi-page-support",level:2},{value:"Using Local Sub-Pages",id:"using-local-sub-pages",level:3},{value:"Using Remote Sub-Pages",id:"using-remote-sub-pages",level:3},{value:"Restrictions",id:"restrictions",level:3},{value:"URL Structure",id:"url-structure",level:3},{value:"Layout",id:"layout",level:2},{value:"Making a section wider or taller",id:"making-a-section-wider-or-taller",level:3},{value:"Items inside a section",id:"items-inside-a-section",level:3},{value:"Sub-Items",id:"sub-items",level:3}];function d(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"pages-and-sections",children:"Pages and Sections"})}),"\n",(0,t.jsx)(n.h2,{id:"page-metadata",children:"Page Metadata"}),"\n",(0,t.jsxs)(n.p,{children:["Set your dashboard's branding under ",(0,t.jsx)(n.code,{children:"pageInfo"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pageInfo:\n title: My Dashboard # Used for main h1 title, and browser tab text\n description: Home server links\n logo: /web-icons/my-logo.png # path/URL to optional logo to display next to title\n favicon: 'https://example.com/path/to/icon' # path/URL to a favicon (shows in browser tab)\n color: '#2a7cf0' # Hex color, to set the browser/address bar color on mobile (supported browsers only)\n footer: '\xa9 2026 Me' # Optional text or HTML content, to display in the pages footer\n"})}),"\n",(0,t.jsx)(n.p,{children:"If you have multiple configs/pages, then these values swap automatically as you navigate between sub-pages."}),"\n",(0,t.jsxs)(n.p,{children:["The only caveat being, if you install Dashy as a PWA, the installed app's name, icon, and splash-screen colour come from the bundled ",(0,t.jsx)(n.code,{children:"manifest.webmanifest"})," (baked at build time) rather than ",(0,t.jsx)(n.code,{children:"pageInfo"}),". Runtime values only apply when browsing in a regular tab or browser."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"multi-page-support",children:"Multi-Page Support"}),"\n",(0,t.jsx)(n.p,{children:"You can have additional pages within your dashboard, with each having it's own config file. The config files for sub-pages can either be stored locally, or hosted separately. A link to each additional page will be displayed in the navigation bar."}),"\n",(0,t.jsx)(n.p,{children:"You can edit additional pages using the interactive editor, exactly the same was as your primary page (so long as it's local). But please save changes to one page, before you start editing the next."}),"\n",(0,t.jsx)(n.h3,{id:"using-local-sub-pages",children:"Using Local Sub-Pages"}),"\n",(0,t.jsxs)(n.p,{children:["To get started, create a new ",(0,t.jsx)(n.code,{children:".yml"})," config file for your sub-page, placing it within ",(0,t.jsx)(n.code,{children:"/app/user-data"}),". Then within your primary ",(0,t.jsx)(n.code,{children:"conf.yml"}),", choose a name, and specify the path to the new file."]}),"\n",(0,t.jsx)(n.p,{children:"This is an example. Make sure to add this to the topmost line above appConfig:, or anywhere else appropriately, to match the yml syntax."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n- name: Networking Services\n path: 'networking.yml'\n- name: Work Stuff\n path: 'work.yml'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["The next step is to create the new page, if you mounted ",(0,t.jsx)(n.code,{children:"/app/user-data"})," in the docker command and not a volume, you can simply create the new page into that folder on the host."]}),"\n",(0,t.jsxs)(n.p,{children:["If you mounted ",(0,t.jsx)(n.code,{children:"/app/user-data/conf.yml"})," in docker, you can either switch to the volume or create another bind mount to your new additional page."]}),"\n",(0,t.jsxs)(n.p,{children:["If you're sub-page is located within ",(0,t.jsx)(n.code,{children:"/app/user-data"}),", then you only need to specify the filename, but if it's anywhere else, then the full path is required."]}),"\n",(0,t.jsxs)(n.p,{children:["A default template a page can be found here: ",(0,t.jsx)(n.a,{href:"https://github.com/lissy93/dashy/blob/master/user-data/conf.yml",children:"https://github.com/lissy93/dashy/blob/master/user-data/conf.yml"})," Keep in mind the appConfig cannot be used on subpages and should be removed, for further info see ",(0,t.jsx)(n.a,{href:"#restrictions",children:"Restrictions"})]}),"\n",(0,t.jsx)(n.p,{children:"Now if you reload the page, on the top right there should be a new button to navigate to the new page. \ud83c\udf89"}),"\n",(0,t.jsx)(n.h3,{id:"using-remote-sub-pages",children:"Using Remote Sub-Pages"}),"\n",(0,t.jsx)(n.p,{children:"Config files don't need to be local, you can store them anywhere, and data will be imported as sub-pages on page load."}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n- name: Getting Started\n path: 'https://snippet.host/tvcw/raw'\n- name: Homelab\n path: 'https://snippet.host/tetp/raw'\n- name: Browser Startpage\n path: 'https://snippet.host/zcom/raw'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["There are many options of how this can be used. You could store your config within a Git repository, in order to easily track and rollback changes. Or host your config on your NAS, to have it backed up with the rest of your files. Or use a hosted paste service, for example ",(0,t.jsx)(n.a,{href:"https://snippet.host/",children:"snippet.host"}),", which supports never-expiring CORS-enabled pastes, which can also be edited later."]}),"\n",(0,t.jsx)(n.p,{children:"You will obviously not be able to write updates to remote configs directly through the UI editor, but you can still make and preview changes, then use the export menu to get a copy of the new config, which can then be pasted to the remote source manually.\nThe config file must, of course be accessible from within Dashy. If your config contains sensitive info (like API keys, credentials, secret URLs, etc), take care not to expose it to the internet."}),"\n",(0,t.jsxs)(n.p,{children:["The following example shows creating a config, publishing it as a ",(0,t.jsx)(n.a,{href:"https://gist.github.com/",children:"Gist"}),", copying the URL to the raw file, and using it within your dashboard."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{width:"700",alt:"Public config in a gist demo",src:"https://i.ibb.co/55jm3LG/how-to-use-remote-config-sub-page.gif"})}),"\n",(0,t.jsx)(n.h3,{id:"restrictions",children:"Restrictions"}),"\n",(0,t.jsxs)(n.p,{children:["Only top-level fields supported by sub-pages are ",(0,t.jsx)(n.code,{children:"pageInfo"})," and ",(0,t.jsx)(n.code,{children:"sections"}),". The ",(0,t.jsx)(n.code,{children:"appConfig"})," and ",(0,t.jsx)(n.code,{children:"pages"})," will always be inherited from your main ",(0,t.jsx)(n.code,{children:"conf.yml"})," file. Other than that, sub-pages behave exactly the same as your default view, and can contain sections, items, widgets and page info like nav links, title and logo."]}),"\n",(0,t.jsx)(n.h3,{id:"url-structure",children:"URL Structure"}),"\n",(0,t.jsx)(n.p,{children:"Every view in Dashy shares the same URL shape, so any config can be reached from any view:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"/ Landing \u2014 whichever view you set as the default\n/ Root config in \n// Sub-config in \n///
Single section of in \n//main/
Single section of the root config (main is a reserved page id)\n"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:""})," is one of ",(0,t.jsx)(n.code,{children:"home"}),", ",(0,t.jsx)(n.code,{children:"minimal"}),", or ",(0,t.jsx)(n.code,{children:"workspace"}),". ",(0,t.jsx)(n.code,{children:""})," is the sub-config id \u2014 the sub-page's ",(0,t.jsx)(n.code,{children:"name"})," (from the ",(0,t.jsx)(n.code,{children:"pages"})," array) converted to lowercase-and-dashes (emoji and other non-word characters stripped). ",(0,t.jsx)(n.code,{children:"
"})," follows the same slugging rules. Workspace has no single-section URL; it uses its sidebar instead."]}),"\n",(0,t.jsx)(n.p,{children:"Examples:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:'/home Home view, main config\n/home/homelab Home view, "Homelab" sub-config\n/home/homelab/media Home view, "Homelab" sub-config, "Media" section only\n/home/main/getting-started Home view, main config, "Getting Started" section only\n/minimal/homelab Minimal view, "Homelab" sub-config\n/minimal/homelab/media Minimal view, "Homelab" sub-config, "Media" section pre-selected\n/workspace/homelab Workspace view, "Homelab" sub-config\n'})}),"\n",(0,t.jsx)(n.p,{children:'The view switcher, sub-page nav links, and section deep-links all preserve your current view and sub-page \u2014 so clicking through a single-section view and then hitting "back to all" returns you to the same sub-page you came from.'}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"layout",children:"Layout"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"appConfig.layout"})," controls how your sections sit on the page:"]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"auto"})," (default): responsive CSS grid, sections size by ",(0,t.jsx)(n.code,{children:"displayData.cols"}),"/",(0,t.jsx)(n.code,{children:"rows"})]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"horizontal"}),": sections stacked top to bottom, each full width"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"vertical"}),": sections side by side in columns"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"masonry"}),": responsive grid where heights follow content so shorter sections fill gaps under taller ones (",(0,t.jsx)(n.code,{children:"rows"})," is ignored)"]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["You can also switch between these from the settings menu. Add ",(0,t.jsx)(n.code,{children:"appConfig.colCount"})," to force a specific number of columns."]}),"\n",(0,t.jsx)(n.h3,{id:"making-a-section-wider-or-taller",children:"Making a section wider or taller"}),"\n",(0,t.jsxs)(n.p,{children:["Use ",(0,t.jsx)(n.code,{children:"displayData.cols"})," (1 to 5) and ",(0,t.jsx)(n.code,{children:"displayData.rows"})," (1 to 5) to control a section's footprint:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Important Links\n displayData:\n cols: 2\n rows: 2\n collapsed: false\n items: [...]\n"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"rows"})," is honoured by the ",(0,t.jsx)(n.code,{children:"auto"})," layout. Under ",(0,t.jsx)(n.code,{children:"masonry"}),", section heights always follow content."]}),"\n",(0,t.jsx)(n.h3,{id:"items-inside-a-section",children:"Items inside a section"}),"\n",(0,t.jsxs)(n.p,{children:["Items wrap responsively by default. The useful knobs on ",(0,t.jsx)(n.code,{children:"displayData"}),":"]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"itemSize"}),": ",(0,t.jsx)(n.code,{children:"small"}),", ",(0,t.jsx)(n.code,{children:"medium"})," (default), or ",(0,t.jsx)(n.code,{children:"large"})," (large tiles also show a description)"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"sortBy"}),": ",(0,t.jsx)(n.code,{children:"alphabetical"}),", ",(0,t.jsx)(n.code,{children:"reverse-alphabetical"}),", ",(0,t.jsx)(n.code,{children:"most-used"}),", ",(0,t.jsx)(n.code,{children:"last-used"}),", ",(0,t.jsx)(n.code,{children:"random"})]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"sectionLayout: grid"})," plus ",(0,t.jsx)(n.code,{children:"itemCountX"})," and/or ",(0,t.jsx)(n.code,{children:"itemCountY"})," if you want a fixed grid instead of auto wrapping"]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["See ",(0,t.jsx)(n.a,{href:"/docs/configuring#sectiondisplaydata-optional",children:"configuring.md"})," for the full list of options."]}),"\n",(0,t.jsx)(n.h3,{id:"sub-items",children:"Sub-Items"}),"\n",(0,t.jsx)(n.p,{children:"A normal section will contain zero or more items, for example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Coding\n icon: far fa-code\n items:\n - title: GitHub\n url: https://github.com/\n - title: StackOverflow\n url: http://stackoverflow.com/\n"})}),"\n",(0,t.jsx)(n.p,{children:"But items can also be grouped together, referred to as sub-items. This is useful for a group of less frequently used items, which you don't want to take up too much space."}),"\n",(0,t.jsx)(n.p,{children:"Item groups may also have an optional title."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Coding\n icon: far fa-code\n items:\n - title: Normal Item 1\n - title: Normal Item 2\n - title: Languages\n subItems:\n - title: JavaScript\n url: https://developer.mozilla.org\n icon: si-javascript\n - title: TypeScript\n url: https://www.typescriptlang.org/docs\n icon: si-typescript\n - title: Svelte\n url: https://svelte.dev/docs\n icon: si-svelte\n - title: Go\n url: https://go.dev/doc\n icon: si-go\n"})})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>a,x:()=>r});var i=s(6540);const t={},o=i.createContext(t);function a(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:a(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[146],{9327(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>r,default:()=>h,frontMatter:()=>a,metadata:()=>i,toc:()=>c});const i=JSON.parse('{"id":"pages-and-sections","title":"Pages and Sections","description":"Page Metadata","source":"@site/docs/pages-and-sections.md","sourceDirName":".","slug":"/pages-and-sections","permalink":"/docs/pages-and-sections","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/pages-and-sections.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Cloud Backup and Restore","permalink":"/docs/backup-restore"},"next":{"title":"*Dashy Showcase* \ud83c\udf1f","permalink":"/docs/showcase"}}');var t=s(4848),o=s(8453);const a={},r="Pages and Sections",l={},c=[{value:"Page Metadata",id:"page-metadata",level:2},{value:"Multi-Page Support",id:"multi-page-support",level:2},{value:"Using Local Sub-Pages",id:"using-local-sub-pages",level:3},{value:"Using Remote Sub-Pages",id:"using-remote-sub-pages",level:3},{value:"Restrictions",id:"restrictions",level:3},{value:"URL Structure",id:"url-structure",level:3},{value:"Layout",id:"layout",level:2},{value:"Making a section wider or taller",id:"making-a-section-wider-or-taller",level:3},{value:"Items inside a section",id:"items-inside-a-section",level:3},{value:"Sub-Items",id:"sub-items",level:3}];function d(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"pages-and-sections",children:"Pages and Sections"})}),"\n",(0,t.jsx)(n.h2,{id:"page-metadata",children:"Page Metadata"}),"\n",(0,t.jsxs)(n.p,{children:["Set your dashboard's branding under ",(0,t.jsx)(n.code,{children:"pageInfo"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pageInfo:\n title: My Dashboard # Used for main h1 title, and browser tab text\n description: Home server links\n logo: /web-icons/my-logo.png # path/URL to optional logo to display next to title\n favicon: 'https://example.com/path/to/icon' # path/URL to a favicon (shows in browser tab)\n color: '#2a7cf0' # Hex color, to set the browser/address bar color on mobile (supported browsers only)\n footer: '\xa9 2026 Me' # Optional text or HTML content, to display in the pages footer\n"})}),"\n",(0,t.jsx)(n.p,{children:"If you have multiple configs/pages, then these values swap automatically as you navigate between sub-pages."}),"\n",(0,t.jsxs)(n.p,{children:["The only caveat being, if you install Dashy as a PWA, the installed app's name, icon, and splash-screen colour come from the bundled ",(0,t.jsx)(n.code,{children:"manifest.webmanifest"})," (baked at build time) rather than ",(0,t.jsx)(n.code,{children:"pageInfo"}),". Runtime values only apply when browsing in a regular tab or browser."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"multi-page-support",children:"Multi-Page Support"}),"\n",(0,t.jsx)(n.p,{children:"You can have additional pages within your dashboard, with each having it's own config file. The config files for sub-pages can either be stored locally, or hosted separately. A link to each additional page will be displayed in the navigation bar."}),"\n",(0,t.jsx)(n.p,{children:"You can edit additional pages using the interactive editor, exactly the same was as your primary page (so long as it's local). But please save changes to one page, before you start editing the next."}),"\n",(0,t.jsx)(n.h3,{id:"using-local-sub-pages",children:"Using Local Sub-Pages"}),"\n",(0,t.jsxs)(n.p,{children:["To get started, create a new ",(0,t.jsx)(n.code,{children:".yml"})," config file for your sub-page, placing it within ",(0,t.jsx)(n.code,{children:"/app/user-data"}),". Then within your primary ",(0,t.jsx)(n.code,{children:"conf.yml"}),", choose a name, and specify the path to the new file."]}),"\n",(0,t.jsx)(n.p,{children:"This is an example. Make sure to add this to the topmost line above appConfig:, or anywhere else appropriately, to match the yml syntax."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n- name: Networking Services\n path: 'networking.yml'\n- name: Work Stuff\n path: 'work.yml'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["The next step is to create the new page, if you mounted ",(0,t.jsx)(n.code,{children:"/app/user-data"})," in the docker command and not a volume, you can simply create the new page into that folder on the host."]}),"\n",(0,t.jsxs)(n.p,{children:["If you mounted ",(0,t.jsx)(n.code,{children:"/app/user-data/conf.yml"})," in docker, you can either switch to the volume or create another bind mount to your new additional page."]}),"\n",(0,t.jsxs)(n.p,{children:["If you're sub-page is located within ",(0,t.jsx)(n.code,{children:"/app/user-data"}),", then you only need to specify the filename, but if it's anywhere else, then the full path is required."]}),"\n",(0,t.jsxs)(n.p,{children:["A default template a page can be found here: ",(0,t.jsx)(n.a,{href:"https://github.com/lissy93/dashy/blob/master/user-data/conf.yml",children:"https://github.com/lissy93/dashy/blob/master/user-data/conf.yml"})," Keep in mind the appConfig cannot be used on subpages and should be removed, for further info see ",(0,t.jsx)(n.a,{href:"#restrictions",children:"Restrictions"})]}),"\n",(0,t.jsx)(n.p,{children:"Now if you reload the page, on the top right there should be a new button to navigate to the new page. \ud83c\udf89"}),"\n",(0,t.jsx)(n.h3,{id:"using-remote-sub-pages",children:"Using Remote Sub-Pages"}),"\n",(0,t.jsx)(n.p,{children:"Config files don't need to be local, you can store them anywhere, and data will be imported as sub-pages on page load."}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n- name: Getting Started\n path: 'https://snippet.host/tvcw/raw'\n- name: Homelab\n path: 'https://snippet.host/tetp/raw'\n- name: Browser Startpage\n path: 'https://snippet.host/zcom/raw'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["There are many options of how this can be used. You could store your config within a Git repository, in order to easily track and rollback changes. Or host your config on your NAS, to have it backed up with the rest of your files. Or use a hosted paste service, for example ",(0,t.jsx)(n.a,{href:"https://snippet.host/",children:"snippet.host"}),", which supports never-expiring CORS-enabled pastes, which can also be edited later."]}),"\n",(0,t.jsx)(n.p,{children:"You will obviously not be able to write updates to remote configs directly through the UI editor, but you can still make and preview changes, then use the export menu to get a copy of the new config, which can then be pasted to the remote source manually.\nThe config file must, of course be accessible from within Dashy. If your config contains sensitive info (like API keys, credentials, secret URLs, etc), take care not to expose it to the internet."}),"\n",(0,t.jsxs)(n.p,{children:["The following example shows creating a config, publishing it as a ",(0,t.jsx)(n.a,{href:"https://gist.github.com/",children:"Gist"}),", copying the URL to the raw file, and using it within your dashboard."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{width:"700",alt:"Public config in a gist demo",src:"https://i.ibb.co/55jm3LG/how-to-use-remote-config-sub-page.gif"})}),"\n",(0,t.jsx)(n.h3,{id:"restrictions",children:"Restrictions"}),"\n",(0,t.jsxs)(n.p,{children:["Only top-level fields supported by sub-pages are ",(0,t.jsx)(n.code,{children:"pageInfo"})," and ",(0,t.jsx)(n.code,{children:"sections"}),". The ",(0,t.jsx)(n.code,{children:"appConfig"})," and ",(0,t.jsx)(n.code,{children:"pages"})," will always be inherited from your main ",(0,t.jsx)(n.code,{children:"conf.yml"})," file. Other than that, sub-pages behave exactly the same as your default view, and can contain sections, items, widgets and page info like nav links, title and logo."]}),"\n",(0,t.jsx)(n.h3,{id:"url-structure",children:"URL Structure"}),"\n",(0,t.jsx)(n.p,{children:"Every view in Dashy shares the same URL shape, so any config can be reached from any view:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"/ Landing \u2014 whichever view you set as the default\n/ Root config in \n// Sub-config in \n///
Single section of in \n//main/
Single section of the root config (main is a reserved page id)\n"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:""})," is one of ",(0,t.jsx)(n.code,{children:"home"}),", ",(0,t.jsx)(n.code,{children:"minimal"}),", or ",(0,t.jsx)(n.code,{children:"workspace"}),". ",(0,t.jsx)(n.code,{children:""})," is the sub-config id \u2014 the sub-page's ",(0,t.jsx)(n.code,{children:"name"})," (from the ",(0,t.jsx)(n.code,{children:"pages"})," array) converted to lowercase-and-dashes (emoji and other non-word characters stripped). ",(0,t.jsx)(n.code,{children:"
"})," follows the same slugging rules. Workspace has no single-section URL; it uses its sidebar instead."]}),"\n",(0,t.jsx)(n.p,{children:"Examples:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:'/home Home view, main config\n/home/homelab Home view, "Homelab" sub-config\n/home/homelab/media Home view, "Homelab" sub-config, "Media" section only\n/home/main/getting-started Home view, main config, "Getting Started" section only\n/minimal/homelab Minimal view, "Homelab" sub-config\n/minimal/homelab/media Minimal view, "Homelab" sub-config, "Media" section pre-selected\n/workspace/homelab Workspace view, "Homelab" sub-config\n'})}),"\n",(0,t.jsx)(n.p,{children:'The view switcher, sub-page nav links, and section deep-links all preserve your current view and sub-page \u2014 so clicking through a single-section view and then hitting "back to all" returns you to the same sub-page you came from.'}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"layout",children:"Layout"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"appConfig.layout"})," controls how your sections sit on the page:"]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"auto"})," (default): responsive CSS grid, sections size by ",(0,t.jsx)(n.code,{children:"displayData.cols"}),"/",(0,t.jsx)(n.code,{children:"rows"})]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"horizontal"}),": sections stacked top to bottom, each full width"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"vertical"}),": sections side by side in columns"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"masonry"}),": responsive grid where heights follow content so shorter sections fill gaps under taller ones (",(0,t.jsx)(n.code,{children:"rows"})," is ignored)"]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["You can also switch between these from the settings menu. Add ",(0,t.jsx)(n.code,{children:"appConfig.colCount"})," to force a specific number of columns."]}),"\n",(0,t.jsx)(n.h3,{id:"making-a-section-wider-or-taller",children:"Making a section wider or taller"}),"\n",(0,t.jsxs)(n.p,{children:["Use ",(0,t.jsx)(n.code,{children:"displayData.cols"})," (1 to 5) and ",(0,t.jsx)(n.code,{children:"displayData.rows"})," (1 to 5) to control a section's footprint:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Important Links\n displayData:\n cols: 2\n rows: 2\n collapsed: false\n items: [...]\n"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"rows"})," is honoured by the ",(0,t.jsx)(n.code,{children:"auto"})," layout. Under ",(0,t.jsx)(n.code,{children:"masonry"}),", section heights always follow content."]}),"\n",(0,t.jsx)(n.h3,{id:"items-inside-a-section",children:"Items inside a section"}),"\n",(0,t.jsxs)(n.p,{children:["Items wrap responsively by default. The useful knobs on ",(0,t.jsx)(n.code,{children:"displayData"}),":"]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"itemSize"}),": ",(0,t.jsx)(n.code,{children:"small"}),", ",(0,t.jsx)(n.code,{children:"medium"})," (default), or ",(0,t.jsx)(n.code,{children:"large"})," (large tiles also show a description)"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"sortBy"}),": ",(0,t.jsx)(n.code,{children:"alphabetical"}),", ",(0,t.jsx)(n.code,{children:"reverse-alphabetical"}),", ",(0,t.jsx)(n.code,{children:"most-used"}),", ",(0,t.jsx)(n.code,{children:"last-used"}),", ",(0,t.jsx)(n.code,{children:"random"})]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"sectionLayout: grid"})," plus ",(0,t.jsx)(n.code,{children:"itemCountX"})," and/or ",(0,t.jsx)(n.code,{children:"itemCountY"})," if you want a fixed grid instead of auto wrapping"]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["See ",(0,t.jsx)(n.a,{href:"/docs/configuring#sectiondisplaydata-optional",children:"configuring.md"})," for the full list of options."]}),"\n",(0,t.jsx)(n.h3,{id:"sub-items",children:"Sub-Items"}),"\n",(0,t.jsx)(n.p,{children:"A normal section will contain zero or more items, for example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Coding\n icon: far fa-code\n items:\n - title: GitHub\n url: https://github.com/\n - title: StackOverflow\n url: http://stackoverflow.com/\n"})}),"\n",(0,t.jsx)(n.p,{children:"But items can also be grouped together, referred to as sub-items. This is useful for a group of less frequently used items, which you don't want to take up too much space."}),"\n",(0,t.jsx)(n.p,{children:"Item groups may also have an optional title."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Coding\n icon: far fa-code\n items:\n - title: Normal Item 1\n - title: Normal Item 2\n - title: Languages\n subItems:\n - title: JavaScript\n url: https://developer.mozilla.org\n icon: si-javascript\n - title: TypeScript\n url: https://www.typescriptlang.org/docs\n icon: si-typescript\n - title: Svelte\n url: https://svelte.dev/docs\n icon: si-svelte\n - title: Go\n url: https://go.dev/doc\n icon: si-go\n"})})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>a,x:()=>r});var i=s(6540);const t={},o=i.createContext(t);function a(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function r(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:a(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/27d9d47d.7d09ecaa.js b/assets/js/27d9d47d.77650063.js similarity index 99% rename from assets/js/27d9d47d.7d09ecaa.js rename to assets/js/27d9d47d.77650063.js index a83d4b55..969a7de8 100644 --- a/assets/js/27d9d47d.7d09ecaa.js +++ b/assets/js/27d9d47d.77650063.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9757],{7177(s,e,n){n.r(e),n.d(e,{assets:()=>o,contentTitle:()=>a,default:()=>l,frontMatter:()=>i,metadata:()=>r,toc:()=>d});const r=JSON.parse('{"id":"showcase","title":"*Dashy Showcase* \ud83c\udf1f","description":"| \ud83d\udc97 Got a sweet dashboard? Submit it to the showcase! \ud83d\udc49 See How |","source":"@site/docs/showcase.md","sourceDirName":".","slug":"/showcase","permalink":"/docs/showcase","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/showcase.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Pages and Sections","permalink":"/docs/pages-and-sections"},"next":{"title":"Contributing","permalink":"/docs/contributing"}}');var h=n(4848),t=n(8453);const i={},a="Dashy Showcase \ud83c\udf1f",o={},d=[{value:"MNDashboard",id:"mndashboard",level:2},{value:"Home Lab 2.0",id:"home-lab-20",level:2},{value:"Dipan's Dash",id:"dipans-dash",level:2},{value:"Ratty222",id:"ratty222",level:2},{value:"Hugalafutro Dashy",id:"hugalafutro-dashy",level:2},{value:"NAS Home Dashboard",id:"nas-home-dashboard",level:2},{value:"Brewhack",id:"brewhack",level:2},{value:"The Dragons Lair",id:"the-dragons-lair",level:2},{value:"Homelab & VPS dashboard",id:"homelab--vps-dashboard",level:2},{value:"Raspberry PI Docker Dashboard",id:"raspberry-pi-docker-dashboard",level:2},{value:"First Week of Self-Hosting",id:"first-week-of-self-hosting",level:2},{value:"EVO Dashboard",id:"evo-dashboard",level:2},{value:"The Private Dashboard",id:"the-private-dashboard",level:2},{value:"Networking Services",id:"networking-services",level:2},{value:"Dashy Live",id:"dashy-live",level:2},{value:"System Monitor",id:"system-monitor",level:2},{value:"Browser Startpage",id:"browser-startpage",level:2},{value:"CFT Toolbox",id:"cft-toolbox",level:2},{value:"Bookmarks",id:"bookmarks",level:2},{value:"Project Management",id:"project-management",level:2},{value:"Dashy Example",id:"dashy-example",level:2},{value:"HomeLAb 3.0",id:"homelab-30",level:2},{value:"Ground Control",id:"ground-control",level:2},{value:"Croco_Grievous",id:"croco_grievous",level:2},{value:"Crypto Dash",id:"crypto-dash",level:2},{value:"Stefantigro",id:"stefantigro",level:2},{value:"HomeLab 3.0",id:"homelab-30-1",level:2},{value:"Morning Dashboard",id:"morning-dashboard",level:2},{value:"Yet Another Homelab",id:"yet-another-homelab",level:2},{value:"Submitting your Dashboard",id:"submitting-your-dashboard",level:2},{value:"How to Submit",id:"how-to-submit",level:3},{value:"What to Include",id:"what-to-include",level:3},{value:"Template",id:"template",level:3}];function c(s){const e={a:"a",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",sup:"sup",table:"table",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...s.components};return(0,h.jsxs)(h.Fragment,{children:[(0,h.jsx)(e.header,{children:(0,h.jsxs)(e.h1,{id:"dashy-showcase-",children:[(0,h.jsx)(e.em,{children:"Dashy Showcase"})," \ud83c\udf1f"]})}),"\n\n\n\n\n\n\n",(0,h.jsx)(e.table,{children:(0,h.jsx)(e.thead,{children:(0,h.jsx)(e.tr,{children:(0,h.jsxs)(e.th,{children:["\ud83d\udc97 Got a sweet dashboard? Submit it to the showcase! \ud83d\udc49 ",(0,h.jsx)(e.a,{href:"#submitting-your-dashboard",children:"See How"})]})})})}),"\n",(0,h.jsx)(e.h2,{id:"mndashboard",children:"MNDashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/mahrnet",children:"@mahrnet"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/1491",children:"#1491"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/mCJRZgp/d2-At-QO4c-PT4u.png",alt:"screenshot-MNDashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"home-lab-20",children:"Home Lab 2.0"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/1-home-lab-material.png",alt:"screenshot-homelab"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dipans-dash",children:"Dipan's Dash"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dipanghosh",children:"@Dipanghosh"})," - ",(0,h.jsx)(e.a,{href:"https://dipan.de/",children:"dipan.de"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/777",children:"#777"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["With the option of embedding iframes, it becomes possible to combine and conquer. I have set up Dashy to include graphs from grafana, and set them so that they update automatically. I also have uptime monitors from uptime Kuma, panels from homeassistant are also possible to be included.",(0,h.jsx)(e.br,{}),"\nDashy is awesome!"]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/bPC6w2n/dipanghosh-dashboard.png",alt:"screenshot-dipanghosh-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"ratty222",children:"Ratty222"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/ratty222",children:"@ratty222"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/384",children:"#384"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/147582551-4c655d37-8bcc-4f95-ab41-164a9d0d6a07.png",alt:"screenshot-ratty222-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"hugalafutro-dashy",children:"Hugalafutro Dashy"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/hugalafutro",children:"@hugalafutro"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/505",children:"#505"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.a,{href:"https://i.ibb.co/PDpLDKS/hugalafutro-dashy.gif",children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/PDpLDKS/hugalafutro-dashy.gif",alt:"hugalafutro-dashy-screenshot"})})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"nas-home-dashboard",children:"NAS Home Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/cerealconyogurt",children:"@cerealconyogurt"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/74",children:"#74"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/6-nas-home-dashboard.png",alt:"screenshot-networking-services"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"brewhack",children:"Brewhack"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/brpeterso",children:"@brpeterso"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/680",children:"#680"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/cNjzPT4/brewhack.png",alt:"screenshot-brewhack-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"the-dragons-lair",children:"The Dragons Lair"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dr460nf1r3",children:"dr460nf1r3"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/684",children:"#684"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"My new startpage featuring the stuff I use most. And of course dragons!"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/13-dragons-lair.png",alt:"screenshot-dragons-lair"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab--vps-dashboard",children:"Homelab & VPS dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/shadowking001",children:"@shadowking001"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/86",children:"#86"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/8-shadowking001s-dashy.png",alt:"screenshot-shadowking001-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"raspberry-pi-docker-dashboard",children:"Raspberry PI Docker Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/henkiewie",children:"@henkiewie"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/622",children:"#622"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["I use this dashboard every day. It now even includes a player for a radio stream which I configured with Logitech media server and icecast. I made an smaller version of the grafana dashboard to fit an iframe in kiosk mode, so it monitors the most important values of my RPI. The PI is in Argon m2 case and used as a NAS. The dashboard is a copy of the adventure theme with some changes saved in ",(0,h.jsx)(e.code,{children:"/app/src/styles/user-defined-themes.scss"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/jGzPm6b/henkiewie-dashy-showcase.png",alt:"screenshot-henkiewie-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"first-week-of-self-hosting",children:"First Week of Self-Hosting"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/RickyCZ",children:"u//RickyCZ"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/pose15/just_got_started_a_week_ago_selfhosting_is_very/",children:"Reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/11-ricky-cz.png",alt:"screenshot-week-of-self-hosting"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"evo-dashboard",children:"EVO Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/EVOTk",children:"@EVOTk"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/pull/316",children:"#316"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/12-evo-dashboard.png",alt:"screenshot-evo-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"the-private-dashboard",children:"The Private Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/DylanBeMe",children:"@DylanBeMe"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/419",children:"#419"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/hKS483T/private-dashboard-Dylan-Be-Me.png",alt:"screenshot-private-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"networking-services",children:"Networking Services"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/2-networking-services-minimal-dark.png",alt:"screenshot-networking-services"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dashy-live",children:"Dashy Live"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["A dashboard I made to manage all project development links from one place. View demo at ",(0,h.jsx)(e.a,{href:"https://live.dashy.to/",children:"live.dashy.to"}),"."]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/10-dashy-live.png",alt:"screenshot-dashy-live"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"system-monitor",children:"System Monitor"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"An aggregated board for monitoring system resource usage from a single view"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/xfK6BGb/system-monitor-board.png",alt:"screenshot-monitor"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"browser-startpage",children:"Browser Startpage"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/rs07dS1/startpage.png",alt:"screenshot-startpage"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"cft-toolbox",children:"CFT Toolbox"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/3-cft-toolbox.png",alt:"screenshot-cft-toolbox"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"bookmarks",children:"Bookmarks"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/4-bookmarks-colourful.png",alt:"screenshot-bookmarks"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"project-management",children:"Project Management"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/5-project-management.png",alt:"screenshot-project-management"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dashy-example",children:"Dashy Example"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["An example dashboard, by ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"}),". View live at ",(0,h.jsx)(e.a,{href:"https://demo.dashy.to/",children:"demo.dashy.to"}),"."]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/YbzqPK7/demo-dashy.png",alt:"screenshot-dashy-example"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab-30",children:"HomeLAb 3.0"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/skoogee",children:"@skoogee"})," (",(0,h.jsx)(e.a,{href:"http://zhrn.cc",children:"http://zhrn.cc"}),") ",(0,h.jsx)(e.sup,{children:(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/279",children:"#279"})})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"Dashy, is the most complete dashboard I ever tried, has all the features, and it sets itself apart from the rest. It is my default homepage now. I am thankful to the developer @Lissy93 for sharing such a wonderful creation."}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.a,{href:"https://ibb.co/album/ynSwzm",children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/F5yBTsT/12-skoogee-homelab-3.png?",alt:"screenshot-12-skoogee-homelab-3"})})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"ground-control",children:"Ground Control"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dtctek",children:"@dtctek"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/83",children:"#83"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/149821995-e9b41dab-186c-42e6-b5b3-e233259b241d.png",alt:"screenshot-ground-control"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"croco_grievous",children:"Croco_Grievous"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/Croco_Grievous/",children:"u/Croco_Grievous"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/t4xk3z/everything_started_with_pihole_on_a_raspberry_pi/",children:"reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/59XR8KL/dashy-Croco-Grievous.png",alt:"screenshot-croco-grievous-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"crypto-dash",children:"Crypto Dash"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["Example usage of widgets to monitor cryptocurrencies news, prices and data. Config is ",(0,h.jsx)(e.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10#file-example-8-dashy-crypto-widgets-conf-yml",children:"available here"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/147394584-352fe3bf-740d-4624-a01b-9003a97bc832.png",alt:"screenshot-crypto-dash"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"stefantigro",children:"Stefantigro"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/stefantigro/",children:"u/stefantigro"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/t5oril/been_selfhosting_close_to_half_a_year_now_all/",children:"reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/1Kb43Yy/dashy-stefantigro.png",alt:"screenshot-stefantigro-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab-30-1",children:"HomeLab 3.0"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/1999",children:"#1999"})]})}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://pixelflare.cc/alicia/dashy/12-skoogee-homelab-3",alt:"screenshot-homelab-3"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"morning-dashboard",children:"Morning Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"Displayed on my smart screen between 05:00 - 08:00, and includes all the info that I usually check before leaving for work"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/4Wx8zb7/morning-dashboard.png",alt:"screenshot-morning-dash"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"yet-another-homelab",children:"Yet Another Homelab"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/9-home-lab-oblivion.png",alt:"screenshot-yet-another-homelab"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"submitting-your-dashboard",children:"Submitting your Dashboard"}),"\n",(0,h.jsx)(e.h3,{id:"how-to-submit",children:"How to Submit"}),"\n",(0,h.jsxs)(e.ul,{children:["\n",(0,h.jsx)(e.li,{children:(0,h.jsx)(e.a,{href:"https://git.io/JEtgM",children:"Open an Issue"})}),"\n",(0,h.jsx)(e.li,{children:(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/compare",children:"Open a PR"})}),"\n"]}),"\n",(0,h.jsx)(e.h3,{id:"what-to-include",children:"What to Include"}),"\n",(0,h.jsx)(e.p,{children:"Please include the following information:"}),"\n",(0,h.jsxs)(e.ul,{children:["\n",(0,h.jsx)(e.li,{children:"A single high-quality screenshot of your Dashboard"}),"\n",(0,h.jsx)(e.li,{children:"A short title (it doesn't have to be particularly imaginative)"}),"\n",(0,h.jsx)(e.li,{children:"An optional description, you could include details on anything interesting or unique about your dashboard, or say how you use it, and why it's awesome"}),"\n",(0,h.jsx)(e.li,{children:"Optionally leave your name or username, with a link to your GitHub, Twitter or Website"}),"\n"]}),"\n",(0,h.jsx)(e.h3,{id:"template",children:"Template"}),"\n",(0,h.jsx)(e.p,{children:"If you're submitting a pull request, please use a format similar to this:"}),"\n",(0,h.jsx)(e.pre,{children:(0,h.jsx)(e.code,{className:"language-text",children:"### [Dashboard Name] (required)\n\n> Submitted by [@username](https://github.com/user) (optional)\n\n[An optional text description, or any interesting details] (optional)\n\n![dashboard-screenshot](https://example.com/url-to-screenshot.png) (required)\n\n---\n\n"})})]})}function l(s={}){const{wrapper:e}={...(0,t.R)(),...s.components};return e?(0,h.jsx)(e,{...s,children:(0,h.jsx)(c,{...s})}):c(s)}},8453(s,e,n){n.d(e,{R:()=>i,x:()=>a});var r=n(6540);const h={},t=r.createContext(h);function i(s){const e=r.useContext(t);return r.useMemo(function(){return"function"==typeof s?s(e):{...e,...s}},[e,s])}function a(s){let e;return e=s.disableParentContext?"function"==typeof s.components?s.components(h):s.components||h:i(s.components),r.createElement(t.Provider,{value:e},s.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9757],{7177(s,e,n){n.r(e),n.d(e,{assets:()=>o,contentTitle:()=>a,default:()=>l,frontMatter:()=>i,metadata:()=>r,toc:()=>d});const r=JSON.parse('{"id":"showcase","title":"*Dashy Showcase* \ud83c\udf1f","description":"| \ud83d\udc97 Got a sweet dashboard? Submit it to the showcase! \ud83d\udc49 See How |","source":"@site/docs/showcase.md","sourceDirName":".","slug":"/showcase","permalink":"/docs/showcase","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/showcase.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Pages and Sections","permalink":"/docs/pages-and-sections"},"next":{"title":"Contributing","permalink":"/docs/contributing"}}');var h=n(4848),t=n(8453);const i={},a="Dashy Showcase \ud83c\udf1f",o={},d=[{value:"MNDashboard",id:"mndashboard",level:2},{value:"Home Lab 2.0",id:"home-lab-20",level:2},{value:"Dipan's Dash",id:"dipans-dash",level:2},{value:"Ratty222",id:"ratty222",level:2},{value:"Hugalafutro Dashy",id:"hugalafutro-dashy",level:2},{value:"NAS Home Dashboard",id:"nas-home-dashboard",level:2},{value:"Brewhack",id:"brewhack",level:2},{value:"The Dragons Lair",id:"the-dragons-lair",level:2},{value:"Homelab & VPS dashboard",id:"homelab--vps-dashboard",level:2},{value:"Raspberry PI Docker Dashboard",id:"raspberry-pi-docker-dashboard",level:2},{value:"First Week of Self-Hosting",id:"first-week-of-self-hosting",level:2},{value:"EVO Dashboard",id:"evo-dashboard",level:2},{value:"The Private Dashboard",id:"the-private-dashboard",level:2},{value:"Networking Services",id:"networking-services",level:2},{value:"Dashy Live",id:"dashy-live",level:2},{value:"System Monitor",id:"system-monitor",level:2},{value:"Browser Startpage",id:"browser-startpage",level:2},{value:"CFT Toolbox",id:"cft-toolbox",level:2},{value:"Bookmarks",id:"bookmarks",level:2},{value:"Project Management",id:"project-management",level:2},{value:"Dashy Example",id:"dashy-example",level:2},{value:"HomeLAb 3.0",id:"homelab-30",level:2},{value:"Ground Control",id:"ground-control",level:2},{value:"Croco_Grievous",id:"croco_grievous",level:2},{value:"Crypto Dash",id:"crypto-dash",level:2},{value:"Stefantigro",id:"stefantigro",level:2},{value:"HomeLab 3.0",id:"homelab-30-1",level:2},{value:"Morning Dashboard",id:"morning-dashboard",level:2},{value:"Yet Another Homelab",id:"yet-another-homelab",level:2},{value:"Submitting your Dashboard",id:"submitting-your-dashboard",level:2},{value:"How to Submit",id:"how-to-submit",level:3},{value:"What to Include",id:"what-to-include",level:3},{value:"Template",id:"template",level:3}];function c(s){const e={a:"a",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",sup:"sup",table:"table",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...s.components};return(0,h.jsxs)(h.Fragment,{children:[(0,h.jsx)(e.header,{children:(0,h.jsxs)(e.h1,{id:"dashy-showcase-",children:[(0,h.jsx)(e.em,{children:"Dashy Showcase"})," \ud83c\udf1f"]})}),"\n\n\n\n\n\n\n",(0,h.jsx)(e.table,{children:(0,h.jsx)(e.thead,{children:(0,h.jsx)(e.tr,{children:(0,h.jsxs)(e.th,{children:["\ud83d\udc97 Got a sweet dashboard? Submit it to the showcase! \ud83d\udc49 ",(0,h.jsx)(e.a,{href:"#submitting-your-dashboard",children:"See How"})]})})})}),"\n",(0,h.jsx)(e.h2,{id:"mndashboard",children:"MNDashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/mahrnet",children:"@mahrnet"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/1491",children:"#1491"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/mCJRZgp/d2-At-QO4c-PT4u.png",alt:"screenshot-MNDashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"home-lab-20",children:"Home Lab 2.0"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/1-home-lab-material.png",alt:"screenshot-homelab"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dipans-dash",children:"Dipan's Dash"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dipanghosh",children:"@Dipanghosh"})," - ",(0,h.jsx)(e.a,{href:"https://dipan.de/",children:"dipan.de"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/777",children:"#777"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["With the option of embedding iframes, it becomes possible to combine and conquer. I have set up Dashy to include graphs from grafana, and set them so that they update automatically. I also have uptime monitors from uptime Kuma, panels from homeassistant are also possible to be included.",(0,h.jsx)(e.br,{}),"\nDashy is awesome!"]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/bPC6w2n/dipanghosh-dashboard.png",alt:"screenshot-dipanghosh-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"ratty222",children:"Ratty222"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/ratty222",children:"@ratty222"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/384",children:"#384"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/147582551-4c655d37-8bcc-4f95-ab41-164a9d0d6a07.png",alt:"screenshot-ratty222-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"hugalafutro-dashy",children:"Hugalafutro Dashy"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/hugalafutro",children:"@hugalafutro"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/discussions/505",children:"#505"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.a,{href:"https://i.ibb.co/PDpLDKS/hugalafutro-dashy.gif",children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/PDpLDKS/hugalafutro-dashy.gif",alt:"hugalafutro-dashy-screenshot"})})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"nas-home-dashboard",children:"NAS Home Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/cerealconyogurt",children:"@cerealconyogurt"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/74",children:"#74"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/6-nas-home-dashboard.png",alt:"screenshot-networking-services"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"brewhack",children:"Brewhack"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/brpeterso",children:"@brpeterso"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/680",children:"#680"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/cNjzPT4/brewhack.png",alt:"screenshot-brewhack-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"the-dragons-lair",children:"The Dragons Lair"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dr460nf1r3",children:"dr460nf1r3"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/684",children:"#684"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"My new startpage featuring the stuff I use most. And of course dragons!"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/13-dragons-lair.png",alt:"screenshot-dragons-lair"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab--vps-dashboard",children:"Homelab & VPS dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/shadowking001",children:"@shadowking001"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/86",children:"#86"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/8-shadowking001s-dashy.png",alt:"screenshot-shadowking001-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"raspberry-pi-docker-dashboard",children:"Raspberry PI Docker Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/henkiewie",children:"@henkiewie"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/622",children:"#622"})]})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["I use this dashboard every day. It now even includes a player for a radio stream which I configured with Logitech media server and icecast. I made an smaller version of the grafana dashboard to fit an iframe in kiosk mode, so it monitors the most important values of my RPI. The PI is in Argon m2 case and used as a NAS. The dashboard is a copy of the adventure theme with some changes saved in ",(0,h.jsx)(e.code,{children:"/app/src/styles/user-defined-themes.scss"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/jGzPm6b/henkiewie-dashy-showcase.png",alt:"screenshot-henkiewie-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"first-week-of-self-hosting",children:"First Week of Self-Hosting"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/RickyCZ",children:"u//RickyCZ"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/pose15/just_got_started_a_week_ago_selfhosting_is_very/",children:"Reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/11-ricky-cz.png",alt:"screenshot-week-of-self-hosting"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"evo-dashboard",children:"EVO Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/EVOTk",children:"@EVOTk"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/pull/316",children:"#316"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/12-evo-dashboard.png",alt:"screenshot-evo-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"the-private-dashboard",children:"The Private Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/DylanBeMe",children:"@DylanBeMe"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/419",children:"#419"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/hKS483T/private-dashboard-Dylan-Be-Me.png",alt:"screenshot-private-dashboard"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"networking-services",children:"Networking Services"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/2-networking-services-minimal-dark.png",alt:"screenshot-networking-services"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dashy-live",children:"Dashy Live"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["A dashboard I made to manage all project development links from one place. View demo at ",(0,h.jsx)(e.a,{href:"https://live.dashy.to/",children:"live.dashy.to"}),"."]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/10-dashy-live.png",alt:"screenshot-dashy-live"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"system-monitor",children:"System Monitor"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"An aggregated board for monitoring system resource usage from a single view"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/xfK6BGb/system-monitor-board.png",alt:"screenshot-monitor"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"browser-startpage",children:"Browser Startpage"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/rs07dS1/startpage.png",alt:"screenshot-startpage"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"cft-toolbox",children:"CFT Toolbox"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/3-cft-toolbox.png",alt:"screenshot-cft-toolbox"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"bookmarks",children:"Bookmarks"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/4-bookmarks-colourful.png",alt:"screenshot-bookmarks"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"project-management",children:"Project Management"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/5-project-management.png",alt:"screenshot-project-management"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"dashy-example",children:"Dashy Example"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["An example dashboard, by ",(0,h.jsx)(e.a,{href:"https://github.com/lissy93",children:"@Lissy93"}),". View live at ",(0,h.jsx)(e.a,{href:"https://demo.dashy.to/",children:"demo.dashy.to"}),"."]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/YbzqPK7/demo-dashy.png",alt:"screenshot-dashy-example"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab-30",children:"HomeLAb 3.0"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/skoogee",children:"@skoogee"})," (",(0,h.jsx)(e.a,{href:"http://zhrn.cc",children:"http://zhrn.cc"}),") ",(0,h.jsx)(e.sup,{children:(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/279",children:"#279"})})]}),"\n"]}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"Dashy, is the most complete dashboard I ever tried, has all the features, and it sets itself apart from the rest. It is my default homepage now. I am thankful to the developer @Lissy93 for sharing such a wonderful creation."}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.a,{href:"https://ibb.co/album/ynSwzm",children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/F5yBTsT/12-skoogee-homelab-3.png?",alt:"screenshot-12-skoogee-homelab-3"})})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"ground-control",children:"Ground Control"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://github.com/dtctek",children:"@dtctek"})," ",(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/83",children:"#83"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/149821995-e9b41dab-186c-42e6-b5b3-e233259b241d.png",alt:"screenshot-ground-control"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"croco_grievous",children:"Croco_Grievous"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/Croco_Grievous/",children:"u/Croco_Grievous"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/t4xk3z/everything_started_with_pihole_on_a_raspberry_pi/",children:"reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/59XR8KL/dashy-Croco-Grievous.png",alt:"screenshot-croco-grievous-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"crypto-dash",children:"Crypto Dash"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["Example usage of widgets to monitor cryptocurrencies news, prices and data. Config is ",(0,h.jsx)(e.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10#file-example-8-dashy-crypto-widgets-conf-yml",children:"available here"})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://user-images.githubusercontent.com/1862727/147394584-352fe3bf-740d-4624-a01b-9003a97bc832.png",alt:"screenshot-crypto-dash"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"stefantigro",children:"Stefantigro"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsxs)(e.p,{children:["By ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/user/stefantigro/",children:"u/stefantigro"})," ",(0,h.jsxs)(e.sup,{children:["via ",(0,h.jsx)(e.a,{href:"https://www.reddit.com/r/selfhosted/comments/t5oril/been_selfhosting_close_to_half_a_year_now_all/",children:"reddit"})]})]}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/1Kb43Yy/dashy-stefantigro.png",alt:"screenshot-stefantigro-dashy"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"homelab-30-1",children:"HomeLab 3.0"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:(0,h.jsxs)(e.sup,{children:["Re: ",(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/issues/1999",children:"#1999"})]})}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://pixelflare.cc/alicia/dashy/12-skoogee-homelab-3",alt:"screenshot-homelab-3"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"morning-dashboard",children:"Morning Dashboard"}),"\n",(0,h.jsxs)(e.blockquote,{children:["\n",(0,h.jsx)(e.p,{children:"Displayed on my smart screen between 05:00 - 08:00, and includes all the info that I usually check before leaving for work"}),"\n"]}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://i.ibb.co/4Wx8zb7/morning-dashboard.png",alt:"screenshot-morning-dash"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"yet-another-homelab",children:"Yet Another Homelab"}),"\n",(0,h.jsx)(e.p,{children:(0,h.jsx)(e.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/9-home-lab-oblivion.png",alt:"screenshot-yet-another-homelab"})}),"\n",(0,h.jsx)(e.hr,{}),"\n",(0,h.jsx)(e.h2,{id:"submitting-your-dashboard",children:"Submitting your Dashboard"}),"\n",(0,h.jsx)(e.h3,{id:"how-to-submit",children:"How to Submit"}),"\n",(0,h.jsxs)(e.ul,{children:["\n",(0,h.jsx)(e.li,{children:(0,h.jsx)(e.a,{href:"https://git.io/JEtgM",children:"Open an Issue"})}),"\n",(0,h.jsx)(e.li,{children:(0,h.jsx)(e.a,{href:"https://github.com/Lissy93/dashy/compare",children:"Open a PR"})}),"\n"]}),"\n",(0,h.jsx)(e.h3,{id:"what-to-include",children:"What to Include"}),"\n",(0,h.jsx)(e.p,{children:"Please include the following information:"}),"\n",(0,h.jsxs)(e.ul,{children:["\n",(0,h.jsx)(e.li,{children:"A single high-quality screenshot of your Dashboard"}),"\n",(0,h.jsx)(e.li,{children:"A short title (it doesn't have to be particularly imaginative)"}),"\n",(0,h.jsx)(e.li,{children:"An optional description, you could include details on anything interesting or unique about your dashboard, or say how you use it, and why it's awesome"}),"\n",(0,h.jsx)(e.li,{children:"Optionally leave your name or username, with a link to your GitHub, Twitter or Website"}),"\n"]}),"\n",(0,h.jsx)(e.h3,{id:"template",children:"Template"}),"\n",(0,h.jsx)(e.p,{children:"If you're submitting a pull request, please use a format similar to this:"}),"\n",(0,h.jsx)(e.pre,{children:(0,h.jsx)(e.code,{className:"language-text",children:"### [Dashboard Name] (required)\n\n> Submitted by [@username](https://github.com/user) (optional)\n\n[An optional text description, or any interesting details] (optional)\n\n![dashboard-screenshot](https://example.com/url-to-screenshot.png) (required)\n\n---\n\n"})})]})}function l(s={}){const{wrapper:e}={...(0,t.R)(),...s.components};return e?(0,h.jsx)(e,{...s,children:(0,h.jsx)(c,{...s})}):c(s)}},8453(s,e,n){n.d(e,{R:()=>i,x:()=>a});var r=n(6540);const h={},t=r.createContext(h);function i(s){const e=r.useContext(t);return r.useMemo(function(){return"function"==typeof s?s(e):{...e,...s}},[e,s])}function a(s){let e;return e=s.disableParentContext?"function"==typeof s.components?s.components(h):s.components||h:i(s.components),r.createElement(t.Provider,{value:e},s.children)}}}]); \ No newline at end of file diff --git a/assets/js/4809.daa2d78a.js b/assets/js/4809.d699a3f1.js similarity index 91% rename from assets/js/4809.daa2d78a.js rename to assets/js/4809.d699a3f1.js index eee5edc1..c31f9562 100644 --- a/assets/js/4809.daa2d78a.js +++ b/assets/js/4809.d699a3f1.js @@ -1 +1 @@ -(()=>{"use strict";var e,r,t={4809(e,r,t){t.d(r,{BH:()=>s,Ho:()=>n,IH:()=>a,sx:()=>o});t(8291);const o=[],s=["en"],a="search-index{dir}.json?_=917700dd",n=1}},o={};function s(e){var r=o[e];if(void 0!==r)return r.exports;var a=o[e]={exports:{}};return t[e](a,a.exports,s),a.exports}s.m=t,s.x=()=>{var e=s.O(void 0,[540],()=>s(540));return e=s.O(e)},e=[],s.O=(r,t,o,a)=>{if(!t){var n=1/0;for(f=0;f=a)&&Object.keys(s.O).every(e=>s.O[e](t[p]))?t.splice(p--,1):(i=!1,a0&&e[f-1][2]>a;f--)e[f]=e[f-1];e[f]=[t,o,a]},s.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return s.d(r,{a:r}),r},s.d=(e,r)=>{for(var t in r)s.o(r,t)&&!s.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},s.f={},s.e=e=>Promise.all(Object.keys(s.f).reduce((r,t)=>(s.f[t](e,r),r),[])),s.u=e=>"assets/js/"+e+".93d81217.js",s.miniCssF=e=>{},s.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),s.p="/",s.gca=function(e){return e={}[e]||e,s.p+s.u(e)},(()=>{var e={4809:1};s.f.i=(r,t)=>{e[r]||importScripts(s.p+s.u(r))};var r=globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[],t=r.push.bind(r);r.push=r=>{var[o,a,n]=r;for(var i in a)s.o(a,i)&&(s.m[i]=a[i]);for(n&&n(s);o.length;)e[o.pop()]=1;t(r)}})(),r=s.x,s.x=()=>s.e(540).then(r);s.x()})(); \ No newline at end of file +(()=>{"use strict";var e,r,t={4809(e,r,t){t.d(r,{BH:()=>s,Ho:()=>n,IH:()=>a,sx:()=>o});t(8291);const o=[],s=["en"],a="search-index{dir}.json?_=6035287a",n=1}},o={};function s(e){var r=o[e];if(void 0!==r)return r.exports;var a=o[e]={exports:{}};return t[e](a,a.exports,s),a.exports}s.m=t,s.x=()=>{var e=s.O(void 0,[540],()=>s(540));return e=s.O(e)},e=[],s.O=(r,t,o,a)=>{if(!t){var n=1/0;for(f=0;f=a)&&Object.keys(s.O).every(e=>s.O[e](t[p]))?t.splice(p--,1):(i=!1,a0&&e[f-1][2]>a;f--)e[f]=e[f-1];e[f]=[t,o,a]},s.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return s.d(r,{a:r}),r},s.d=(e,r)=>{for(var t in r)s.o(r,t)&&!s.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},s.f={},s.e=e=>Promise.all(Object.keys(s.f).reduce((r,t)=>(s.f[t](e,r),r),[])),s.u=e=>"assets/js/"+e+".93d81217.js",s.miniCssF=e=>{},s.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),s.p="/",s.gca=function(e){return e={}[e]||e,s.p+s.u(e)},(()=>{var e={4809:1};s.f.i=(r,t)=>{e[r]||importScripts(s.p+s.u(r))};var r=globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[],t=r.push.bind(r);r.push=r=>{var[o,a,n]=r;for(var i in a)s.o(a,i)&&(s.m[i]=a[i]);for(n&&n(s);o.length;)e[o.pop()]=1;t(r)}})(),r=s.x,s.x=()=>s.e(540).then(r);s.x()})(); \ No newline at end of file diff --git a/assets/js/4bdb0d83.7e337ce8.js b/assets/js/4bdb0d83.7e337ce8.js new file mode 100644 index 00000000..8cd1615e --- /dev/null +++ b/assets/js/4bdb0d83.7e337ce8.js @@ -0,0 +1 @@ +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5496],{7395(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>c,default:()=>a,frontMatter:()=>t,metadata:()=>i,toc:()=>o});const i=JSON.parse('{"id":"configuring","title":"Configuring","description":"All app configuration is specified in /user-data/conf.yml which is in YAML Format format. If you\'re using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done through the UI. From the UI you can also export, backup, reset, validate and download your configuration file.","source":"@site/docs/configuring.md","sourceDirName":".","slug":"/configuring","permalink":"/docs/configuring","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/configuring.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Deployment","permalink":"/docs/deployment"},"next":{"title":"App Management","permalink":"/docs/management"}}');var d=s(4848),r=s(8453);const t={},c="Configuring",l={},o=[{value:"There are three ways to edit the config",id:"there-are-three-ways-to-edit-the-config",level:2},{value:"Tips",id:"tips",level:2},{value:"Contents",id:"contents",level:2},{value:"Top-Level Fields",id:"top-level-fields",level:2},{value:"PageInfo",id:"pageinfo",level:2},{value:"pageInfo.navLinks (optional)",id:"pageinfonavlinks-optional",level:2},{value:"pages[] (optional)",id:"pages-optional",level:2},{value:"appConfig (optional)",id:"appconfig-optional",level:2},{value:"appConfig.auth (optional)",id:"appconfigauth-optional",level:2},{value:"appConfig.auth.users (optional)",id:"appconfigauthusers-optional",level:2},{value:"appConfig.auth.keycloak (optional)",id:"appconfigauthkeycloak-optional",level:2},{value:"appConfig.auth.headerAuth (optional)",id:"appconfigauthheaderauth-optional",level:2},{value:"appConfig.auth.oidc (optional)",id:"appconfigauthoidc-optional",level:2},{value:"appConfig.webSearch (optional)",id:"appconfigwebsearch-optional",level:2},{value:"appConfig.hideComponents (optional)",id:"appconfighidecomponents-optional",level:2},{value:"section",id:"section",level:2},{value:"section.item",id:"sectionitem",level:2},{value:"item.displayData (optional)",id:"itemdisplaydata-optional",level:2},{value:"section.widgets (optional)",id:"sectionwidgets-optional",level:2},{value:"section.displayData (optional)",id:"sectiondisplaydata-optional",level:2},{value:"section.icon and section.item.icon",id:"sectionicon-and-sectionitemicon",level:2},{value:"section.displayData.hideForKeycloakUsers, section.displayData.showForKeycloakUsers, item.displayData.hideForKeycloakUsers and item.displayData.showForKeycloakUsers",id:"sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",level:2},{value:"Notes",id:"notes",level:2},{value:"Editing Config through the UI",id:"editing-config-through-the-ui",level:3},{value:"About YAML",id:"about-yaml",level:3},{value:"Config Saving Methods",id:"config-saving-methods",level:3},{value:"Preventing Changes",id:"preventing-changes",level:3},{value:"Example",id:"example",level:3}];function h(e){const n={a:"a",admonition:"admonition",b:"b",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,r.R)(),...e.components};return(0,d.jsxs)(d.Fragment,{children:[(0,d.jsx)(n.header,{children:(0,d.jsx)(n.h1,{id:"configuring",children:"Configuring"})}),"\n",(0,d.jsxs)(n.p,{children:["All app configuration is specified in ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml",children:(0,d.jsx)(n.code,{children:"/user-data/conf.yml"})})," which is in ",(0,d.jsx)(n.a,{href:"https://yaml.org/",children:"YAML Format"})," format. If you're using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done ",(0,d.jsx)(n.a,{href:"#editing-config-through-the-ui",children:"through the UI"}),". From the UI you can also export, backup, reset, validate and download your configuration file."]}),"\n",(0,d.jsx)(n.h2,{id:"there-are-three-ways-to-edit-the-config",children:"There are three ways to edit the config"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"Directly in the YAML file"})," ",(0,d.jsx)(n.em,{children:"(5/5 reliability, 3/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:["Write changes directly to the conf.yml file, optionally using one of the templates provided. This can be done in your favorite editor and uploading to your server, or directly editing the file via SSH, but the easiest method would be to use ",(0,d.jsx)(n.a,{href:"https://github.com/coder/code-server",children:"Code Server"})]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"UI JSON Editor"})," ",(0,d.jsx)(n.em,{children:"(4/5 reliability, 4/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"From the UI, under the config menu there is a JSON editor, with built-in validation, documentation and advanced options"}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"UI Visual Editor"})," ",(0,d.jsx)(n.em,{children:"(3/5 reliability, 5/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"From the UI, enter the Interactive Edit Mode, then click any part of the page to edit. Changes are previewed live, and then saved to disk"}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"REST API"})," ",(0,d.jsx)(n.em,{children:"(Coming soon)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"Programmatically edit config either through the command line, using a script or a third-party application"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsx)(n.h2,{id:"tips",children:"Tips"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:["You may find it helpful to look at some sample config files to get you started, a collection of which can be found ",(0,d.jsx)(n.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"here"})]}),"\n",(0,d.jsxs)(n.li,{children:["You can check that your config file fits the schema, by running ",(0,d.jsx)(n.code,{children:"yarn validate-config"})]}),"\n",(0,d.jsxs)(n.li,{children:["After modifying your config, the app needs to be recompiled, by running ",(0,d.jsx)(n.code,{children:"yarn build"})," - this happens automatically if you're using Docker"]}),"\n",(0,d.jsxs)(n.li,{children:["It is recommended to keep a backup of your config file. You can download it under Config menu, or use the ",(0,d.jsx)(n.a,{href:"/docs/backup-restore",children:"Cloud Backup"})," feature."]}),"\n",(0,d.jsx)(n.li,{children:"You can make use of YAML features, like anchors, comments, multi-line strings, etc to reuse attributes and keep your config file readable"}),"\n",(0,d.jsxs)(n.li,{children:["Once you have finished configuring your dashboard, you can choose to ",(0,d.jsx)(n.a,{href:"#preventing-changes",children:"disable UI config"})," if you wish"]}),"\n",(0,d.jsx)(n.li,{children:"All fields are optional, unless otherwise stated."}),"\n"]}),"\n",(0,d.jsx)(n.p,{children:"The following file provides a reference of all supported configuration options."}),"\n",(0,d.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pageinfo",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pageInfo"})})})," - Header text, footer, title, navigation, etc\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pageinfonavlinks-optional",children:(0,d.jsx)(n.code,{children:"navLinks"})})," - Links to display in the navigation bar"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pages-optional",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pages"})})})," - List of additional config files, for multi-page dashboards"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfig-optional",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"appConfig"})})})," - Main application settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigwebsearch-optional",children:(0,d.jsx)(n.code,{children:"webSearch"})})," - Configure web search engine options"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfighidecomponents-optional",children:(0,d.jsx)(n.code,{children:"hideComponents"})})," - Show/ hide page components"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauth-optional",children:(0,d.jsx)(n.code,{children:"auth"})})," - Built-in authentication setup\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthusers-optional",children:(0,d.jsx)(n.code,{children:"users"})})," - List or users (for simple auth)"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthkeycloak-optional",children:(0,d.jsx)(n.code,{children:"keycloak"})})," - Auth config for Keycloak"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthheaderauth-optional",children:(0,d.jsx)(n.code,{children:"headerAuth"})})," - Auth config for HeaderAuth"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#section",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sections"})})})," - List of sections\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})," - Section display settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:(0,d.jsx)(n.code,{children:"show/hideForKeycloakUsers"})})," - Set user controls"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"icon"})})," - Icon for a section"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionitem",children:(0,d.jsx)(n.code,{children:"items"})})," - List of items\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"icon"})})," - Icon for an item"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#itemdisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})," - Item display settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:(0,d.jsx)(n.code,{children:"show/hideForKeycloakUsers"})})," - Set user controls"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionwidgets-optional",children:(0,d.jsx)(n.code,{children:"widgets"})})," - List of widgets"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#notes",children:(0,d.jsx)(n.strong,{children:"Notes"})}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#editing-config-through-the-ui",children:"Editing Config through the UI"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#about-yaml",children:"About YAML"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#config-saving-methods",children:"Config Saving Methods"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#preventing-changes",children:"Preventing Changes"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#example",children:"Example"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsx)(n.hr,{}),"\n",(0,d.jsx)(n.h2,{id:"top-level-fields",children:"Top-Level Fields"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pageInfo"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["Basic meta data like title, description, nav bar links, footer text. See ",(0,d.jsx)(n.a,{href:"#pageinfo",children:(0,d.jsx)(n.code,{children:"pageInfo"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"appConfig"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Settings related to how the app functions, including API keys and global styles. See ",(0,d.jsx)(n.a,{href:"#appconfig-optional",children:(0,d.jsx)(n.code,{children:"appConfig"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sections"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["An array of sections, each containing an array of items, which will be displayed as links. See ",(0,d.jsx)(n.a,{href:"#section",children:(0,d.jsx)(n.code,{children:"section"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pages"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array additional config files, used for multi-page dashboards. See ",(0,d.jsx)(n.a,{href:"#pages-optional",children:(0,d.jsx)(n.code,{children:"pages"})})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"pageinfo",children:(0,d.jsx)(n.code,{children:"PageInfo"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"Your dashboard title, displayed in the header and browser tab"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"description"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Description of your dashboard, also displayed as a subtitle"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"navLinks"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Optional list of a maximum of 6 links, which will be displayed in the navigation bar. See ",(0,d.jsx)(n.a,{href:"#pageinfonavlinks-optional",children:(0,d.jsx)(n.code,{children:"navLinks"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"footer"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Text to display in the footer. When omitted, no footer is rendered. Supports inline HTML (sanitized before render)"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"logo"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The path to an image to display in the header (to the right of the title). This can be either local, where ",(0,d.jsx)(n.code,{children:"/"})," is the root of ",(0,d.jsx)(n.code,{children:"./public"}),", or any remote image, such as ",(0,d.jsx)(n.code,{children:"https://i.ibb.co/yhbt6CY/dashy.png"}),". It's recommended to scale your image down, so that it doesn't impact load times"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"favicon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["URL or path to a custom favicon shown in the browser tab. Can be absolute (",(0,d.jsx)(n.code,{children:"https://..."}),"), root-relative (",(0,d.jsx)(n.code,{children:"/icons/x.png"}),"), or a ",(0,d.jsx)(n.code,{children:"data:"})," URI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme colour applied to the browser chrome (mobile address bar). Any valid CSS color (e.g. ",(0,d.jsx)(n.code,{children:"#ff00a7"}),") is accepted"]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"pageinfonavlinks-optional",children:[(0,d.jsx)(n.code,{children:"pageInfo.navLinks"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The text to display on the link button"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"path"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The URL to navigate to when clicked. Can be relative (e.g. ",(0,d.jsx)(n.code,{children:"/about"}),") or absolute (e.g. ",(0,d.jsx)(n.code,{children:"https://example.com"})," or ",(0,d.jsx)(n.code,{children:"http://192.168.1.1"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"target"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The opening method (external links only). Can be either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"newwindow"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"pages-optional",children:[(0,d.jsx)(n.code,{children:"pages[]"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"name"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"A unique name for that page"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"path"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The path (local or remote) to the config file to use.",(0,d.jsx)(n.br,{}),"For files located within ",(0,d.jsx)(n.code,{children:"/user-data"}),", you only need to specify filename, for externally hosted files you must include the full URL"]})]})]})]}),"\n",(0,d.jsxs)(n.p,{children:["For more info, see the",(0,d.jsx)(n.a,{href:"/docs/pages-and-sections#multi-page-support",children:"Multi-Page docs"})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfig-optional",children:[(0,d.jsx)(n.code,{children:"appConfig"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"language"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The 2 (or 4-digit) ",(0,d.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes",children:"ISO 639-1 code"})," for your language, e.g. ",(0,d.jsx)(n.code,{children:"en"})," or ",(0,d.jsx)(n.code,{children:"en-GB"}),". This must be a language that the app has already been ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/assets/locales",children:"translated"})," into. If your language is unavailable, Dashy will fallback to English. By default Dashy will attempt to auto-detect your language, although this may not work on some privacy browsers."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"startingView"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Which view to land on when visiting ",(0,d.jsx)(n.code,{children:"/"}),". One of ",(0,d.jsx)(n.code,{children:"home"}),", ",(0,d.jsx)(n.code,{children:"minimal"})," or ",(0,d.jsx)(n.code,{children:"workspace"}),". Defaults to ",(0,d.jsx)(n.code,{children:"home"}),". Applied at runtime, so no rebuild is needed. You can always switch views from the UI. (Legacy value ",(0,d.jsx)(n.code,{children:"default"})," is accepted as an alias for ",(0,d.jsx)(n.code,{children:"home"}),".)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"defaultOpeningMethod"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The default opening method for items, if no ",(0,d.jsx)(n.code,{children:"target"})," is specified for a given item. Can be either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"modal"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),", ",(0,d.jsx)(n.code,{children:"clipboard"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheck"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", Dashy will ping each of your services and display their status as a dot next to each item. This can be overridden by setting ",(0,d.jsx)(n.code,{children:"statusCheck"})," under each item. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckInterval"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The number of seconds between checks. If set to ",(0,d.jsx)(n.code,{children:"0"})," then service will only be checked on initial page load, which is usually the desired functionality. If value is less than ",(0,d.jsx)(n.code,{children:"10"})," you may experience a hit in performance. Defaults to ",(0,d.jsx)(n.code,{children:"0"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAccessibility"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", status indicators will use distinct shapes to indicate status for color-blind users. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"webSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Configuration options for the web search feature, set your default search engine, opening method or disable web search. See ",(0,d.jsx)(n.a,{href:"#appconfigwebsearch-optional",children:(0,d.jsx)(n.code,{children:"webSearch"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"backgroundImg"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Path to an optional full-screen app background image. This can be either remote (http) or local (relative to /app/public/item-icons/ inside the container). Note that this will slow down initial load"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableFontAwesome"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"})," font-awesome will be loaded, if set to ",(0,d.jsx)(n.code,{children:"false"})," they will not be. if left blank font-awesome will be enabled only if required by 1 or more icons"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableMaterialDesignIcons"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"})," mdi icons will be loaded, if set to ",(0,d.jsx)(n.code,{children:"false"})," they will not be. Where ",(0,d.jsx)(n.code,{children:"true"})," is enabled, if left blank material design icons will be enabled only if required by 1 or more icons"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"fontAwesomeKey"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you have a font-awesome key, then you can use it here and make use of premium icons. It is a 10-digit alpha-numeric string from you're FA kit URL (e.g. ",(0,d.jsx)(n.code,{children:"13014ae648"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"faviconApi"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Only applicable if you are using favicons for item icons. Specifies which service to use to resolve favicons. Set to ",(0,d.jsx)(n.code,{children:"local"})," to do this locally, without using an API. Services running locally will use this option always. Available options are: ",(0,d.jsx)(n.code,{children:"local"}),", ",(0,d.jsx)(n.code,{children:"allesedv"}),", ",(0,d.jsx)(n.code,{children:"iconhorse"}),", ",(0,d.jsx)(n.code,{children:"faviconkit"}),", ",(0,d.jsx)(n.code,{children:"duckduckgo"}),", ",(0,d.jsx)(n.code,{children:"yandex"}),", ",(0,d.jsx)(n.code,{children:"google"}),", ",(0,d.jsx)(n.code,{children:"besticon"}),", ",(0,d.jsx)(n.code,{children:"webmasterapi"})," and ",(0,d.jsx)(n.code,{children:"mcapi"}),". Defaults to ",(0,d.jsx)(n.code,{children:"allesedv"}),". See ",(0,d.jsx)(n.a,{href:"/docs/icons#favicons",children:"Icons"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"auth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["All settings relating to user authentication. See ",(0,d.jsx)(n.a,{href:"#appconfigauth-optional",children:(0,d.jsx)(n.code,{children:"auth"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"defaultIcon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An icon to be applied to any items, which don't already have an icon set. See ",(0,d.jsx)(n.a,{href:"/docs/icons#default-icon",children:"Icon Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"layout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Layout for homepage, one of ",(0,d.jsx)(n.code,{children:"auto"}),", ",(0,d.jsx)(n.code,{children:"horizontal"}),", ",(0,d.jsx)(n.code,{children:"vertical"})," or ",(0,d.jsx)(n.code,{children:"masonry"}),". Defaults to ",(0,d.jsx)(n.code,{children:"auto"}),". Specifies the layout and direction of how sections are positioned on the home screen. ",(0,d.jsx)(n.code,{children:"auto"})," uses a responsive CSS grid where each section's footprint is controlled by ",(0,d.jsx)(n.code,{children:"displayData.cols"})," and ",(0,d.jsx)(n.code,{children:"displayData.rows"}),". ",(0,d.jsx)(n.code,{children:"masonry"})," uses a responsive grid where heights follow content, so shorter sections flow into the gaps left by taller neighbours (",(0,d.jsx)(n.code,{children:"rows"})," is ignored, ",(0,d.jsx)(n.code,{children:"cols"})," still controls width). This can also be modified and overridden from the UI."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"iconSize"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The size of link items / icons. Can be either ",(0,d.jsx)(n.code,{children:"small"}),", ",(0,d.jsx)(n.code,{children:"medium,"})," or ",(0,d.jsx)(n.code,{children:"large"}),". Defaults to ",(0,d.jsx)(n.code,{children:"medium"}),". This can also be set directly from the UI."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"colCount"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The number of columns of sections displayed on the homepage, using the default view. Should be in integer between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"8"}),". Note that by default this is applied responsively, based on current screen size, and specifying a value here will override this behavior, which may not be desirable."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"contentMaxWidth"})})}),(0,d.jsxs)(n.td,{children:[(0,d.jsx)(n.code,{children:"string"})," or ",(0,d.jsx)(n.code,{children:"number"})]}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Sets the max width of the main sections area on the homepage, overriding the responsive default. Can be a percentage, or any CSS unit"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"theme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The default theme for first load (you can change this later from the UI)"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"dayTheme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme to apply when the OS is set to light mode. Overrides ",(0,d.jsx)(n.code,{children:"theme"})," on initial load"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"nightTheme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme to apply when the OS is set to dark mode. Overrides ",(0,d.jsx)(n.code,{children:"theme"})," on initial load"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cssThemes"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An array of custom theme names which can be used in the theme switcher dropdown"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customColors"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Enables you to apply a custom color palette to any given theme. Use the theme name (lowercase) as the key, for an object including key-value-pairs, with the color variable name as keys, and 6-digit hex code as value. See ",(0,d.jsx)(n.a,{href:"/docs/theming#modifying-theme-colors",children:"Theming"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"externalStyleSheet"})})}),(0,d.jsxs)(n.td,{children:[(0,d.jsx)(n.code,{children:"string"})," or ",(0,d.jsx)(n.code,{children:"string[]"})]}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Either a URL to an external stylesheet or an array or URLs, which can be applied as themes within the UI"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customCss"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Raw CSS that will be applied to the page. This can also be set from the UI. Please minify it first."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideComponents"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A list of key page components (header, nav, search, settings) that are present by default, but can be removed using this option. See ",(0,d.jsx)(n.a,{href:"#appconfighidecomponents-optional",children:(0,d.jsx)(n.code,{children:"appConfig.hideComponents"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableMultiTasking"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, will keep apps open in the background when in the workspace view. Useful for quickly switching between multiple sites, and preserving their state, but comes at the cost of performance."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"workspaceLandingUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The URL or an app, service or website to launch when the workspace view is opened, before another service has been launched"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"preventWriteToDisk"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", users will be prevented from saving config changes to disk through the UI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"preventLocalSave"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", users will be prevented from applying config changes to local storage"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableConfiguration"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, no users will be able to view or edit the config through the UI"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, only admin users will be able to view or edit the config through the UI. disableConfiguration must not be set to true."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"widgetsAlwaysUseProxy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", requests made by widgets will always be proxied, same as setting ",(0,d.jsx)(n.code,{children:"useProxy: true"})," on each widget. Note that this may break some widgets."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showSplashScreen"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", a loading screen will be shown. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableErrorReporting"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Enable reporting of unexpected errors and crashes. This is off by default, and ",(0,d.jsx)(n.strong,{children:"no data will ever be captured unless you explicitly enable it"}),". Turning on error reporting helps previously unknown bugs get discovered and fixed. Dashy uses ",(0,d.jsx)(n.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," for error reporting. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sentryDsn"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you need to monitor errors in your instance, then you can use Sentry to collect and process bug reports. Sentry can be self-hosted, or used as SaaS, once your instance is setup, then all you need to do is pass in the DSN here, and enable error reporting. You can learn more on the ",(0,d.jsx)(n.a,{href:"https://docs.sentry.io/product/sentry-basics/dsn-explainer/",children:"Sentry DSN Docs"}),". Note that this will only ever be used if ",(0,d.jsx)(n.code,{children:"enableErrorReporting"})," is explicitly enabled."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableSmartSort"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["For the most-used and last-used app sort functions to work, a basic open-count is stored in local storage. If you do not want this to happen, then disable smart sort here, but you wil no longer be able to use these sort options. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableUpdateChecks"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to true, Dashy will not check for updates. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableServiceWorker"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Service workers cache web applications to improve load times and offer basic offline functionality, and are disabled by default in Dashy. The service worker can sometimes cause older content to be cached, requiring the app to be hard-refreshed. If you do not want SW functionality, or are having issues with caching, set this property to ",(0,d.jsx)(n.code,{children:"false"})," to disable all service workers."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableContextMenu"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the custom right-click context menu will be disabled. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauth-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n",(0,d.jsx)(n.admonition,{type:"note",children:(0,d.jsxs)(n.p,{children:["Since the auth is initiated in the main app entry point (for security), a rebuild is required to apply changes to the auth configuration.\nRun ",(0,d.jsx)(n.code,{children:"yarn build"})," in the root directory, then restart the server."]})}),"\n",(0,d.jsx)(n.admonition,{type:"warning",children:(0,d.jsxs)(n.p,{children:["Built-in auth should ",(0,d.jsx)(n.strong,{children:"not be used"})," for security-critical applications, or if your Dashy instance is publicly accessible.\nFor these, it is recommended to use an ",(0,d.jsx)(n.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternate authentication method"}),"."]})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"users"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of objects containing usernames and hashed passwords. If this is not provided, then authentication will be off by default, and you will not need any credentials to access the app. See ",(0,d.jsx)(n.a,{href:"#appconfigauthusers-optional",children:(0,d.jsx)(n.code,{children:"appConfig.auth.users"})}),". ",(0,d.jsx)(n.br,{}),(0,d.jsx)(n.strong,{children:"Note"})," this method of authentication is handled on the client side, so for security critical situations, it is recommended to use an ",(0,d.jsx)(n.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternate authentication method"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableKeycloak"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using Keycloak will be enabled. Note that you need to have an instance running, and have also configured ",(0,d.jsx)(n.code,{children:"auth.keycloak"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"keycloak"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dashy to your Keycloak server. Requires ",(0,d.jsx)(n.code,{children:"enableKeycloak: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthkeycloak-optional",children:(0,d.jsx)(n.code,{children:"auth.keycloak"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableHeaderAuth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using HeaderAuth will be enabled. Note that you need to have your web server/reverse proxy running, and have also configured ",(0,d.jsx)(n.code,{children:"auth.headerAuth"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"headerAuth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dashy to your headers for authentication. Requires ",(0,d.jsx)(n.code,{children:"enableHeaderAuth: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthheaderauth-optional",children:(0,d.jsx)(n.code,{children:"auth.headerAuth"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableOidc"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using OIDC will be enabled. Note that you need to have a configured OIDC server and configure it with ",(0,d.jsx)(n.code,{children:"auth.oidc"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"oidc"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dash to your OIDC configuration. Request ",(0,d.jsx)(n.code,{children:"enableOidc: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthoidc-optional",children:(0,d.jsx)(n.code,{children:"auth.oidc"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableGuestAccess"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", an unauthenticated user will be able to access the dashboard, with read-only access, without having to login. Requires ",(0,d.jsx)(n.code,{children:"auth.users"})," to be configured. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]})]})]}),"\n",(0,d.jsxs)(n.p,{children:["For more info, see the ",(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthusers-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.users"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"user"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"Username to log in with"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hash"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"A SHA-256 hashed password"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"type"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The user type, either admin or normal"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthkeycloak-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.keycloak"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"serverUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The URL (or URL/ IP + Port) where your keycloak server is running"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"realm"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The name of the realm (must already be created) that you want to use"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"clientId"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The Client ID of the client you created for use with Dashy"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"idpHint"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The alias of an Identity Provider configured in your realm. If set, Keycloak will skip its login page and redirect straight to that external IdP"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"legacySupport"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If using Keycloak 17 or older, then set this to ",(0,d.jsx)(n.code,{children:"true"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthheaderauth-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.headerAuth"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"userHeader"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The HTTP header name containing the authenticated username (default: ",(0,d.jsx)(n.code,{children:"Remote-User"}),"). Case insensitive"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"proxyWhitelist"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"An array of upstream proxy server IPs to accept authenticated requests from"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthoidc-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.oidc"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"clientId"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The client id registered in the OIDC server"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"endpoint"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The URL of the OIDC server that should be used."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"adminRole"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The role that will be considered as admin."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"adminGroup"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The group that will be considered as admin."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"scope"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The scope(s) to request from the OIDC provider"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigwebsearch-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.webSearch"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableWebSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Web search is enabled by default, but can be disabled by setting this property to ",(0,d.jsx)(n.code,{children:"true"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"searchEngine"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Set the key name for your search engine. Can also use a custom engine by setting this property to ",(0,d.jsx)(n.code,{children:"custom"}),". Currently supported: ",(0,d.jsx)(n.code,{children:"duckduckgo"}),", ",(0,d.jsx)(n.code,{children:"google"}),", ",(0,d.jsx)(n.code,{children:"whoogle"}),", ",(0,d.jsx)(n.code,{children:"qwant"}),", ",(0,d.jsx)(n.code,{children:"startpage"}),", ",(0,d.jsx)(n.code,{children:"searx-bar"})," and ",(0,d.jsx)(n.code,{children:"searx-info"}),". Defaults to ",(0,d.jsx)(n.code,{children:"duckduckgo"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customSearchEngine"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["You can also use a custom search engine, or your own self-hosted instance. This requires ",(0,d.jsx)(n.code,{children:"searchEngine: custom"})," to be set. Then add the URL of your service, with GET query string included here"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"openingMethod"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Set your preferred opening method for search results: ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"searchBangs"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A key-value-pair set of custom search ",(0,d.jsx)(n.em,{children:"bangs"})," for redirecting query to a specific app or search engine. The key of each should be the bang you will type (typically starting with ",(0,d.jsx)(n.code,{children:"/"}),", ",(0,d.jsx)(n.code,{children:"!"})," or ",(0,d.jsx)(n.code,{children:":"}),"), and value is the destination, either as a search engine key (e.g. ",(0,d.jsx)(n.code,{children:"reddit"}),") or a URL with search parameters (e.g. ",(0,d.jsx)(n.code,{children:"https://en.wikipedia.org/w/?search="}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"openUrlsDirectly"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If ",(0,d.jsx)(n.code,{children:"true"}),", queries that look like URLs will be opened directly instead of searched. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfighidecomponents-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.hideComponents"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideHeading"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the page title & sub-title will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideNav"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the navigation menu will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the search bar will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideSettings"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the settings menu will be initially collapsed. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"section",children:(0,d.jsx)(n.code,{children:"section"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"name"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The title for the section"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An single icon to be displayed next to the title. See ",(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"section.icon"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"items"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of items to be displayed within the section. See ",(0,d.jsx)(n.a,{href:"#sectionitem",children:(0,d.jsx)(n.code,{children:"item"})}),". Sections must include either 1 or more items, or 1 or more widgets."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"widgets"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of widgets to be displayed within the section. See ",(0,d.jsx)(n.a,{href:"#sectionwidgets-optional",children:(0,d.jsx)(n.code,{children:"widget"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"displayData"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Meta-data to optionally override display settings for a given section. See ",(0,d.jsx)(n.a,{href:"#sectiondisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"sectionitem",children:(0,d.jsx)(n.code,{children:"section.item"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The text to display/ title of a given item. Max length ",(0,d.jsx)(n.code,{children:"18"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"description"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Additional info about an item, which is shown in the tooltip on hover, or visible on large tiles"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"url"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The URL / location of web address for when the item is clicked"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The icon for a given item. Can be a font-awesome icon, favicon, remote URL or local URL. See ",(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"item.icon"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"target"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The opening method for when the item is clicked, either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"modal"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),", ",(0,d.jsx)(n.code,{children:"clipboard"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Where ",(0,d.jsx)(n.code,{children:"newtab"})," will open the link in a new tab, ",(0,d.jsx)(n.code,{children:"sametab"})," will open it in the current tab, and ",(0,d.jsx)(n.code,{children:"modal"})," will open a pop-up modal, ",(0,d.jsx)(n.code,{children:"workspace"})," will open in the Workspace view and ",(0,d.jsx)(n.code,{children:"clipboard"})," will copy the URL to system clipboard (but not launch app). Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hotkey"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Give frequently opened applications a numeric hotkey, between ",(0,d.jsx)(n.code,{children:"0 - 9"}),". You can then just press that key to launch that application."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"tags"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"A list of tags, which can be used for improved search"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheck"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", Dashy will ping the URL associated with the current service, and display its status as a dot next to the item. The value here will override ",(0,d.jsx)(n.code,{children:"appConfig.statusCheck"})," so you can turn off or on checks for a given service. Defaults to ",(0,d.jsx)(n.code,{children:"appConfig.statusCheck"}),", falls back to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you've enabled ",(0,d.jsx)(n.code,{children:"statusCheck"}),", and want to use a different URL to what is defined under the item, then specify it here"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckHeaders"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If you're endpoint requires any specific headers for the status checking, then define them here"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAllowInsecure"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["By default, any request to insecure content will be blocked. Setting this option to ",(0,d.jsx)(n.code,{children:"true"})," will disable the ",(0,d.jsx)(n.code,{children:"rejectUnauthorized"})," option, enabling you to ping non-HTTPS services for the current item. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAcceptCodes"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If your service's response code is anything other than 2xx, then you can opt to specify an alternative success code. E.g. if you expect your server to return 403, but still want the status indicator to be green, set this value to ",(0,d.jsx)(n.code,{children:"403"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckMaxRedirects"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If your service redirects to another page, and you would like status checks to follow redirects, then specify the maximum number of redirects here. Defaults to ",(0,d.jsx)(n.code,{children:"0"})," / will not follow redirects"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An optional color for the text and font-awesome icon to be displayed in. Note that this will override the current theme and so may not display well"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"rel"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The value of the ",(0,d.jsx)(n.code,{children:"rel"})," attribute for the link. Useful for specifying the relationship between the target link/document and Dashy. Defaults to ",(0,d.jsx)(n.code,{children:"noopener noreferrer"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"backgroundColor"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An optional background fill color for the that given item. Again, this will override the current theme and so might not display well against the background"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"provider"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The name of the provider for a given service, useful for when including hosted apps. In some themes, this is visible under the item name"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"displayData"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Meta-data to optionally override display settings for a given item. See ",(0,d.jsx)(n.a,{href:"#itemdisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"subItems"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An optional list of nested sub-items, rendered as smaller icons within the parent. Each sub-item supports ",(0,d.jsx)(n.code,{children:"title"}),", ",(0,d.jsx)(n.code,{children:"url"}),", ",(0,d.jsx)(n.code,{children:"icon"}),", ",(0,d.jsx)(n.code,{children:"target"}),", ",(0,d.jsx)(n.code,{children:"color"})," and ",(0,d.jsx)(n.code,{children:"backgroundColor"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"itemdisplaydata-optional",children:[(0,d.jsx)(n.code,{children:"item.displayData"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current item will be visible to all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current item will be hidden from all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForGuests"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible for logged in users, but not for guests (see ",(0,d.jsx)(n.code,{children:"appConfig.enableGuestAccess"}),"). Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible to all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be hidden from all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideFromWorkspace"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible in the default view but not in the Workspace view sidebar. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectionwidgets-optional",children:[(0,d.jsx)(n.code,{children:"section.widgets"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"type"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The widget type. See ",(0,d.jsx)(n.a,{href:"/docs/widgets",children:"Widget Docs"})," for full list of supported widgets"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"options"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Some widgets accept either optional or required additional options. Again, see the ",(0,d.jsx)(n.a,{href:"/docs/widgets",children:"Widget Docs"})," for full list of options"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"updateInterval"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["You can keep a widget constantly updated by specifying an update interval, in seconds. See ",(0,d.jsx)(n.a,{href:"/docs/widgets#continuous-updates",children:"Continuous Updates Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"useProxy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Some widgets make API requests to services that are not CORS-enabled. For these instances, you will need to route requests through a proxy, Dashy has a built in CORS-proxy, which you can use by setting this option to ",(0,d.jsx)(n.code,{children:"true"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"}),". See the ",(0,d.jsx)(n.a,{href:"/docs/widgets#proxying-requests",children:"Proxying Requests Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"timeout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Request timeout in milliseconds, defaults to \xbd a second (",(0,d.jsx)(n.code,{children:"500"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"ignoreErrors"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Prevent an error message being displayed, if a network request or something else fails. Useful for false-positives"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"label"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Add custom label to a given widget. Useful for identification, if there are multiple of the same type of widget in a single section"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectiondisplaydata-optional",children:[(0,d.jsx)(n.code,{children:"section.displayData"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sortBy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The sort order for items within the current section. By default items are displayed in the order in which they are listed in within the config. The following sort options are supported: ",(0,d.jsx)(n.code,{children:"most-used"})," (most opened apps first), ",(0,d.jsx)(n.code,{children:"last-used"})," (the most recently used apps), ",(0,d.jsx)(n.code,{children:"alphabetical"}),", ",(0,d.jsx)(n.code,{children:"reverse-alphabetical"}),", ",(0,d.jsx)(n.code,{children:"random"})," and ",(0,d.jsx)(n.code,{children:"default"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"collapsed"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If true, the section will be collapsed initially, and will need to be clicked to open. Useful for less regularly used, or very long sections. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cutToHeight"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"By default, sections will fill available space. Set this option to true to match section height with content height"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"rows"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Height of the section, specified as the number of rows it should span vertically, e.g. ",(0,d.jsx)(n.code,{children:"2"}),". Defaults to ",(0,d.jsx)(n.code,{children:"1"}),". Max is ",(0,d.jsx)(n.code,{children:"5"}),". Applies to the default ",(0,d.jsx)(n.code,{children:"auto"})," layout; ignored by ",(0,d.jsx)(n.code,{children:"masonry"})," (where heights follow content) and the flex-based ",(0,d.jsx)(n.code,{children:"horizontal"}),"/",(0,d.jsx)(n.code,{children:"vertical"})," layouts."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cols"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Width of the section, specified as the number of columns the section should span horizontally, e.g. ",(0,d.jsx)(n.code,{children:"2"}),". Defaults to ",(0,d.jsx)(n.code,{children:"1"}),". Max is ",(0,d.jsx)(n.code,{children:"5"}),". Will be clamped to the page's active column count so that a section never exceeds the available grid width."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemSize"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Specify the size for items within this group, either ",(0,d.jsx)(n.code,{children:"small"}),", ",(0,d.jsx)(n.code,{children:"medium"})," or ",(0,d.jsx)(n.code,{children:"large"}),". Note that this will override any settings specified through the UI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A custom accent color for the section, as a hex code or HTML color (e.g. ",(0,d.jsx)(n.code,{children:"#fff"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customStyles"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Custom CSS properties that should be applied to that section, e.g. ",(0,d.jsx)(n.code,{children:"border: 2px dashed #ff0000;"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sectionLayout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Specify which CSS layout will be used to responsively place items. Can be either ",(0,d.jsx)(n.code,{children:"auto"})," (which uses flex layout), or ",(0,d.jsx)(n.code,{children:"grid"}),". Defaults to ",(0,d.jsx)(n.code,{children:"auto"}),". Setting ",(0,d.jsx)(n.code,{children:"itemCountX"})," or ",(0,d.jsx)(n.code,{children:"itemCountY"})," below will also switch the section to grid layout automatically"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemCountX"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Number of items per row / horizontally. If not set, it will be calculated automatically based on available space. Setting this switches the section to grid layout. Must be a whole number between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"12"}),"; values above ",(0,d.jsx)(n.code,{children:"8"})," rely on the grid's responsive column sizing (no explicit minimum-width rule)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemCountY"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Number of explicit rows before items flow into implicit rows. Setting this switches the section to grid layout. Must be a whole number between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"12"}),". Row heights size to their content (section heights follow content in the masonry layout)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current section will be visible to all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current section will be hidden from all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForGuests"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible for logged in users, but not for guests (see ",(0,d.jsx)(n.code,{children:"appConfig.enableGuestAccess"}),"). Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible to all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be hidden from all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideFromWorkspace"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible in the default view but not in the Workspace view sidebar. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectionicon-and-sectionitemicon",children:[(0,d.jsx)(n.code,{children:"section.icon"})," and ",(0,d.jsx)(n.code,{children:"section.item.icon"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsx)(n.tbody,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The icon for a given item or section. ",(0,d.jsx)(n.br,{}),"See ",(0,d.jsx)(n.a,{href:"/docs/icons",children:"Icon Docs"})," for all available supported icon types, including: auto-fetched favicons, generative icons, emoji icons, home-lab service logos, font-awesome, simple-icons, material icons, selfh.st icons, and icons specified by URL"]})]})})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:[(0,d.jsx)(n.code,{children:"section.displayData.hideForKeycloakUsers"}),", ",(0,d.jsx)(n.code,{children:"section.displayData.showForKeycloakUsers"}),", ",(0,d.jsx)(n.code,{children:"item.displayData.hideForKeycloakUsers"})," and ",(0,d.jsx)(n.code,{children:"item.displayData.showForKeycloakUsers"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"groups"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current Section or Item will be hidden or shown based on the user having any of the groups in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"roles"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current Section or Item will be hidden or shown based on the user having any of the roles in this list"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.hr,{}),"\n",(0,d.jsx)(n.h2,{id:"notes",children:"Notes"}),"\n",(0,d.jsx)(n.h3,{id:"editing-config-through-the-ui",children:"Editing Config through the UI"}),"\n",(0,d.jsxs)(n.p,{children:["Config can be modified directly through the UI, and then written to disk, or applied locally. This can be done wither with the raw config editor (introduced in V 0.6.5 / ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/pull/3",children:"#3"}),"), or the interactive editor (introduced in V 1.8.9 / ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/pull/298",children:"#298"}),")."]}),"\n ",(0,d.jsxs)(n.a,{href:"https://ibb.co/CzkyMNb",children:["\n ",(0,d.jsx)(n.b,{children:"Interactive Editor"}),(0,d.jsx)(n.br,{}),"\n ",(0,d.jsx)(n.img,{alt:"Interactive Editor demo",src:"https://user-images.githubusercontent.com/1862727/139543020-b0576d28-0830-476f-afc8-a815d4de6def.gif",width:"600"}),"\n "]}),"\n ",(0,d.jsx)(n.br,{}),"\n ",(0,d.jsxs)(n.a,{href:"https://ibb.co/zRv542H",children:["\n ",(0,d.jsx)(n.b,{children:"JSON Editor"}),(0,d.jsx)(n.br,{}),"\n ",(0,d.jsx)(n.img,{alt:"Config Editor demo",src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/config-editor-demo.gif",width:"600"}),"\n "]}),"\n",(0,d.jsx)(n.h3,{id:"about-yaml",children:"About YAML"}),"\n",(0,d.jsxs)(n.p,{children:["If you're new to YAML, it's pretty straight-forward. The format is exactly the same as that of JSON, but instead of using curly braces, structure is denoted using whitespace. This ",(0,d.jsx)(n.a,{href:"https://linuxhandbook.com/yaml-basics/",children:"quick guide"})," should get you up to speed in a few minutes, for more advanced topics take a look at this ",(0,d.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/YAML",children:"Wikipedia article"}),"."]}),"\n",(0,d.jsx)(n.h3,{id:"config-saving-methods",children:"Config Saving Methods"}),"\n",(0,d.jsxs)(n.p,{children:["When updating the config through the JSON editor in the UI, you have two save options: ",(0,d.jsx)(n.strong,{children:"Local"})," or ",(0,d.jsx)(n.strong,{children:"Write to Disk"}),"."]}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"Changes saved locally will only be applied to the current user through the browser, and will not apply to other instances - you either need to use the cloud sync feature, or manually update the conf.yml file."}),"\n",(0,d.jsxs)(n.li,{children:["On the other-hand, if you choose to write changes to disk, then your main ",(0,d.jsx)(n.code,{children:"conf.yml"})," file will be updated, and changes will be applied to all users, and visible across all devices. For this functionality to work, you must be running Dashy with using the Docker container, or the Node server. A backup of your current configuration will also be saved in the same directory."]}),"\n"]}),"\n",(0,d.jsx)(n.h3,{id:"preventing-changes",children:"Preventing Changes"}),"\n",(0,d.jsxs)(n.p,{children:["If you have authentication set up, then any user who is not an admin (with ",(0,d.jsx)(n.code,{children:"type: admin"}),") will not be able to write changes to disk."]}),"\n",(0,d.jsxs)(n.p,{children:["You can also prevent changes from any user being written to disk, using ",(0,d.jsx)(n.code,{children:"preventWriteToDisk"}),". Or prevent any changes from being saved locally in browser storage, using ",(0,d.jsx)(n.code,{children:"preventLocalSave"}),"."]}),"\n",(0,d.jsxs)(n.p,{children:["To disable all UI config features, set ",(0,d.jsx)(n.code,{children:"disableConfiguration"}),". Alternatively you can disable UI config features for all non Admin users by setting ",(0,d.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})," to true."]}),"\n",(0,d.jsx)(n.h3,{id:"example",children:"Example"}),"\n",(0,d.jsx)(n.pre,{children:(0,d.jsx)(n.code,{className:"language-yaml",children:"---\npageInfo:\n title: Home Lab\nsections: # An array of sections\n- name: Section 1 - Getting Started\n items: # An array of items\n - title: GitHub\n description: Source code and documentation on GitHub\n icon: fab fa-github\n url: https://github.com/Lissy93/dashy\n - title: Issues\n description: View currently open issues, or raise a new one\n icon: fas fa-bug\n url: https://github.com/Lissy93/dashy/issues\n - title: Demo\n description: A live demo\n icon: far fa-rocket\n url: https://dashy-demo-1.netlify.app\n- name: Section 2 - Local Services\n items:\n - title: Firewall\n icon: favicon\n url: http://192.168.1.1/\n - title: Game Server\n icon: https://i.ibb.co/710B3Yc/space-invader-x256.png\n url: http://192.168.130.1/\n"})}),"\n",(0,d.jsxs)(n.p,{children:["For more example config files, see: ",(0,d.jsx)(n.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"this gist"})]}),"\n",(0,d.jsxs)(n.p,{children:["If you need any help, feel free to ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=Lissy93&labels=%F0%9F%A4%B7%E2%80%8D%E2%99%82%EF%B8%8F+Question&template=question.md&title=%5BQUESTION%5D",children:"Raise an Issue"})," or ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions",children:"Start a Discussion"})]}),"\n",(0,d.jsx)(n.p,{children:"Happy Configuring \ud83e\udd13\ud83d\udd27"}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})})]})}function a(e={}){const{wrapper:n}={...(0,r.R)(),...e.components};return n?(0,d.jsx)(n,{...e,children:(0,d.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>c});var i=s(6540);const d={},r=i.createContext(d);function t(e){const n=i.useContext(r);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function c(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(d):e.components||d:t(e.components),i.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/4bdb0d83.dafd009d.js b/assets/js/4bdb0d83.dafd009d.js deleted file mode 100644 index 597cc2a5..00000000 --- a/assets/js/4bdb0d83.dafd009d.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5496],{7395(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>c,default:()=>a,frontMatter:()=>t,metadata:()=>i,toc:()=>o});const i=JSON.parse('{"id":"configuring","title":"Configuring","description":"All app configuration is specified in /user-data/conf.yml which is in YAML Format format. If you\'re using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done through the UI. From the UI you can also export, backup, reset, validate and download your configuration file.","source":"@site/docs/configuring.md","sourceDirName":".","slug":"/configuring","permalink":"/docs/configuring","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/configuring.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Deployment","permalink":"/docs/deployment"},"next":{"title":"App Management","permalink":"/docs/management"}}');var d=s(4848),r=s(8453);const t={},c="Configuring",l={},o=[{value:"There are three ways to edit the config",id:"there-are-three-ways-to-edit-the-config",level:2},{value:"Tips",id:"tips",level:2},{value:"Contents",id:"contents",level:2},{value:"Top-Level Fields",id:"top-level-fields",level:2},{value:"PageInfo",id:"pageinfo",level:2},{value:"pageInfo.navLinks (optional)",id:"pageinfonavlinks-optional",level:2},{value:"pages[] (optional)",id:"pages-optional",level:2},{value:"appConfig (optional)",id:"appconfig-optional",level:2},{value:"appConfig.auth (optional)",id:"appconfigauth-optional",level:2},{value:"appConfig.auth.users (optional)",id:"appconfigauthusers-optional",level:2},{value:"appConfig.auth.keycloak (optional)",id:"appconfigauthkeycloak-optional",level:2},{value:"appConfig.auth.headerAuth (optional)",id:"appconfigauthheaderauth-optional",level:2},{value:"appConfig.auth.oidc (optional)",id:"appconfigauthoidc-optional",level:2},{value:"appConfig.webSearch (optional)",id:"appconfigwebsearch-optional",level:2},{value:"appConfig.hideComponents (optional)",id:"appconfighidecomponents-optional",level:2},{value:"section",id:"section",level:2},{value:"section.item",id:"sectionitem",level:2},{value:"item.displayData (optional)",id:"itemdisplaydata-optional",level:2},{value:"section.widgets (optional)",id:"sectionwidgets-optional",level:2},{value:"section.displayData (optional)",id:"sectiondisplaydata-optional",level:2},{value:"section.icon and section.item.icon",id:"sectionicon-and-sectionitemicon",level:2},{value:"section.displayData.hideForKeycloakUsers, section.displayData.showForKeycloakUsers, item.displayData.hideForKeycloakUsers and item.displayData.showForKeycloakUsers",id:"sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",level:2},{value:"Notes",id:"notes",level:2},{value:"Editing Config through the UI",id:"editing-config-through-the-ui",level:3},{value:"About YAML",id:"about-yaml",level:3},{value:"Config Saving Methods",id:"config-saving-methods",level:3},{value:"Preventing Changes",id:"preventing-changes",level:3},{value:"Example",id:"example",level:3}];function h(e){const n={a:"a",admonition:"admonition",b:"b",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,r.R)(),...e.components};return(0,d.jsxs)(d.Fragment,{children:[(0,d.jsx)(n.header,{children:(0,d.jsx)(n.h1,{id:"configuring",children:"Configuring"})}),"\n",(0,d.jsxs)(n.p,{children:["All app configuration is specified in ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml",children:(0,d.jsx)(n.code,{children:"/user-data/conf.yml"})})," which is in ",(0,d.jsx)(n.a,{href:"https://yaml.org/",children:"YAML Format"})," format. If you're using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done ",(0,d.jsx)(n.a,{href:"#editing-config-through-the-ui",children:"through the UI"}),". From the UI you can also export, backup, reset, validate and download your configuration file."]}),"\n",(0,d.jsx)(n.h2,{id:"there-are-three-ways-to-edit-the-config",children:"There are three ways to edit the config"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"Directly in the YAML file"})," ",(0,d.jsx)(n.em,{children:"(5/5 reliability, 3/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:["Write changes directly to the conf.yml file, optionally using one of the templates provided. This can be done in your favorite editor and uploading to your server, or directly editing the file via SSH, but the easiest method would be to use ",(0,d.jsx)(n.a,{href:"https://github.com/coder/code-server",children:"Code Server"})]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"UI JSON Editor"})," ",(0,d.jsx)(n.em,{children:"(4/5 reliability, 4/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"From the UI, under the config menu there is a JSON editor, with built-in validation, documentation and advanced options"}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"UI Visual Editor"})," ",(0,d.jsx)(n.em,{children:"(3/5 reliability, 5/5 usability)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"From the UI, enter the Interactive Edit Mode, then click any part of the page to edit. Changes are previewed live, and then saved to disk"}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.strong,{children:"REST API"})," ",(0,d.jsx)(n.em,{children:"(Coming soon)"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"Programmatically edit config either through the command line, using a script or a third-party application"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsx)(n.h2,{id:"tips",children:"Tips"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:["You may find it helpful to look at some sample config files to get you started, a collection of which can be found ",(0,d.jsx)(n.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"here"})]}),"\n",(0,d.jsxs)(n.li,{children:["You can check that your config file fits the schema, by running ",(0,d.jsx)(n.code,{children:"yarn validate-config"})]}),"\n",(0,d.jsxs)(n.li,{children:["After modifying your config, the app needs to be recompiled, by running ",(0,d.jsx)(n.code,{children:"yarn build"})," - this happens automatically if you're using Docker"]}),"\n",(0,d.jsxs)(n.li,{children:["It is recommended to keep a backup of your config file. You can download it under Config menu, or use the ",(0,d.jsx)(n.a,{href:"/docs/backup-restore",children:"Cloud Backup"})," feature."]}),"\n",(0,d.jsx)(n.li,{children:"You can make use of YAML features, like anchors, comments, multi-line strings, etc to reuse attributes and keep your config file readable"}),"\n",(0,d.jsxs)(n.li,{children:["Once you have finished configuring your dashboard, you can choose to ",(0,d.jsx)(n.a,{href:"#preventing-changes",children:"disable UI config"})," if you wish"]}),"\n",(0,d.jsx)(n.li,{children:"All fields are optional, unless otherwise stated."}),"\n"]}),"\n",(0,d.jsx)(n.p,{children:"The following file provides a reference of all supported configuration options."}),"\n",(0,d.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pageinfo",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pageInfo"})})})," - Header text, footer, title, navigation, etc\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pageinfonavlinks-optional",children:(0,d.jsx)(n.code,{children:"navLinks"})})," - Links to display in the navigation bar"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#pages-optional",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pages"})})})," - List of additional config files, for multi-page dashboards"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfig-optional",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"appConfig"})})})," - Main application settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigwebsearch-optional",children:(0,d.jsx)(n.code,{children:"webSearch"})})," - Configure web search engine options"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfighidecomponents-optional",children:(0,d.jsx)(n.code,{children:"hideComponents"})})," - Show/ hide page components"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauth-optional",children:(0,d.jsx)(n.code,{children:"auth"})})," - Built-in authentication setup\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthusers-optional",children:(0,d.jsx)(n.code,{children:"users"})})," - List or users (for simple auth)"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthkeycloak-optional",children:(0,d.jsx)(n.code,{children:"keycloak"})})," - Auth config for Keycloak"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#appconfigauthheaderauth-optional",children:(0,d.jsx)(n.code,{children:"headerAuth"})})," - Auth config for HeaderAuth"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#section",children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sections"})})})," - List of sections\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})," - Section display settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:(0,d.jsx)(n.code,{children:"show/hideForKeycloakUsers"})})," - Set user controls"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"icon"})})," - Icon for a section"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionitem",children:(0,d.jsx)(n.code,{children:"items"})})," - List of items\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"icon"})})," - Icon for an item"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#itemdisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})," - Item display settings\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:(0,d.jsx)(n.code,{children:"show/hideForKeycloakUsers"})})," - Set user controls"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#sectionwidgets-optional",children:(0,d.jsx)(n.code,{children:"widgets"})})," - List of widgets"]}),"\n"]}),"\n"]}),"\n",(0,d.jsxs)(n.li,{children:[(0,d.jsx)(n.a,{href:"#notes",children:(0,d.jsx)(n.strong,{children:"Notes"})}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#editing-config-through-the-ui",children:"Editing Config through the UI"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#about-yaml",children:"About YAML"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#config-saving-methods",children:"Config Saving Methods"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#preventing-changes",children:"Preventing Changes"})}),"\n",(0,d.jsx)(n.li,{children:(0,d.jsx)(n.a,{href:"#example",children:"Example"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,d.jsx)(n.hr,{}),"\n",(0,d.jsx)(n.h2,{id:"top-level-fields",children:"Top-Level Fields"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pageInfo"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["Basic meta data like title, description, nav bar links, footer text. See ",(0,d.jsx)(n.a,{href:"#pageinfo",children:(0,d.jsx)(n.code,{children:"pageInfo"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"appConfig"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Settings related to how the app functions, including API keys and global styles. See ",(0,d.jsx)(n.a,{href:"#appconfig-optional",children:(0,d.jsx)(n.code,{children:"appConfig"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sections"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["An array of sections, each containing an array of items, which will be displayed as links. See ",(0,d.jsx)(n.a,{href:"#section",children:(0,d.jsx)(n.code,{children:"section"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"pages"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array additional config files, used for multi-page dashboards. See ",(0,d.jsx)(n.a,{href:"#pages-optional",children:(0,d.jsx)(n.code,{children:"pages"})})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"pageinfo",children:(0,d.jsx)(n.code,{children:"PageInfo"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"Your dashboard title, displayed in the header and browser tab"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"description"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Description of your dashboard, also displayed as a subtitle"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"navLinks"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Optional list of a maximum of 6 links, which will be displayed in the navigation bar. See ",(0,d.jsx)(n.a,{href:"#pageinfonavlinks-optional",children:(0,d.jsx)(n.code,{children:"navLinks"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"footer"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Text to display in the footer. When omitted, no footer is rendered. Supports inline HTML (sanitized before render)"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"logo"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The path to an image to display in the header (to the right of the title). This can be either local, where ",(0,d.jsx)(n.code,{children:"/"})," is the root of ",(0,d.jsx)(n.code,{children:"./public"}),", or any remote image, such as ",(0,d.jsx)(n.code,{children:"https://i.ibb.co/yhbt6CY/dashy.png"}),". It's recommended to scale your image down, so that it doesn't impact load times"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"favicon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["URL or path to a custom favicon shown in the browser tab. Can be absolute (",(0,d.jsx)(n.code,{children:"https://..."}),"), root-relative (",(0,d.jsx)(n.code,{children:"/icons/x.png"}),"), or a ",(0,d.jsx)(n.code,{children:"data:"})," URI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme colour applied to the browser chrome (mobile address bar). Any valid CSS color (e.g. ",(0,d.jsx)(n.code,{children:"#ff00a7"}),") is accepted"]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"pageinfonavlinks-optional",children:[(0,d.jsx)(n.code,{children:"pageInfo.navLinks"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The text to display on the link button"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"path"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The URL to navigate to when clicked. Can be relative (e.g. ",(0,d.jsx)(n.code,{children:"/about"}),") or absolute (e.g. ",(0,d.jsx)(n.code,{children:"https://example.com"})," or ",(0,d.jsx)(n.code,{children:"http://192.168.1.1"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"target"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The opening method (external links only). Can be either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"pages-optional",children:[(0,d.jsx)(n.code,{children:"pages[]"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"name"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"A unique name for that page"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"path"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The path (local or remote) to the config file to use.",(0,d.jsx)(n.br,{}),"For files located within ",(0,d.jsx)(n.code,{children:"/user-data"}),", you only need to specify filename, for externally hosted files you must include the full URL"]})]})]})]}),"\n",(0,d.jsxs)(n.p,{children:["For more info, see the",(0,d.jsx)(n.a,{href:"/docs/pages-and-sections#multi-page-support",children:"Multi-Page docs"})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfig-optional",children:[(0,d.jsx)(n.code,{children:"appConfig"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"language"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The 2 (or 4-digit) ",(0,d.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes",children:"ISO 639-1 code"})," for your language, e.g. ",(0,d.jsx)(n.code,{children:"en"})," or ",(0,d.jsx)(n.code,{children:"en-GB"}),". This must be a language that the app has already been ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/assets/locales",children:"translated"})," into. If your language is unavailable, Dashy will fallback to English. By default Dashy will attempt to auto-detect your language, although this may not work on some privacy browsers."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"startingView"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Which view to land on when visiting ",(0,d.jsx)(n.code,{children:"/"}),". One of ",(0,d.jsx)(n.code,{children:"home"}),", ",(0,d.jsx)(n.code,{children:"minimal"})," or ",(0,d.jsx)(n.code,{children:"workspace"}),". Defaults to ",(0,d.jsx)(n.code,{children:"home"}),". Applied at runtime, so no rebuild is needed. You can always switch views from the UI. (Legacy value ",(0,d.jsx)(n.code,{children:"default"})," is accepted as an alias for ",(0,d.jsx)(n.code,{children:"home"}),".)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"defaultOpeningMethod"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The default opening method for items, if no ",(0,d.jsx)(n.code,{children:"target"})," is specified for a given item. Can be either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"modal"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),", ",(0,d.jsx)(n.code,{children:"clipboard"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheck"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", Dashy will ping each of your services and display their status as a dot next to each item. This can be overridden by setting ",(0,d.jsx)(n.code,{children:"statusCheck"})," under each item. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckInterval"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The number of seconds between checks. If set to ",(0,d.jsx)(n.code,{children:"0"})," then service will only be checked on initial page load, which is usually the desired functionality. If value is less than ",(0,d.jsx)(n.code,{children:"10"})," you may experience a hit in performance. Defaults to ",(0,d.jsx)(n.code,{children:"0"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAccessibility"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", status indicators will use distinct shapes to indicate status for color-blind users. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"webSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Configuration options for the web search feature, set your default search engine, opening method or disable web search. See ",(0,d.jsx)(n.a,{href:"#appconfigwebsearch-optional",children:(0,d.jsx)(n.code,{children:"webSearch"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"backgroundImg"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Path to an optional full-screen app background image. This can be either remote (http) or local (relative to /app/public/item-icons/ inside the container). Note that this will slow down initial load"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableFontAwesome"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"})," font-awesome will be loaded, if set to ",(0,d.jsx)(n.code,{children:"false"})," they will not be. if left blank font-awesome will be enabled only if required by 1 or more icons"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableMaterialDesignIcons"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"})," mdi icons will be loaded, if set to ",(0,d.jsx)(n.code,{children:"false"})," they will not be. Where ",(0,d.jsx)(n.code,{children:"true"})," is enabled, if left blank material design icons will be enabled only if required by 1 or more icons"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"fontAwesomeKey"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you have a font-awesome key, then you can use it here and make use of premium icons. It is a 10-digit alpha-numeric string from you're FA kit URL (e.g. ",(0,d.jsx)(n.code,{children:"13014ae648"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"faviconApi"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Only applicable if you are using favicons for item icons. Specifies which service to use to resolve favicons. Set to ",(0,d.jsx)(n.code,{children:"local"})," to do this locally, without using an API. Services running locally will use this option always. Available options are: ",(0,d.jsx)(n.code,{children:"local"}),", ",(0,d.jsx)(n.code,{children:"allesedv"}),", ",(0,d.jsx)(n.code,{children:"iconhorse"}),", ",(0,d.jsx)(n.code,{children:"faviconkit"}),", ",(0,d.jsx)(n.code,{children:"duckduckgo"}),", ",(0,d.jsx)(n.code,{children:"yandex"}),", ",(0,d.jsx)(n.code,{children:"google"}),", ",(0,d.jsx)(n.code,{children:"besticon"}),", ",(0,d.jsx)(n.code,{children:"webmasterapi"})," and ",(0,d.jsx)(n.code,{children:"mcapi"}),". Defaults to ",(0,d.jsx)(n.code,{children:"allesedv"}),". See ",(0,d.jsx)(n.a,{href:"/docs/icons#favicons",children:"Icons"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"auth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["All settings relating to user authentication. See ",(0,d.jsx)(n.a,{href:"#appconfigauth-optional",children:(0,d.jsx)(n.code,{children:"auth"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"defaultIcon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An icon to be applied to any items, which don't already have an icon set. See ",(0,d.jsx)(n.a,{href:"/docs/icons#default-icon",children:"Icon Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"layout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Layout for homepage, one of ",(0,d.jsx)(n.code,{children:"auto"}),", ",(0,d.jsx)(n.code,{children:"horizontal"}),", ",(0,d.jsx)(n.code,{children:"vertical"})," or ",(0,d.jsx)(n.code,{children:"masonry"}),". Defaults to ",(0,d.jsx)(n.code,{children:"auto"}),". Specifies the layout and direction of how sections are positioned on the home screen. ",(0,d.jsx)(n.code,{children:"auto"})," uses a responsive CSS grid where each section's footprint is controlled by ",(0,d.jsx)(n.code,{children:"displayData.cols"})," and ",(0,d.jsx)(n.code,{children:"displayData.rows"}),". ",(0,d.jsx)(n.code,{children:"masonry"})," uses a responsive grid where heights follow content, so shorter sections flow into the gaps left by taller neighbours (",(0,d.jsx)(n.code,{children:"rows"})," is ignored, ",(0,d.jsx)(n.code,{children:"cols"})," still controls width). This can also be modified and overridden from the UI."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"iconSize"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"enum"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The size of link items / icons. Can be either ",(0,d.jsx)(n.code,{children:"small"}),", ",(0,d.jsx)(n.code,{children:"medium,"})," or ",(0,d.jsx)(n.code,{children:"large"}),". Defaults to ",(0,d.jsx)(n.code,{children:"medium"}),". This can also be set directly from the UI."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"colCount"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The number of columns of sections displayed on the homepage, using the default view. Should be in integer between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"8"}),". Note that by default this is applied responsively, based on current screen size, and specifying a value here will override this behavior, which may not be desirable."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"contentMaxWidth"})})}),(0,d.jsxs)(n.td,{children:[(0,d.jsx)(n.code,{children:"string"})," or ",(0,d.jsx)(n.code,{children:"number"})]}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Sets the max width of the main sections area on the homepage, overriding the responsive default. Can be a percentage, or any CSS unit"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"theme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The default theme for first load (you can change this later from the UI)"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"dayTheme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme to apply when the OS is set to light mode. Overrides ",(0,d.jsx)(n.code,{children:"theme"})," on initial load"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"nightTheme"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Theme to apply when the OS is set to dark mode. Overrides ",(0,d.jsx)(n.code,{children:"theme"})," on initial load"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cssThemes"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An array of custom theme names which can be used in the theme switcher dropdown"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customColors"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Enables you to apply a custom color palette to any given theme. Use the theme name (lowercase) as the key, for an object including key-value-pairs, with the color variable name as keys, and 6-digit hex code as value. See ",(0,d.jsx)(n.a,{href:"/docs/theming#modifying-theme-colors",children:"Theming"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"externalStyleSheet"})})}),(0,d.jsxs)(n.td,{children:[(0,d.jsx)(n.code,{children:"string"})," or ",(0,d.jsx)(n.code,{children:"string[]"})]}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Either a URL to an external stylesheet or an array or URLs, which can be applied as themes within the UI"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customCss"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Raw CSS that will be applied to the page. This can also be set from the UI. Please minify it first."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideComponents"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A list of key page components (header, nav, search, settings) that are present by default, but can be removed using this option. See ",(0,d.jsx)(n.a,{href:"#appconfighidecomponents-optional",children:(0,d.jsx)(n.code,{children:"appConfig.hideComponents"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableMultiTasking"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, will keep apps open in the background when in the workspace view. Useful for quickly switching between multiple sites, and preserving their state, but comes at the cost of performance."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"workspaceLandingUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The URL or an app, service or website to launch when the workspace view is opened, before another service has been launched"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"preventWriteToDisk"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", users will be prevented from saving config changes to disk through the UI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"preventLocalSave"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", users will be prevented from applying config changes to local storage"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableConfiguration"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, no users will be able to view or edit the config through the UI"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If set to true, only admin users will be able to view or edit the config through the UI. disableConfiguration must not be set to true."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"widgetsAlwaysUseProxy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", requests made by widgets will always be proxied, same as setting ",(0,d.jsx)(n.code,{children:"useProxy: true"})," on each widget. Note that this may break some widgets."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showSplashScreen"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", a loading screen will be shown. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableErrorReporting"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Enable reporting of unexpected errors and crashes. This is off by default, and ",(0,d.jsx)(n.strong,{children:"no data will ever be captured unless you explicitly enable it"}),". Turning on error reporting helps previously unknown bugs get discovered and fixed. Dashy uses ",(0,d.jsx)(n.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," for error reporting. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sentryDsn"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you need to monitor errors in your instance, then you can use Sentry to collect and process bug reports. Sentry can be self-hosted, or used as SaaS, once your instance is setup, then all you need to do is pass in the DSN here, and enable error reporting. You can learn more on the ",(0,d.jsx)(n.a,{href:"https://docs.sentry.io/product/sentry-basics/dsn-explainer/",children:"Sentry DSN Docs"}),". Note that this will only ever be used if ",(0,d.jsx)(n.code,{children:"enableErrorReporting"})," is explicitly enabled."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableSmartSort"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["For the most-used and last-used app sort functions to work, a basic open-count is stored in local storage. If you do not want this to happen, then disable smart sort here, but you wil no longer be able to use these sort options. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableUpdateChecks"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to true, Dashy will not check for updates. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableServiceWorker"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Service workers cache web applications to improve load times and offer basic offline functionality, and are disabled by default in Dashy. The service worker can sometimes cause older content to be cached, requiring the app to be hard-refreshed. If you do not want SW functionality, or are having issues with caching, set this property to ",(0,d.jsx)(n.code,{children:"false"})," to disable all service workers."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableContextMenu"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the custom right-click context menu will be disabled. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauth-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n",(0,d.jsx)(n.admonition,{type:"note",children:(0,d.jsxs)(n.p,{children:["Since the auth is initiated in the main app entry point (for security), a rebuild is required to apply changes to the auth configuration.\nRun ",(0,d.jsx)(n.code,{children:"yarn build"})," in the root directory, then restart the server."]})}),"\n",(0,d.jsx)(n.admonition,{type:"warning",children:(0,d.jsxs)(n.p,{children:["Built-in auth should ",(0,d.jsx)(n.strong,{children:"not be used"})," for security-critical applications, or if your Dashy instance is publicly accessible.\nFor these, it is recommended to use an ",(0,d.jsx)(n.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternate authentication method"}),"."]})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"users"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of objects containing usernames and hashed passwords. If this is not provided, then authentication will be off by default, and you will not need any credentials to access the app. See ",(0,d.jsx)(n.a,{href:"#appconfigauthusers-optional",children:(0,d.jsx)(n.code,{children:"appConfig.auth.users"})}),". ",(0,d.jsx)(n.br,{}),(0,d.jsx)(n.strong,{children:"Note"})," this method of authentication is handled on the client side, so for security critical situations, it is recommended to use an ",(0,d.jsx)(n.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternate authentication method"}),"."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableKeycloak"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using Keycloak will be enabled. Note that you need to have an instance running, and have also configured ",(0,d.jsx)(n.code,{children:"auth.keycloak"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"keycloak"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dashy to your Keycloak server. Requires ",(0,d.jsx)(n.code,{children:"enableKeycloak: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthkeycloak-optional",children:(0,d.jsx)(n.code,{children:"auth.keycloak"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableHeaderAuth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using HeaderAuth will be enabled. Note that you need to have your web server/reverse proxy running, and have also configured ",(0,d.jsx)(n.code,{children:"auth.headerAuth"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"headerAuth"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dashy to your headers for authentication. Requires ",(0,d.jsx)(n.code,{children:"enableHeaderAuth: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthheaderauth-optional",children:(0,d.jsx)(n.code,{children:"auth.headerAuth"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableOidc"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", then authentication using OIDC will be enabled. Note that you need to have a configured OIDC server and configure it with ",(0,d.jsx)(n.code,{children:"auth.oidc"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"oidc"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Config options to point Dash to your OIDC configuration. Request ",(0,d.jsx)(n.code,{children:"enableOidc: true"}),". See ",(0,d.jsx)(n.a,{href:"#appconfigauthoidc-optional",children:(0,d.jsx)(n.code,{children:"auth.oidc"})})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"enableGuestAccess"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", an unauthenticated user will be able to access the dashboard, with read-only access, without having to login. Requires ",(0,d.jsx)(n.code,{children:"auth.users"})," to be configured. Defaults to ",(0,d.jsx)(n.code,{children:"false"}),"."]})]})]})]}),"\n",(0,d.jsxs)(n.p,{children:["For more info, see the ",(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"/docs/authentication",children:"Authentication Docs"})})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthusers-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.users"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"user"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"Username to log in with"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hash"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"A SHA-256 hashed password"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"type"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The user type, either admin or normal"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthkeycloak-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.keycloak"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"serverUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The URL (or URL/ IP + Port) where your keycloak server is running"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"realm"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The name of the realm (must already be created) that you want to use"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"clientId"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The Client ID of the client you created for use with Dashy"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"idpHint"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The alias of an Identity Provider configured in your realm. If set, Keycloak will skip its login page and redirect straight to that external IdP"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"legacySupport"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If using Keycloak 17 or older, then set this to ",(0,d.jsx)(n.code,{children:"true"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthheaderauth-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.headerAuth"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"userHeader"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The HTTP header name containing the authenticated username (default: ",(0,d.jsx)(n.code,{children:"Remote-User"}),"). Case insensitive"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"proxyWhitelist"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"An array of upstream proxy server IPs to accept authenticated requests from"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigauthoidc-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.auth.oidc"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"clientId"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The client id registered in the OIDC server"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"endpoint"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The URL of the OIDC server that should be used."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"adminRole"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The role that will be considered as admin."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"adminGroup"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The group that will be considered as admin."})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"scope"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The scope(s) to request from the OIDC provider"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfigwebsearch-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.webSearch"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"disableWebSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Web search is enabled by default, but can be disabled by setting this property to ",(0,d.jsx)(n.code,{children:"true"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"searchEngine"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Set the key name for your search engine. Can also use a custom engine by setting this property to ",(0,d.jsx)(n.code,{children:"custom"}),". Currently supported: ",(0,d.jsx)(n.code,{children:"duckduckgo"}),", ",(0,d.jsx)(n.code,{children:"google"}),", ",(0,d.jsx)(n.code,{children:"whoogle"}),", ",(0,d.jsx)(n.code,{children:"qwant"}),", ",(0,d.jsx)(n.code,{children:"startpage"}),", ",(0,d.jsx)(n.code,{children:"searx-bar"})," and ",(0,d.jsx)(n.code,{children:"searx-info"}),". Defaults to ",(0,d.jsx)(n.code,{children:"duckduckgo"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customSearchEngine"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["You can also use a custom search engine, or your own self-hosted instance. This requires ",(0,d.jsx)(n.code,{children:"searchEngine: custom"})," to be set. Then add the URL of your service, with GET query string included here"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"openingMethod"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Set your preferred opening method for search results: ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),". Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"searchBangs"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A key-value-pair set of custom search ",(0,d.jsx)(n.em,{children:"bangs"})," for redirecting query to a specific app or search engine. The key of each should be the bang you will type (typically starting with ",(0,d.jsx)(n.code,{children:"/"}),", ",(0,d.jsx)(n.code,{children:"!"})," or ",(0,d.jsx)(n.code,{children:":"}),"), and value is the destination, either as a search engine key (e.g. ",(0,d.jsx)(n.code,{children:"reddit"}),") or a URL with search parameters (e.g. ",(0,d.jsx)(n.code,{children:"https://en.wikipedia.org/w/?search="}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"openUrlsDirectly"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If ",(0,d.jsx)(n.code,{children:"true"}),", queries that look like URLs will be opened directly instead of searched. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"appconfighidecomponents-optional",children:[(0,d.jsx)(n.code,{children:"appConfig.hideComponents"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideHeading"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the page title & sub-title will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideNav"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the navigation menu will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideSearch"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the search bar will not be visible. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideSettings"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If set to ",(0,d.jsx)(n.code,{children:"true"}),", the settings menu will be initially collapsed. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"section",children:(0,d.jsx)(n.code,{children:"section"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"name"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsx)(n.td,{children:"The title for the section"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An single icon to be displayed next to the title. See ",(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"section.icon"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"items"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of items to be displayed within the section. See ",(0,d.jsx)(n.a,{href:"#sectionitem",children:(0,d.jsx)(n.code,{children:"item"})}),". Sections must include either 1 or more items, or 1 or more widgets."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"widgets"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An array of widgets to be displayed within the section. See ",(0,d.jsx)(n.a,{href:"#sectionwidgets-optional",children:(0,d.jsx)(n.code,{children:"widget"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"displayData"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Meta-data to optionally override display settings for a given section. See ",(0,d.jsx)(n.a,{href:"#sectiondisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.h2,{id:"sectionitem",children:(0,d.jsx)(n.code,{children:"section.item"})}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"title"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The text to display/ title of a given item. Max length ",(0,d.jsx)(n.code,{children:"18"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"description"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Additional info about an item, which is shown in the tooltip on hover, or visible on large tiles"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"url"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The URL / location of web address for when the item is clicked"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The icon for a given item. Can be a font-awesome icon, favicon, remote URL or local URL. See ",(0,d.jsx)(n.a,{href:"#sectionicon-and-sectionitemicon",children:(0,d.jsx)(n.code,{children:"item.icon"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"target"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The opening method for when the item is clicked, either ",(0,d.jsx)(n.code,{children:"newtab"}),", ",(0,d.jsx)(n.code,{children:"sametab"}),", ",(0,d.jsx)(n.code,{children:"modal"}),", ",(0,d.jsx)(n.code,{children:"workspace"}),", ",(0,d.jsx)(n.code,{children:"clipboard"}),", ",(0,d.jsx)(n.code,{children:"top"})," or ",(0,d.jsx)(n.code,{children:"parent"}),". Where ",(0,d.jsx)(n.code,{children:"newtab"})," will open the link in a new tab, ",(0,d.jsx)(n.code,{children:"sametab"})," will open it in the current tab, and ",(0,d.jsx)(n.code,{children:"modal"})," will open a pop-up modal, ",(0,d.jsx)(n.code,{children:"workspace"})," will open in the Workspace view and ",(0,d.jsx)(n.code,{children:"clipboard"})," will copy the URL to system clipboard (but not launch app). Defaults to ",(0,d.jsx)(n.code,{children:"newtab"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hotkey"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Give frequently opened applications a numeric hotkey, between ",(0,d.jsx)(n.code,{children:"0 - 9"}),". You can then just press that key to launch that application."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"tags"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"A list of tags, which can be used for improved search"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheck"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["When set to ",(0,d.jsx)(n.code,{children:"true"}),", Dashy will ping the URL associated with the current service, and display its status as a dot next to the item. The value here will override ",(0,d.jsx)(n.code,{children:"appConfig.statusCheck"})," so you can turn off or on checks for a given service. Defaults to ",(0,d.jsx)(n.code,{children:"appConfig.statusCheck"}),", falls back to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckUrl"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If you've enabled ",(0,d.jsx)(n.code,{children:"statusCheck"}),", and want to use a different URL to what is defined under the item, then specify it here"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckHeaders"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"If you're endpoint requires any specific headers for the status checking, then define them here"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAllowInsecure"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["By default, any request to insecure content will be blocked. Setting this option to ",(0,d.jsx)(n.code,{children:"true"})," will disable the ",(0,d.jsx)(n.code,{children:"rejectUnauthorized"})," option, enabling you to ping non-HTTPS services for the current item. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckAcceptCodes"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If your service's response code is anything other than 2xx, then you can opt to specify an alternative success code. E.g. if you expect your server to return 403, but still want the status indicator to be green, set this value to ",(0,d.jsx)(n.code,{children:"403"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"statusCheckMaxRedirects"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If your service redirects to another page, and you would like status checks to follow redirects, then specify the maximum number of redirects here. Defaults to ",(0,d.jsx)(n.code,{children:"0"})," / will not follow redirects"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An optional color for the text and font-awesome icon to be displayed in. Note that this will override the current theme and so may not display well"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"rel"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The value of the ",(0,d.jsx)(n.code,{children:"rel"})," attribute for the link. Useful for specifying the relationship between the target link/document and Dashy. Defaults to ",(0,d.jsx)(n.code,{children:"noopener noreferrer"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"backgroundColor"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"An optional background fill color for the that given item. Again, this will override the current theme and so might not display well against the background"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"provider"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"The name of the provider for a given service, useful for when including hosted apps. In some themes, this is visible under the item name"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"displayData"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Meta-data to optionally override display settings for a given item. See ",(0,d.jsx)(n.a,{href:"#itemdisplaydata-optional",children:(0,d.jsx)(n.code,{children:"displayData"})})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"subItems"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"array"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["An optional list of nested sub-items, rendered as smaller icons within the parent. Each sub-item supports ",(0,d.jsx)(n.code,{children:"title"}),", ",(0,d.jsx)(n.code,{children:"url"}),", ",(0,d.jsx)(n.code,{children:"icon"}),", ",(0,d.jsx)(n.code,{children:"target"}),", ",(0,d.jsx)(n.code,{children:"color"})," and ",(0,d.jsx)(n.code,{children:"backgroundColor"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"itemdisplaydata-optional",children:[(0,d.jsx)(n.code,{children:"item.displayData"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current item will be visible to all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current item will be hidden from all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForGuests"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible for logged in users, but not for guests (see ",(0,d.jsx)(n.code,{children:"appConfig.enableGuestAccess"}),"). Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible to all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be hidden from all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideFromWorkspace"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current item will be visible in the default view but not in the Workspace view sidebar. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectionwidgets-optional",children:[(0,d.jsx)(n.code,{children:"section.widgets"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"type"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:"Required"}),(0,d.jsxs)(n.td,{children:["The widget type. See ",(0,d.jsx)(n.a,{href:"/docs/widgets",children:"Widget Docs"})," for full list of supported widgets"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"options"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Some widgets accept either optional or required additional options. Again, see the ",(0,d.jsx)(n.a,{href:"/docs/widgets",children:"Widget Docs"})," for full list of options"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"updateInterval"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["You can keep a widget constantly updated by specifying an update interval, in seconds. See ",(0,d.jsx)(n.a,{href:"/docs/widgets#continuous-updates",children:"Continuous Updates Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"useProxy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Some widgets make API requests to services that are not CORS-enabled. For these instances, you will need to route requests through a proxy, Dashy has a built in CORS-proxy, which you can use by setting this option to ",(0,d.jsx)(n.code,{children:"true"}),". Defaults to ",(0,d.jsx)(n.code,{children:"false"}),". See the ",(0,d.jsx)(n.a,{href:"/docs/widgets#proxying-requests",children:"Proxying Requests Docs"})," for more info"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"timeout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Request timeout in milliseconds, defaults to \xbd a second (",(0,d.jsx)(n.code,{children:"500"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"ignoreErrors"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Prevent an error message being displayed, if a network request or something else fails. Useful for false-positives"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"label"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Add custom label to a given widget. Useful for identification, if there are multiple of the same type of widget in a single section"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectiondisplaydata-optional",children:[(0,d.jsx)(n.code,{children:"section.displayData"})," ",(0,d.jsx)(n.em,{children:"(optional)"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sortBy"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The sort order for items within the current section. By default items are displayed in the order in which they are listed in within the config. The following sort options are supported: ",(0,d.jsx)(n.code,{children:"most-used"})," (most opened apps first), ",(0,d.jsx)(n.code,{children:"last-used"})," (the most recently used apps), ",(0,d.jsx)(n.code,{children:"alphabetical"}),", ",(0,d.jsx)(n.code,{children:"reverse-alphabetical"}),", ",(0,d.jsx)(n.code,{children:"random"})," and ",(0,d.jsx)(n.code,{children:"default"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"collapsed"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["If true, the section will be collapsed initially, and will need to be clicked to open. Useful for less regularly used, or very long sections. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cutToHeight"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"By default, sections will fill available space. Set this option to true to match section height with content height"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"rows"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Height of the section, specified as the number of rows it should span vertically, e.g. ",(0,d.jsx)(n.code,{children:"2"}),". Defaults to ",(0,d.jsx)(n.code,{children:"1"}),". Max is ",(0,d.jsx)(n.code,{children:"5"}),". Applies to the default ",(0,d.jsx)(n.code,{children:"auto"})," layout; ignored by ",(0,d.jsx)(n.code,{children:"masonry"})," (where heights follow content) and the flex-based ",(0,d.jsx)(n.code,{children:"horizontal"}),"/",(0,d.jsx)(n.code,{children:"vertical"})," layouts."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"cols"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Width of the section, specified as the number of columns the section should span horizontally, e.g. ",(0,d.jsx)(n.code,{children:"2"}),". Defaults to ",(0,d.jsx)(n.code,{children:"1"}),". Max is ",(0,d.jsx)(n.code,{children:"5"}),". Will be clamped to the page's active column count so that a section never exceeds the available grid width."]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemSize"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Specify the size for items within this group, either ",(0,d.jsx)(n.code,{children:"small"}),", ",(0,d.jsx)(n.code,{children:"medium"})," or ",(0,d.jsx)(n.code,{children:"large"}),". Note that this will override any settings specified through the UI"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"color"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["A custom accent color for the section, as a hex code or HTML color (e.g. ",(0,d.jsx)(n.code,{children:"#fff"}),")"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"customStyles"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Custom CSS properties that should be applied to that section, e.g. ",(0,d.jsx)(n.code,{children:"border: 2px dashed #ff0000;"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"sectionLayout"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Specify which CSS layout will be used to responsively place items. Can be either ",(0,d.jsx)(n.code,{children:"auto"})," (which uses flex layout), or ",(0,d.jsx)(n.code,{children:"grid"}),". Defaults to ",(0,d.jsx)(n.code,{children:"auto"}),". Setting ",(0,d.jsx)(n.code,{children:"itemCountX"})," or ",(0,d.jsx)(n.code,{children:"itemCountY"})," below will also switch the section to grid layout automatically"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemCountX"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Number of items per row / horizontally. If not set, it will be calculated automatically based on available space. Setting this switches the section to grid layout. Must be a whole number between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"12"}),"; values above ",(0,d.jsx)(n.code,{children:"8"})," rely on the grid's responsive column sizing (no explicit minimum-width rule)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"itemCountY"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"number"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Number of explicit rows before items flow into implicit rows. Setting this switches the section to grid layout. Must be a whole number between ",(0,d.jsx)(n.code,{children:"1"})," and ",(0,d.jsx)(n.code,{children:"12"}),". Row heights size to their content (section heights follow content in the masonry layout)"]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current section will be visible to all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current section will be hidden from all users, except for those specified in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForGuests"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible for logged in users, but not for guests (see ",(0,d.jsx)(n.code,{children:"appConfig.enableGuestAccess"}),"). Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible to all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"hideForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"object"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be hidden from all keycloak users, except for those configured via these groups and roles. See ",(0,d.jsx)(n.code,{children:"showForKeycloakUsers"})]})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"hideFromWorkspace"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"boolean"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["Current section will be visible in the default view but not in the Workspace view sidebar. Defaults to ",(0,d.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectionicon-and-sectionitemicon",children:[(0,d.jsx)(n.code,{children:"section.icon"})," and ",(0,d.jsx)(n.code,{children:"section.item.icon"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsx)(n.tbody,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"icon"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsxs)(n.td,{children:["The icon for a given item or section. ",(0,d.jsx)(n.br,{}),"See ",(0,d.jsx)(n.a,{href:"/docs/icons",children:"Icon Docs"})," for all available supported icon types, including: auto-fetched favicons, generative icons, emoji icons, home-lab service logos, font-awesome, simple-icons, material icons, selfh.st icons, and icons specified by URL"]})]})})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsxs)(n.h2,{id:"sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers",children:[(0,d.jsx)(n.code,{children:"section.displayData.hideForKeycloakUsers"}),", ",(0,d.jsx)(n.code,{children:"section.displayData.showForKeycloakUsers"}),", ",(0,d.jsx)(n.code,{children:"item.displayData.hideForKeycloakUsers"})," and ",(0,d.jsx)(n.code,{children:"item.displayData.showForKeycloakUsers"})]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,d.jsxs)(n.table,{children:[(0,d.jsx)(n.thead,{children:(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Field"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Type"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Required"})}),(0,d.jsx)(n.th,{children:(0,d.jsx)(n.strong,{children:"Description"})})]})}),(0,d.jsxs)(n.tbody,{children:[(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"groups"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current Section or Item will be hidden or shown based on the user having any of the groups in this list"})]}),(0,d.jsxs)(n.tr,{children:[(0,d.jsx)(n.td,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.code,{children:"roles"})})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.code,{children:"string[]"})}),(0,d.jsx)(n.td,{children:(0,d.jsx)(n.em,{children:"Optional"})}),(0,d.jsx)(n.td,{children:"Current Section or Item will be hidden or shown based on the user having any of the roles in this list"})]})]})]}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,d.jsx)(n.hr,{}),"\n",(0,d.jsx)(n.h2,{id:"notes",children:"Notes"}),"\n",(0,d.jsx)(n.h3,{id:"editing-config-through-the-ui",children:"Editing Config through the UI"}),"\n",(0,d.jsxs)(n.p,{children:["Config can be modified directly through the UI, and then written to disk, or applied locally. This can be done wither with the raw config editor (introduced in V 0.6.5 / ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/pull/3",children:"#3"}),"), or the interactive editor (introduced in V 1.8.9 / ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/pull/298",children:"#298"}),")."]}),"\n ",(0,d.jsxs)(n.a,{href:"https://ibb.co/CzkyMNb",children:["\n ",(0,d.jsx)(n.b,{children:"Interactive Editor"}),(0,d.jsx)(n.br,{}),"\n ",(0,d.jsx)(n.img,{alt:"Interactive Editor demo",src:"https://user-images.githubusercontent.com/1862727/139543020-b0576d28-0830-476f-afc8-a815d4de6def.gif",width:"600"}),"\n "]}),"\n ",(0,d.jsx)(n.br,{}),"\n ",(0,d.jsxs)(n.a,{href:"https://ibb.co/zRv542H",children:["\n ",(0,d.jsx)(n.b,{children:"JSON Editor"}),(0,d.jsx)(n.br,{}),"\n ",(0,d.jsx)(n.img,{alt:"Config Editor demo",src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/config-editor-demo.gif",width:"600"}),"\n "]}),"\n",(0,d.jsx)(n.h3,{id:"about-yaml",children:"About YAML"}),"\n",(0,d.jsxs)(n.p,{children:["If you're new to YAML, it's pretty straight-forward. The format is exactly the same as that of JSON, but instead of using curly braces, structure is denoted using whitespace. This ",(0,d.jsx)(n.a,{href:"https://linuxhandbook.com/yaml-basics/",children:"quick guide"})," should get you up to speed in a few minutes, for more advanced topics take a look at this ",(0,d.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/YAML",children:"Wikipedia article"}),"."]}),"\n",(0,d.jsx)(n.h3,{id:"config-saving-methods",children:"Config Saving Methods"}),"\n",(0,d.jsxs)(n.p,{children:["When updating the config through the JSON editor in the UI, you have two save options: ",(0,d.jsx)(n.strong,{children:"Local"})," or ",(0,d.jsx)(n.strong,{children:"Write to Disk"}),"."]}),"\n",(0,d.jsxs)(n.ul,{children:["\n",(0,d.jsx)(n.li,{children:"Changes saved locally will only be applied to the current user through the browser, and will not apply to other instances - you either need to use the cloud sync feature, or manually update the conf.yml file."}),"\n",(0,d.jsxs)(n.li,{children:["On the other-hand, if you choose to write changes to disk, then your main ",(0,d.jsx)(n.code,{children:"conf.yml"})," file will be updated, and changes will be applied to all users, and visible across all devices. For this functionality to work, you must be running Dashy with using the Docker container, or the Node server. A backup of your current configuration will also be saved in the same directory."]}),"\n"]}),"\n",(0,d.jsx)(n.h3,{id:"preventing-changes",children:"Preventing Changes"}),"\n",(0,d.jsxs)(n.p,{children:["If you have authentication set up, then any user who is not an admin (with ",(0,d.jsx)(n.code,{children:"type: admin"}),") will not be able to write changes to disk."]}),"\n",(0,d.jsxs)(n.p,{children:["You can also prevent changes from any user being written to disk, using ",(0,d.jsx)(n.code,{children:"preventWriteToDisk"}),". Or prevent any changes from being saved locally in browser storage, using ",(0,d.jsx)(n.code,{children:"preventLocalSave"}),"."]}),"\n",(0,d.jsxs)(n.p,{children:["To disable all UI config features, set ",(0,d.jsx)(n.code,{children:"disableConfiguration"}),". Alternatively you can disable UI config features for all non Admin users by setting ",(0,d.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})," to true."]}),"\n",(0,d.jsx)(n.h3,{id:"example",children:"Example"}),"\n",(0,d.jsx)(n.pre,{children:(0,d.jsx)(n.code,{className:"language-yaml",children:"---\npageInfo:\n title: Home Lab\nsections: # An array of sections\n- name: Section 1 - Getting Started\n items: # An array of items\n - title: GitHub\n description: Source code and documentation on GitHub\n icon: fab fa-github\n url: https://github.com/Lissy93/dashy\n - title: Issues\n description: View currently open issues, or raise a new one\n icon: fas fa-bug\n url: https://github.com/Lissy93/dashy/issues\n - title: Demo\n description: A live demo\n icon: far fa-rocket\n url: https://dashy-demo-1.netlify.app\n- name: Section 2 - Local Services\n items:\n - title: Firewall\n icon: favicon\n url: http://192.168.1.1/\n - title: Game Server\n icon: https://i.ibb.co/710B3Yc/space-invader-x256.png\n url: http://192.168.130.1/\n"})}),"\n",(0,d.jsxs)(n.p,{children:["For more example config files, see: ",(0,d.jsx)(n.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"this gist"})]}),"\n",(0,d.jsxs)(n.p,{children:["If you need any help, feel free to ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=Lissy93&labels=%F0%9F%A4%B7%E2%80%8D%E2%99%82%EF%B8%8F+Question&template=question.md&title=%5BQUESTION%5D",children:"Raise an Issue"})," or ",(0,d.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions",children:"Start a Discussion"})]}),"\n",(0,d.jsx)(n.p,{children:"Happy Configuring \ud83e\udd13\ud83d\udd27"}),"\n",(0,d.jsx)(n.p,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.strong,{children:(0,d.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})})]})}function a(e={}){const{wrapper:n}={...(0,r.R)(),...e.components};return n?(0,d.jsx)(n,{...e,children:(0,d.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>c});var i=s(6540);const d={},r=i.createContext(d);function t(e){const n=i.useContext(r);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function c(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(d):e.components||d:t(e.components),i.createElement(r.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/4d54d076.a1ec67fb.js b/assets/js/4d54d076.4433267d.js similarity index 99% rename from assets/js/4d54d076.a1ec67fb.js rename to assets/js/4d54d076.4433267d.js index 80f9fb91..380b3f1e 100644 --- a/assets/js/4d54d076.a1ec67fb.js +++ b/assets/js/4d54d076.4433267d.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1459],{2199(e,s,t){t.r(s),t.d(s,{assets:()=>l,contentTitle:()=>n,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>h});const o=JSON.parse('{"id":"contributing","title":"Contributing","description":"First off, thank you for considering contributing towards Dashy! \ud83d\ude4c","source":"@site/docs/contributing.md","sourceDirName":".","slug":"/contributing","permalink":"/docs/contributing","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/contributing.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"*Dashy Showcase* \ud83c\udf1f","permalink":"/docs/showcase"},"next":{"title":"Developing","permalink":"/docs/developing"}}');var a=t(4848),i=t(8453);const r={},n="Contributing",l={},h=[{value:"Take a 2-minute survey",id:"take-a-2-minute-survey",level:2},{value:"Share your dashboard",id:"share-your-dashboard",level:2},{value:"Make a small donation",id:"make-a-small-donation",level:2},{value:"You can also send a one-off small contribution using crypto",id:"you-can-also-send-a-one-off-small-contribution-using-crypto",level:3},{value:"Enable Anonymous Bug Reports",id:"enable-anonymous-bug-reports",level:2},{value:"Add Translations",id:"add-translations",level:2},{value:"Submit a PR",id:"submit-a-pr",level:2},{value:"Improve the Docs",id:"improve-the-docs",level:2},{value:"Raise a bug",id:"raise-a-bug",level:2},{value:"Join the discussion",id:"join-the-discussion",level:2},{value:"Request a feature via BountySource",id:"request-a-feature-via-bountysource",level:2},{value:"Spread the word",id:"spread-the-word",level:2},{value:"Star, Upvote or Leave a Review",id:"star-upvote-or-leave-a-review",level:2},{value:"Follow for More",id:"follow-for-more",level:2},{value:"Contributors",id:"contributors",level:3},{value:"Star-Gazers Over Time",id:"star-gazers-over-time",level:3}];function d(e){const s={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,i.R)(),...e.components};return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(s.header,{children:(0,a.jsx)(s.h1,{id:"contributing",children:"Contributing"})}),"\n",(0,a.jsxs)(s.p,{children:["First off, thank you for considering contributing towards Dashy! \ud83d\ude4c\nThere are several ways that you can help out, and any contributions, however small will always be very much appreciated.\nYou will be appropriately credited in the readme - huge thank you to ",(0,a.jsx)(s.a,{href:"/docs/credits",children:"everyone who has helped"})," so far \ud83d\udc9e"]}),"\n",(0,a.jsx)(s.h2,{id:"take-a-2-minute-survey",children:"Take a 2-minute survey"}),"\n",(0,a.jsx)(s.p,{children:"Help improve Dashy by taking a very short, 6-question survey. This will give me a better understanding of what is important to you, so that I can make Dashy better in the future :)"}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://survey.typeform.com/to/gl0L68ou",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Take_the-Survey-%231a86fd?style=for-the-badge&logo=buddy",alt:"Take the Survey"})})}),"\n",(0,a.jsx)(s.h2,{id:"share-your-dashboard",children:"Share your dashboard"}),"\n",(0,a.jsxs)(s.p,{children:["Dashy now has a ",(0,a.jsx)(s.a,{href:"/docs/showcase",children:"Showcase"})," where you can show off a screenshot of your dashboard, and get inspiration from other users (and I really love seeing how people are using Dashy). To ",(0,a.jsx)(s.a,{href:"/docs/showcase#submitting-your-dashboard",children:"submit your dashboard"}),", either open a PR or raise an issue."]}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=&labels=%F0%9F%92%AF+Showcase&template=showcase-addition.yml&title=%5BSHOWCASE%5D+%3Ctitle%3E",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Add_your_Dashboard-Showcase-%238616ee?style=for-the-badge&logo=feathub&logoColor=8616ee",alt:"Add your Dashboard to the Showcase"})})}),"\n",(0,a.jsx)(s.h2,{id:"make-a-small-donation",children:"Make a small donation"}),"\n",(0,a.jsx)(s.p,{children:"Donations help to cover server costs, development time and caffeine ;)\nDon't feel any pressure to donate anything, as Dashy and my other projects will always be 100% free, for everyone, for ever."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/sponsors/Lissy93",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Sponsor_on_GitHub-Lissy93-%23ff4dda?style=for-the-badge&logo=githubsponsors&logoColor=ff4dda",alt:"Sponsor Lissy93 on GitHub"})})}),"\n",(0,a.jsxs)(s.p,{children:["Sponsoring will give you several perks - for $1 / \xa30.75 per month, you'll get a sponsor badge on your profile, be credited on the Dashy's readme, with a link to your website/ profile/ socials, get priority support, have your feature ideas implemented, plus lots more. For more info, see ",(0,a.jsx)(s.a,{href:"https://github.com/sponsors/Lissy93",children:"@Lissy93's Sponsor Page"}),"."]}),"\n",(0,a.jsx)(s.h3,{id:"you-can-also-send-a-one-off-small-contribution-using-crypto",children:"You can also send a one-off small contribution using crypto"}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://en.cryptobadges.io/donate/3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC",children:(0,a.jsx)(s.img,{src:"https://en.cryptobadges.io/badge/big/3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC",alt:"Donate with BTC"})}),(0,a.jsx)(s.a,{href:"https://en.cryptobadges.io/donate/0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017",children:(0,a.jsx)(s.img,{src:"https://en.cryptobadges.io/badge/big/0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017",alt:"Donate with Ethereum"})})]}),"\n",(0,a.jsxs)(s.ul,{children:["\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"BTC"}),": ",(0,a.jsx)(s.code,{children:"3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"ETH"}),": ",(0,a.jsx)(s.code,{children:"0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017"})," / ",(0,a.jsx)(s.code,{children:"aliciasykes.eth"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"XMR"}),": ",(0,a.jsx)(s.code,{children:"471KZdxb6N63aABR4WYwMRjTVkc1p1x7wGsUTEF7AMYzL8L94A5pCuYWkosgJQ5Ze8Y2PscVCGZFJa3hDPg6MaDq47GUm8r"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"LTC"}),": ",(0,a.jsx)(s.code,{children:"MAuck6Ea1qaNihwKfXutkR1R6BorMth86H"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"ZEC"}),": ",(0,a.jsx)(s.code,{children:"t1bw1SefijsXRDQVxC9w64XsRK8hBhtQohQ"})]}),"\n"]}),"\n",(0,a.jsx)(s.h2,{id:"enable-anonymous-bug-reports",children:"Enable Anonymous Bug Reports"}),"\n",(0,a.jsxs)(s.p,{children:["Bug reports helps me to discover bugs I was unaware of, and then fix them, in order to make Dashy more reliable long term. This is a simple, yet really helpful step you can take to help improve Dashy. ",(0,a.jsx)(s.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," is an open source error tracking and performance monitoring tool, which enables the identification any errors which occur in the production app (only if you enable it)."]}),"\n",(0,a.jsx)(s.p,{children:"To enable error reporting:"}),"\n",(0,a.jsx)(s.pre,{children:(0,a.jsx)(s.code,{className:"language-yaml",children:"appConfig:\n enableErrorReporting: true\n"})}),"\n",(0,a.jsxs)(s.p,{children:["All reporting is ",(0,a.jsx)(s.strong,{children:"disabled"})," by default, and no data will ever be sent to any external endpoint without your explicit consent. All statistics are anonymized and stored securely. For more about privacy and security, see the ",(0,a.jsx)(s.a,{href:"https://sentry.io/security/",children:"Sentry Security Docs"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"add-translations",children:"Add Translations"}),"\n",(0,a.jsxs)(s.p,{children:["If you speak another language, then adding translations will help make Dashy available to non-native English speakers. This is a very quick and easy task, as all application text is located in ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/assets/locales/en.json",children:(0,a.jsx)(s.code,{children:"locales/en.json"})}),", so adding a new language is as simple as copying this file and translating the values. You don't have to translate it all, as any missing attributes will just fallback to English. For a full tutorial, see the ",(0,a.jsx)(s.a,{href:"/docs/multi-language-support",children:"Multi-Language Support Docs"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"submit-a-pr",children:"Submit a PR"}),"\n",(0,a.jsxs)(s.p,{children:["Contributing to the code or docs is super helpful. You can fix a bug, add a new feature or improve an existing one. If you've built your own custom widget, theme or view, consider sharing it in a PR. I've written ",(0,a.jsx)(s.a,{href:"/docs/development-guides",children:"several guides"})," to help you get started, and the steps for setting up the development environment are outlined in the ",(0,a.jsx)(s.a,{href:"/docs/developing",children:"Development Docs"}),". Feel free to ask if you have any questions."]}),"\n",(0,a.jsx)(s.h2,{id:"improve-the-docs",children:"Improve the Docs"}),"\n",(0,a.jsxs)(s.p,{children:["Found a typo, or something that isn't as clear as it could be? Maybe I've missed something off altogether, or you hit a roadblock that took you a while to figure out. Submitting a pull request to add to or improve the documentation will help future users get Dashy up and running more easily.\nAll content is located either in the ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/README",children:(0,a.jsx)(s.code,{children:"./README.md"})})," or ",(0,a.jsx)(s.a,{href:"/docs",children:(0,a.jsx)(s.code,{children:"/docs/"})})," directory, and synced to the Wiki and website using a GH ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/actions/workflows/wiki-sync.yml",children:"action"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"raise-a-bug",children:"Raise a bug"}),"\n",(0,a.jsx)(s.p,{children:"If you've found a bug, then please do raise it as an issue. This will help me know if there's something that needs fixing. Try and include as much detail as possible, such as your environment, steps to reproduce, any console output and maybe an example screenshot or recording if necessary."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=lissy93&labels=%F0%9F%90%9B+Bug&template=bug.yml&title=%5BBUG%5D+%3Ctitle%3E",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Raise_a-Bug-%23dc2d76?style=for-the-badge&logo=dependabot",alt:"Raise a Bug"})})}),"\n",(0,a.jsx)(s.h2,{id:"join-the-discussion",children:"Join the discussion"}),"\n",(0,a.jsx)(s.p,{children:"I've enabled the discussion feature on GitHub, here you can share tips and tricks, useful information, or your dashboard. You can also ask questions, and offer basic support to other users."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/discussions",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Join_the-Discussion-%23ffd000?style=for-the-badge&logo=livechat",alt:"Join the Discussion on GitHub"})})}),"\n",(0,a.jsx)(s.h2,{id:"request-a-feature-via-bountysource",children:"Request a feature via BountySource"}),"\n",(0,a.jsx)(s.p,{children:"BountySource is a platform for sponsoring the development of certain features on open source projects. If there is a feature you'd like implemented into Dashy, but either isn't high enough priority or is deemed to be more work than it's worth, then you can instead contribute a bounty towards it's development. You won't pay a penny until your proposal is fully built, and you are satisfied with the result. This helps support the developers, and makes Dashy better for everyone."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://www.bountysource.com/teams/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/BountySource-Dashy-%23F67909?style=for-the-badge&logo=openbugbounty",alt:"Request a Feature on BountySource"})})}),"\n",(0,a.jsx)(s.h2,{id:"spread-the-word",children:"Spread the word"}),"\n",(0,a.jsx)(s.p,{children:"Dashy is still a relatively young project, and as such not many people know of it. It would be great to see more users, and so it would be awesome if you could consider sharing with your friends or on social platforms."}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://mastodon.social/?text=Check%20out%20Dashy%2C%20the%20privacy-friendly%2C%20self-hosted%20startpage%20for%20organizing%20your%20life%3A%20https%3A%2F%2Fgithub.com%2FLissy93%2Fdashy%20-%20By%20%40lissy93%40mastodon.social",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Mastodon-%232b90d9?style=flat-square&logo=mastodon",alt:"Share Dashy on Mastodon"})}),"\n",(0,a.jsx)(s.a,{href:"http://www.reddit.com/submit?url=https://github.com/Lissy93/dashy&title=Dashy%20-%20The%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Reddit-%23FF5700?style=flat-square&logo=reddit",alt:"Share Dashy on Reddit"})}),"\n",(0,a.jsx)(s.a,{href:"https://twitter.com/intent/tweet?url=https://github.com/lissy93/dashy&text=Check%20out%20Dashy%20by%20@Lissy_Sykes,%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Twitter-%231DA1F2?style=flat-square&logo=twitter",alt:"Share Dashy on Twitter"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.facebook.com/sharer/sharer.php?u=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Facebook-%234267B2?style=flat-square&logo=facebook",alt:"Share Dashy on Facebook"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.linkedin.com/shareArticle?mini=true&url=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-LinkedIn-%230077b5?style=flat-square&logo=linkedin",alt:"Share Dashy on LinkedIn"})}),"\n",(0,a.jsx)(s.a,{href:"https://pinterest.com/pin/create/button/?url=https://github.com/lissy93/dashy&media=https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/1-home-lab-material.png&description=Check%20out%20Dashy,%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Pinterest-%23E60023?style=flat-square&logo=pinterest",alt:"Share Dashy on Pinterest"})}),"\n",(0,a.jsx)(s.a,{href:"https://vk.com/share.php?url=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy%2F&title=Check%20out%20Dashy%20-%20The%20Self-Hosted%20Dashboard%20for%20your%20Homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-VK-%234C75A3?style=flat-square&logo=vk",alt:"Share Dashy on VK"})}),"\n",(0,a.jsx)(s.a,{href:"viber://forward?text=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy%0ACheck%20out%20Dashy%2C%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Viber-%238176d6?style=flat-square&logo=viber",alt:"Share Dashy via Viber"})}),"\n",(0,a.jsx)(s.a,{href:"https://t.me/share/url?url=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy&text=Check%20out%20Dashy%2C%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Telegram-%230088cc?style=flat-square&logo=telegram",alt:"Share Dashy via Telegram"})}),"\n",(0,a.jsx)(s.a,{href:"mailto:info@example.com?&subject=Check%20out%20Dashy%20-%20The%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80&cc=&bcc=&body=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Email-%238A90C7?style=flat-square&logo=protonmail",alt:"Share Dashy via Email"})})]}),"\n",(0,a.jsx)(s.h2,{id:"star-upvote-or-leave-a-review",children:"Star, Upvote or Leave a Review"}),"\n",(0,a.jsx)(s.p,{children:"Dashy is on the following platforms, and if you could spare a few seconds to give it an upvote or review, this will also help new users discover Dashy"}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://www.producthunt.com/posts/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-ProductHunt-%23b74424?style=flat-square&logo=producthunt",alt:"ProductHunt"})}),"\n",(0,a.jsx)(s.a,{href:"https://alternativeto.net/software/dashy/about/",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-AlternativeTo-%235581a6?style=flat-square&logo=abletonlive",alt:"AlternativeTo"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.slant.co/improve/topics/27783/viewpoints/1/~self-hosted-homelab-startpage~dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-Slant-%2346a1df?style=flat-square&logo=capacitor",alt:"Slant"})}),"\n",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/stargazers",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/github/stars/Lissy93/Dashy?color=ba96d6&label=Star%20-%20GitHub&logo=github&style=flat-square",alt:"Star on GitHub"})}),"\n",(0,a.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/docker/stars/lissy93/dashy?color=4cb6e0&label=Star%20-%20Docker&logo=docker&style=flat-square",alt:"Star on DockerHub"})})]}),"\n",(0,a.jsx)(s.h2,{id:"follow-for-more",children:"Follow for More"}),"\n",(0,a.jsx)(s.p,{children:"If you've enjoyed Dashy, you can follow the me to get updates about other projects that I am working on."}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://twitter.com/Lissy_Sykes",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/twitter/follow/Lissy_Sykes?style=social&logo=twitter",alt:"Alicia Sykes on Twitter"})}),"\n",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/github/followers/lissy93?label=Lissy93&style=social",alt:"Alicia Sykes on GitHub"})}),"\n",(0,a.jsx)(s.a,{href:"https://mastodon.social/web/accounts/1032965",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/mastodon/follow/1032965?domain=https%3A%2F%2Fmastodon.social",alt:"Alicia Sykes on Mastodon"})}),"\n",(0,a.jsx)(s.a,{href:"https://keybase.io/aliciasykes",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/aliciasykes--lightgrey?style=social&logo=Keybase",alt:"Alicia Sykes on Keybase"})}),"\n",(0,a.jsx)(s.a,{href:"https://aliciasykes.com",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/aliciasykes.com--lightgrey?style=social&logo=Tencent%20QQ",alt:"Alicia Sykes's Website"})}),"\n",(0,a.jsx)(s.a,{href:"https://notes.aliciasykes.com/",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Blog--lightgrey?style=social&logo=micro.blog",alt:"Alicia Sykes's Blog"})}),"\n",(0,a.jsx)(s.a,{href:"https://keybase.io/aliciasykes/pgp_keys.asc",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/PGP--lightgrey?style=social&logo=Let%E2%80%99s%20Encrypt",alt:"Alicia Sykes's PGP"})})]}),"\n",(0,a.jsxs)(s.p,{children:["If you like, you could also consider ",(0,a.jsx)(s.a,{href:"https://notes.aliciasykes.com/subscribe",children:"subscribing to my mailing list"})," for occasional blog post updates."]}),"\n",(0,a.jsx)(s.hr,{}),"\n",(0,a.jsx)(s.h3,{id:"contributors",children:"Contributors"}),"\n",(0,a.jsxs)(s.p,{children:["For a full list of Dashy's contributors, see the ",(0,a.jsx)(s.a,{href:"/docs/credits",children:"Credits Page"})]}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"/docs/credits",children:(0,a.jsx)(s.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/CONTRIBUTORS.svg",alt:"Auto-generated contributors"})})}),"\n",(0,a.jsx)(s.h3,{id:"star-gazers-over-time",children:"Star-Gazers Over Time"}),"\n",(0,a.jsxs)(s.p,{children:["](",(0,a.jsx)(s.a,{href:"https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy",children:"https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy"}),")"]})]})}function c(e={}){const{wrapper:s}={...(0,i.R)(),...e.components};return s?(0,a.jsx)(s,{...e,children:(0,a.jsx)(d,{...e})}):d(e)}},8453(e,s,t){t.d(s,{R:()=>r,x:()=>n});var o=t(6540);const a={},i=o.createContext(a);function r(e){const s=o.useContext(i);return o.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function n(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(a):e.components||a:r(e.components),o.createElement(i.Provider,{value:s},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1459],{2199(e,s,t){t.r(s),t.d(s,{assets:()=>l,contentTitle:()=>n,default:()=>c,frontMatter:()=>r,metadata:()=>o,toc:()=>h});const o=JSON.parse('{"id":"contributing","title":"Contributing","description":"First off, thank you for considering contributing towards Dashy! \ud83d\ude4c","source":"@site/docs/contributing.md","sourceDirName":".","slug":"/contributing","permalink":"/docs/contributing","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/contributing.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"*Dashy Showcase* \ud83c\udf1f","permalink":"/docs/showcase"},"next":{"title":"Developing","permalink":"/docs/developing"}}');var a=t(4848),i=t(8453);const r={},n="Contributing",l={},h=[{value:"Take a 2-minute survey",id:"take-a-2-minute-survey",level:2},{value:"Share your dashboard",id:"share-your-dashboard",level:2},{value:"Make a small donation",id:"make-a-small-donation",level:2},{value:"You can also send a one-off small contribution using crypto",id:"you-can-also-send-a-one-off-small-contribution-using-crypto",level:3},{value:"Enable Anonymous Bug Reports",id:"enable-anonymous-bug-reports",level:2},{value:"Add Translations",id:"add-translations",level:2},{value:"Submit a PR",id:"submit-a-pr",level:2},{value:"Improve the Docs",id:"improve-the-docs",level:2},{value:"Raise a bug",id:"raise-a-bug",level:2},{value:"Join the discussion",id:"join-the-discussion",level:2},{value:"Request a feature via BountySource",id:"request-a-feature-via-bountysource",level:2},{value:"Spread the word",id:"spread-the-word",level:2},{value:"Star, Upvote or Leave a Review",id:"star-upvote-or-leave-a-review",level:2},{value:"Follow for More",id:"follow-for-more",level:2},{value:"Contributors",id:"contributors",level:3},{value:"Star-Gazers Over Time",id:"star-gazers-over-time",level:3}];function d(e){const s={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,i.R)(),...e.components};return(0,a.jsxs)(a.Fragment,{children:[(0,a.jsx)(s.header,{children:(0,a.jsx)(s.h1,{id:"contributing",children:"Contributing"})}),"\n",(0,a.jsxs)(s.p,{children:["First off, thank you for considering contributing towards Dashy! \ud83d\ude4c\nThere are several ways that you can help out, and any contributions, however small will always be very much appreciated.\nYou will be appropriately credited in the readme - huge thank you to ",(0,a.jsx)(s.a,{href:"/docs/credits",children:"everyone who has helped"})," so far \ud83d\udc9e"]}),"\n",(0,a.jsx)(s.h2,{id:"take-a-2-minute-survey",children:"Take a 2-minute survey"}),"\n",(0,a.jsx)(s.p,{children:"Help improve Dashy by taking a very short, 6-question survey. This will give me a better understanding of what is important to you, so that I can make Dashy better in the future :)"}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://survey.typeform.com/to/gl0L68ou",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Take_the-Survey-%231a86fd?style=for-the-badge&logo=buddy",alt:"Take the Survey"})})}),"\n",(0,a.jsx)(s.h2,{id:"share-your-dashboard",children:"Share your dashboard"}),"\n",(0,a.jsxs)(s.p,{children:["Dashy now has a ",(0,a.jsx)(s.a,{href:"/docs/showcase",children:"Showcase"})," where you can show off a screenshot of your dashboard, and get inspiration from other users (and I really love seeing how people are using Dashy). To ",(0,a.jsx)(s.a,{href:"/docs/showcase#submitting-your-dashboard",children:"submit your dashboard"}),", either open a PR or raise an issue."]}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=&labels=%F0%9F%92%AF+Showcase&template=showcase-addition.yml&title=%5BSHOWCASE%5D+%3Ctitle%3E",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Add_your_Dashboard-Showcase-%238616ee?style=for-the-badge&logo=feathub&logoColor=8616ee",alt:"Add your Dashboard to the Showcase"})})}),"\n",(0,a.jsx)(s.h2,{id:"make-a-small-donation",children:"Make a small donation"}),"\n",(0,a.jsx)(s.p,{children:"Donations help to cover server costs, development time and caffeine ;)\nDon't feel any pressure to donate anything, as Dashy and my other projects will always be 100% free, for everyone, for ever."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/sponsors/Lissy93",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Sponsor_on_GitHub-Lissy93-%23ff4dda?style=for-the-badge&logo=githubsponsors&logoColor=ff4dda",alt:"Sponsor Lissy93 on GitHub"})})}),"\n",(0,a.jsxs)(s.p,{children:["Sponsoring will give you several perks - for $1 / \xa30.75 per month, you'll get a sponsor badge on your profile, be credited on the Dashy's readme, with a link to your website/ profile/ socials, get priority support, have your feature ideas implemented, plus lots more. For more info, see ",(0,a.jsx)(s.a,{href:"https://github.com/sponsors/Lissy93",children:"@Lissy93's Sponsor Page"}),"."]}),"\n",(0,a.jsx)(s.h3,{id:"you-can-also-send-a-one-off-small-contribution-using-crypto",children:"You can also send a one-off small contribution using crypto"}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://en.cryptobadges.io/donate/3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC",children:(0,a.jsx)(s.img,{src:"https://en.cryptobadges.io/badge/big/3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC",alt:"Donate with BTC"})}),(0,a.jsx)(s.a,{href:"https://en.cryptobadges.io/donate/0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017",children:(0,a.jsx)(s.img,{src:"https://en.cryptobadges.io/badge/big/0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017",alt:"Donate with Ethereum"})})]}),"\n",(0,a.jsxs)(s.ul,{children:["\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"BTC"}),": ",(0,a.jsx)(s.code,{children:"3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"ETH"}),": ",(0,a.jsx)(s.code,{children:"0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017"})," / ",(0,a.jsx)(s.code,{children:"aliciasykes.eth"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"XMR"}),": ",(0,a.jsx)(s.code,{children:"471KZdxb6N63aABR4WYwMRjTVkc1p1x7wGsUTEF7AMYzL8L94A5pCuYWkosgJQ5Ze8Y2PscVCGZFJa3hDPg6MaDq47GUm8r"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"LTC"}),": ",(0,a.jsx)(s.code,{children:"MAuck6Ea1qaNihwKfXutkR1R6BorMth86H"})]}),"\n",(0,a.jsxs)(s.li,{children:[(0,a.jsx)(s.strong,{children:"ZEC"}),": ",(0,a.jsx)(s.code,{children:"t1bw1SefijsXRDQVxC9w64XsRK8hBhtQohQ"})]}),"\n"]}),"\n",(0,a.jsx)(s.h2,{id:"enable-anonymous-bug-reports",children:"Enable Anonymous Bug Reports"}),"\n",(0,a.jsxs)(s.p,{children:["Bug reports helps me to discover bugs I was unaware of, and then fix them, in order to make Dashy more reliable long term. This is a simple, yet really helpful step you can take to help improve Dashy. ",(0,a.jsx)(s.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," is an open source error tracking and performance monitoring tool, which enables the identification any errors which occur in the production app (only if you enable it)."]}),"\n",(0,a.jsx)(s.p,{children:"To enable error reporting:"}),"\n",(0,a.jsx)(s.pre,{children:(0,a.jsx)(s.code,{className:"language-yaml",children:"appConfig:\n enableErrorReporting: true\n"})}),"\n",(0,a.jsxs)(s.p,{children:["All reporting is ",(0,a.jsx)(s.strong,{children:"disabled"})," by default, and no data will ever be sent to any external endpoint without your explicit consent. All statistics are anonymized and stored securely. For more about privacy and security, see the ",(0,a.jsx)(s.a,{href:"https://sentry.io/security/",children:"Sentry Security Docs"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"add-translations",children:"Add Translations"}),"\n",(0,a.jsxs)(s.p,{children:["If you speak another language, then adding translations will help make Dashy available to non-native English speakers. This is a very quick and easy task, as all application text is located in ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/assets/locales/en.json",children:(0,a.jsx)(s.code,{children:"locales/en.json"})}),", so adding a new language is as simple as copying this file and translating the values. You don't have to translate it all, as any missing attributes will just fallback to English. For a full tutorial, see the ",(0,a.jsx)(s.a,{href:"/docs/multi-language-support",children:"Multi-Language Support Docs"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"submit-a-pr",children:"Submit a PR"}),"\n",(0,a.jsxs)(s.p,{children:["Contributing to the code or docs is super helpful. You can fix a bug, add a new feature or improve an existing one. If you've built your own custom widget, theme or view, consider sharing it in a PR. I've written ",(0,a.jsx)(s.a,{href:"/docs/development-guides",children:"several guides"})," to help you get started, and the steps for setting up the development environment are outlined in the ",(0,a.jsx)(s.a,{href:"/docs/developing",children:"Development Docs"}),". Feel free to ask if you have any questions."]}),"\n",(0,a.jsx)(s.h2,{id:"improve-the-docs",children:"Improve the Docs"}),"\n",(0,a.jsxs)(s.p,{children:["Found a typo, or something that isn't as clear as it could be? Maybe I've missed something off altogether, or you hit a roadblock that took you a while to figure out. Submitting a pull request to add to or improve the documentation will help future users get Dashy up and running more easily.\nAll content is located either in the ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/README",children:(0,a.jsx)(s.code,{children:"./README.md"})})," or ",(0,a.jsx)(s.a,{href:"/docs",children:(0,a.jsx)(s.code,{children:"/docs/"})})," directory, and synced to the Wiki and website using a GH ",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/actions/workflows/wiki-sync.yml",children:"action"}),"."]}),"\n",(0,a.jsx)(s.h2,{id:"raise-a-bug",children:"Raise a bug"}),"\n",(0,a.jsx)(s.p,{children:"If you've found a bug, then please do raise it as an issue. This will help me know if there's something that needs fixing. Try and include as much detail as possible, such as your environment, steps to reproduce, any console output and maybe an example screenshot or recording if necessary."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=lissy93&labels=%F0%9F%90%9B+Bug&template=bug.yml&title=%5BBUG%5D+%3Ctitle%3E",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Raise_a-Bug-%23dc2d76?style=for-the-badge&logo=dependabot",alt:"Raise a Bug"})})}),"\n",(0,a.jsx)(s.h2,{id:"join-the-discussion",children:"Join the discussion"}),"\n",(0,a.jsx)(s.p,{children:"I've enabled the discussion feature on GitHub, here you can share tips and tricks, useful information, or your dashboard. You can also ask questions, and offer basic support to other users."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/discussions",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Join_the-Discussion-%23ffd000?style=for-the-badge&logo=livechat",alt:"Join the Discussion on GitHub"})})}),"\n",(0,a.jsx)(s.h2,{id:"request-a-feature-via-bountysource",children:"Request a feature via BountySource"}),"\n",(0,a.jsx)(s.p,{children:"BountySource is a platform for sponsoring the development of certain features on open source projects. If there is a feature you'd like implemented into Dashy, but either isn't high enough priority or is deemed to be more work than it's worth, then you can instead contribute a bounty towards it's development. You won't pay a penny until your proposal is fully built, and you are satisfied with the result. This helps support the developers, and makes Dashy better for everyone."}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"https://www.bountysource.com/teams/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/BountySource-Dashy-%23F67909?style=for-the-badge&logo=openbugbounty",alt:"Request a Feature on BountySource"})})}),"\n",(0,a.jsx)(s.h2,{id:"spread-the-word",children:"Spread the word"}),"\n",(0,a.jsx)(s.p,{children:"Dashy is still a relatively young project, and as such not many people know of it. It would be great to see more users, and so it would be awesome if you could consider sharing with your friends or on social platforms."}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://mastodon.social/?text=Check%20out%20Dashy%2C%20the%20privacy-friendly%2C%20self-hosted%20startpage%20for%20organizing%20your%20life%3A%20https%3A%2F%2Fgithub.com%2FLissy93%2Fdashy%20-%20By%20%40lissy93%40mastodon.social",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Mastodon-%232b90d9?style=flat-square&logo=mastodon",alt:"Share Dashy on Mastodon"})}),"\n",(0,a.jsx)(s.a,{href:"http://www.reddit.com/submit?url=https://github.com/Lissy93/dashy&title=Dashy%20-%20The%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Reddit-%23FF5700?style=flat-square&logo=reddit",alt:"Share Dashy on Reddit"})}),"\n",(0,a.jsx)(s.a,{href:"https://twitter.com/intent/tweet?url=https://github.com/lissy93/dashy&text=Check%20out%20Dashy%20by%20@Lissy_Sykes,%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Twitter-%231DA1F2?style=flat-square&logo=twitter",alt:"Share Dashy on Twitter"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.facebook.com/sharer/sharer.php?u=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Facebook-%234267B2?style=flat-square&logo=facebook",alt:"Share Dashy on Facebook"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.linkedin.com/shareArticle?mini=true&url=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-LinkedIn-%230077b5?style=flat-square&logo=linkedin",alt:"Share Dashy on LinkedIn"})}),"\n",(0,a.jsx)(s.a,{href:"https://pinterest.com/pin/create/button/?url=https://github.com/lissy93/dashy&media=https://raw.githubusercontent.com/Lissy93/dashy/master/docs/showcase/1-home-lab-material.png&description=Check%20out%20Dashy,%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Pinterest-%23E60023?style=flat-square&logo=pinterest",alt:"Share Dashy on Pinterest"})}),"\n",(0,a.jsx)(s.a,{href:"https://vk.com/share.php?url=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy%2F&title=Check%20out%20Dashy%20-%20The%20Self-Hosted%20Dashboard%20for%20your%20Homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-VK-%234C75A3?style=flat-square&logo=vk",alt:"Share Dashy on VK"})}),"\n",(0,a.jsx)(s.a,{href:"viber://forward?text=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy%0ACheck%20out%20Dashy%2C%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Viber-%238176d6?style=flat-square&logo=viber",alt:"Share Dashy via Viber"})}),"\n",(0,a.jsx)(s.a,{href:"https://t.me/share/url?url=https%3A%2F%2Fgithub.com%2Flissy93%2Fdashy&text=Check%20out%20Dashy%2C%20the%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Telegram-%230088cc?style=flat-square&logo=telegram",alt:"Share Dashy via Telegram"})}),"\n",(0,a.jsx)(s.a,{href:"mailto:info@example.com?&subject=Check%20out%20Dashy%20-%20The%20self-hosted%20dashboard%20for%20your%20homelab%20%F0%9F%9A%80&cc=&bcc=&body=https://github.com/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Share-Email-%238A90C7?style=flat-square&logo=protonmail",alt:"Share Dashy via Email"})})]}),"\n",(0,a.jsx)(s.h2,{id:"star-upvote-or-leave-a-review",children:"Star, Upvote or Leave a Review"}),"\n",(0,a.jsx)(s.p,{children:"Dashy is on the following platforms, and if you could spare a few seconds to give it an upvote or review, this will also help new users discover Dashy"}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://www.producthunt.com/posts/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-ProductHunt-%23b74424?style=flat-square&logo=producthunt",alt:"ProductHunt"})}),"\n",(0,a.jsx)(s.a,{href:"https://alternativeto.net/software/dashy/about/",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-AlternativeTo-%235581a6?style=flat-square&logo=abletonlive",alt:"AlternativeTo"})}),"\n",(0,a.jsx)(s.a,{href:"https://www.slant.co/improve/topics/27783/viewpoints/1/~self-hosted-homelab-startpage~dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Review-Slant-%2346a1df?style=flat-square&logo=capacitor",alt:"Slant"})}),"\n",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/stargazers",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/github/stars/Lissy93/Dashy?color=ba96d6&label=Star%20-%20GitHub&logo=github&style=flat-square",alt:"Star on GitHub"})}),"\n",(0,a.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/docker/stars/lissy93/dashy?color=4cb6e0&label=Star%20-%20Docker&logo=docker&style=flat-square",alt:"Star on DockerHub"})})]}),"\n",(0,a.jsx)(s.h2,{id:"follow-for-more",children:"Follow for More"}),"\n",(0,a.jsx)(s.p,{children:"If you've enjoyed Dashy, you can follow the me to get updates about other projects that I am working on."}),"\n",(0,a.jsxs)(s.p,{children:[(0,a.jsx)(s.a,{href:"https://twitter.com/Lissy_Sykes",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/twitter/follow/Lissy_Sykes?style=social&logo=twitter",alt:"Alicia Sykes on Twitter"})}),"\n",(0,a.jsx)(s.a,{href:"https://github.com/Lissy93",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/github/followers/lissy93?label=Lissy93&style=social",alt:"Alicia Sykes on GitHub"})}),"\n",(0,a.jsx)(s.a,{href:"https://mastodon.social/web/accounts/1032965",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/mastodon/follow/1032965?domain=https%3A%2F%2Fmastodon.social",alt:"Alicia Sykes on Mastodon"})}),"\n",(0,a.jsx)(s.a,{href:"https://keybase.io/aliciasykes",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/aliciasykes--lightgrey?style=social&logo=Keybase",alt:"Alicia Sykes on Keybase"})}),"\n",(0,a.jsx)(s.a,{href:"https://aliciasykes.com",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/aliciasykes.com--lightgrey?style=social&logo=Tencent%20QQ",alt:"Alicia Sykes's Website"})}),"\n",(0,a.jsx)(s.a,{href:"https://notes.aliciasykes.com/",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/Blog--lightgrey?style=social&logo=micro.blog",alt:"Alicia Sykes's Blog"})}),"\n",(0,a.jsx)(s.a,{href:"https://keybase.io/aliciasykes/pgp_keys.asc",children:(0,a.jsx)(s.img,{src:"https://img.shields.io/badge/PGP--lightgrey?style=social&logo=Let%E2%80%99s%20Encrypt",alt:"Alicia Sykes's PGP"})})]}),"\n",(0,a.jsxs)(s.p,{children:["If you like, you could also consider ",(0,a.jsx)(s.a,{href:"https://notes.aliciasykes.com/subscribe",children:"subscribing to my mailing list"})," for occasional blog post updates."]}),"\n",(0,a.jsx)(s.hr,{}),"\n",(0,a.jsx)(s.h3,{id:"contributors",children:"Contributors"}),"\n",(0,a.jsxs)(s.p,{children:["For a full list of Dashy's contributors, see the ",(0,a.jsx)(s.a,{href:"/docs/credits",children:"Credits Page"})]}),"\n",(0,a.jsx)(s.p,{children:(0,a.jsx)(s.a,{href:"/docs/credits",children:(0,a.jsx)(s.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/CONTRIBUTORS.svg",alt:"Auto-generated contributors"})})}),"\n",(0,a.jsx)(s.h3,{id:"star-gazers-over-time",children:"Star-Gazers Over Time"}),"\n",(0,a.jsxs)(s.p,{children:["](",(0,a.jsx)(s.a,{href:"https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy",children:"https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy"}),")"]})]})}function c(e={}){const{wrapper:s}={...(0,i.R)(),...e.components};return s?(0,a.jsx)(s,{...e,children:(0,a.jsx)(d,{...e})}):d(e)}},8453(e,s,t){t.d(s,{R:()=>r,x:()=>n});var o=t(6540);const a={},i=o.createContext(a);function r(e){const s=o.useContext(i);return o.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function n(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(a):e.components||a:r(e.components),o.createElement(i.Provider,{value:s},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/52e2a80b.26b5b38e.js b/assets/js/52e2a80b.e96206a9.js similarity index 98% rename from assets/js/52e2a80b.26b5b38e.js rename to assets/js/52e2a80b.e96206a9.js index e57a6ca0..67ebebaa 100644 --- a/assets/js/52e2a80b.26b5b38e.js +++ b/assets/js/52e2a80b.e96206a9.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1306],{1264(e,n,s){s.r(n),s.d(n,{assets:()=>d,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"readme","title":"![Dashy Docs](https://i.ibb.co/4mdNf7M/heading-docs.png)","description":"Running Dashy","source":"@site/docs/readme.md","sourceDirName":".","slug":"/","permalink":"/docs/","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/readme.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{}}');var t=s(4848),o=s(8453);const r={},a="Dashy Docs",d={},l=[{value:"Running Dashy",id:"running-dashy",level:2},{value:"Development and Contributing",id:"development-and-contributing",level:2},{value:"Feature Docs",id:"feature-docs",level:2},{value:"Misc",id:"misc",level:2}];function c(e){const n={a:"a",h1:"h1",h2:"h2",header:"header",img:"img",li:"li",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"dashy-docs",children:(0,t.jsx)(n.img,{src:"https://i.ibb.co/4mdNf7M/heading-docs.png",alt:"Dashy Docs"})})}),"\n",(0,t.jsx)(n.h2,{id:"running-dashy",children:"Running Dashy"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/quick-start",children:"Quick Start"})," - TLDR guide on getting Dashy up and running"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/deployment",children:"Deployment"})," - Full guide on deploying Dashy either locally or online"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/configuring",children:"Configuring"})," - Complete list of all available options in the config file"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/management",children:"App Management"})," - Managing your app, updating, security, web server configuration, etc"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/troubleshooting",children:"Troubleshooting"})," - Common errors and problems, and how to fix them"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"development-and-contributing",children:"Development and Contributing"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/developing",children:"Developing"})," - Running Dashy development server locally, and general workflow"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/development-guides",children:"Development Guides"})," - Common development tasks, to help new contributors"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/contributing",children:"Contributing"})," - How you can help keep Dashy alive"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/showcase",children:"Showcase"})," - See how others are using Dashy, and share your dashboard"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/credits",children:"Credits"})," - List of people and projects that have made Dashy possible"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/release-workflow",children:"Release Workflow"})," - Info about releases, CI and automated tasks"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"feature-docs",children:"Feature Docs"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/authentication",children:"Authentication"})," - Guide to setting up authentication to protect your dashboard"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/alternate-views",children:"Alternate Views"})," - Outline of available pages / views and item opening methods"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/backup-restore",children:"Backup & Restore"})," - Guide to backing up config with Dashy's cloud sync feature"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/icons",children:"Icons"})," - Outline of all available icon types for sections and items, with examples"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/multi-language-support",children:"Language Switching"})," - Details on how to switch language, or add a new locale"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/pages-and-sections",children:"Pages and Sections"})," - Multi-page support, sections, items and sub-items"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/status-indicators",children:"Status Indicators"})," - Using Dashy to monitor uptime and status of your apps"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/searching",children:"Searching & Shortcuts"})," - Searching, launching methods + keyboard shortcuts"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/theming",children:"Theming"})," - Complete guide to applying, writing and modifying themes + styles"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/widgets",children:"Widgets"})," - List of all dynamic content widgets, with usage guides and examples"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"misc",children:"Misc"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/privacy",children:"Privacy & Security"})," - List of requests, potential issues, and security resources"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/LICENSE",children:"License"})," - Copy of the MIT License"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/CODE_OF_CONDUCT",children:"Code of Conduct"})," - Contributor Covenant Code of Conduct"]}),"\n"]})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(c,{...e})}):c(e)}},8453(e,n,s){s.d(n,{R:()=>r,x:()=>a});var i=s(6540);const t={},o=i.createContext(t);function r(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1306],{1264(e,n,s){s.r(n),s.d(n,{assets:()=>d,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"readme","title":"![Dashy Docs](https://i.ibb.co/4mdNf7M/heading-docs.png)","description":"Running Dashy","source":"@site/docs/readme.md","sourceDirName":".","slug":"/","permalink":"/docs/","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/readme.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{}}');var t=s(4848),o=s(8453);const r={},a="Dashy Docs",d={},l=[{value:"Running Dashy",id:"running-dashy",level:2},{value:"Development and Contributing",id:"development-and-contributing",level:2},{value:"Feature Docs",id:"feature-docs",level:2},{value:"Misc",id:"misc",level:2}];function c(e){const n={a:"a",h1:"h1",h2:"h2",header:"header",img:"img",li:"li",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"dashy-docs",children:(0,t.jsx)(n.img,{src:"https://i.ibb.co/4mdNf7M/heading-docs.png",alt:"Dashy Docs"})})}),"\n",(0,t.jsx)(n.h2,{id:"running-dashy",children:"Running Dashy"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/quick-start",children:"Quick Start"})," - TLDR guide on getting Dashy up and running"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/deployment",children:"Deployment"})," - Full guide on deploying Dashy either locally or online"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/configuring",children:"Configuring"})," - Complete list of all available options in the config file"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/management",children:"App Management"})," - Managing your app, updating, security, web server configuration, etc"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/troubleshooting",children:"Troubleshooting"})," - Common errors and problems, and how to fix them"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"development-and-contributing",children:"Development and Contributing"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/developing",children:"Developing"})," - Running Dashy development server locally, and general workflow"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/development-guides",children:"Development Guides"})," - Common development tasks, to help new contributors"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/contributing",children:"Contributing"})," - How you can help keep Dashy alive"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/showcase",children:"Showcase"})," - See how others are using Dashy, and share your dashboard"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/credits",children:"Credits"})," - List of people and projects that have made Dashy possible"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/release-workflow",children:"Release Workflow"})," - Info about releases, CI and automated tasks"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"feature-docs",children:"Feature Docs"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/authentication",children:"Authentication"})," - Guide to setting up authentication to protect your dashboard"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/alternate-views",children:"Alternate Views"})," - Outline of available pages / views and item opening methods"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/backup-restore",children:"Backup & Restore"})," - Guide to backing up config with Dashy's cloud sync feature"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/icons",children:"Icons"})," - Outline of all available icon types for sections and items, with examples"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/multi-language-support",children:"Language Switching"})," - Details on how to switch language, or add a new locale"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/pages-and-sections",children:"Pages and Sections"})," - Multi-page support, sections, items and sub-items"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/status-indicators",children:"Status Indicators"})," - Using Dashy to monitor uptime and status of your apps"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/searching",children:"Searching & Shortcuts"})," - Searching, launching methods + keyboard shortcuts"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/theming",children:"Theming"})," - Complete guide to applying, writing and modifying themes + styles"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/widgets",children:"Widgets"})," - List of all dynamic content widgets, with usage guides and examples"]}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"misc",children:"Misc"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"/docs/privacy",children:"Privacy & Security"})," - List of requests, potential issues, and security resources"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/LICENSE",children:"License"})," - Copy of the MIT License"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/CODE_OF_CONDUCT",children:"Code of Conduct"})," - Contributor Covenant Code of Conduct"]}),"\n"]})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(c,{...e})}):c(e)}},8453(e,n,s){s.d(n,{R:()=>r,x:()=>a});var i=s(6540);const t={},o=i.createContext(t);function r(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/625b1ed7.337a9a29.js b/assets/js/625b1ed7.4beb4e77.js similarity index 99% rename from assets/js/625b1ed7.337a9a29.js rename to assets/js/625b1ed7.4beb4e77.js index 68d691c5..3e3b7539 100644 --- a/assets/js/625b1ed7.337a9a29.js +++ b/assets/js/625b1ed7.4beb4e77.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1831],{5159(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"authentication","title":"Authentication","description":"- Basic Auth","source":"@site/docs/authentication.md","sourceDirName":".","slug":"/authentication","permalink":"/docs/authentication","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/authentication.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Status Indicators","permalink":"/docs/status-indicators"},"next":{"title":"Keyboard Shortcuts","permalink":"/docs/searching"}}');var t=s(4848),o=s(8453);const r={},a="Authentication",c={},l=[{value:"Built-In Auth",id:"built-in-auth",level:2},{value:"Setting Up Authentication",id:"setting-up-authentication",level:3},{value:"Hash Password",id:"hash-password",level:3},{value:"Logging In and Out",id:"logging-in-and-out",level:3},{value:"Enabling Guest Access",id:"enabling-guest-access",level:3},{value:"Granular Access",id:"granular-access",level:3},{value:"Permissions",id:"permissions",level:3},{value:"Using Environment Variables for Passwords",id:"using-environment-variables-for-passwords",level:3},{value:"Adding HTTP Auth to Configuration",id:"adding-http-auth-to-configuration",level:3},{value:"Security",id:"security",level:3},{value:"HTTP Auth",id:"http-auth",level:2},{value:"Using config-file users (recommended)",id:"using-config-file-users-recommended",level:3},{value:"Using static credentials",id:"using-static-credentials",level:3},{value:"Keycloak",id:"keycloak",level:2},{value:"1. Deploy Keycloak",id:"1-deploy-keycloak",level:3},{value:"2. Setup Keycloak Users",id:"2-setup-keycloak-users",level:3},{value:"3. Enable Keycloak in Dashy Config File",id:"3-enable-keycloak-in-dashy-config-file",level:3},{value:"4. Add groups and roles (Optional)",id:"4-add-groups-and-roles-optional",level:3},{value:"CORS Headers",id:"cors-headers",level:3},{value:"How server-side enforcement works",id:"how-server-side-enforcement-works",level:3},{value:"Troubleshooting Keycloak",id:"troubleshooting-keycloak",level:3},{value:"Header Authentication",id:"header-authentication",level:2},{value:"Configuration",id:"configuration",level:3},{value:"How it Works",id:"how-it-works",level:3},{value:"Notes",id:"notes",level:3},{value:"OIDC",id:"oidc",level:2},{value:"How server-side enforcement works",id:"how-server-side-enforcement-works-1",level:3},{value:"authentik",id:"authentik",level:2},{value:"1. Create an OIDC provider",id:"1-create-an-oidc-provider",level:4},{value:"2. Create an application",id:"2-create-an-application",level:4},{value:"3. (Optional) Limiting access via authentik with groups",id:"3-optional-limiting-access-via-authentik-with-groups",level:4},{value:"4. Configure Dashy to use OIDC client",id:"4-configure-dashy-to-use-oidc-client",level:4},{value:"5. (OPTIONAL) Example snippets for dashboard visibility",id:"5-optional-example-snippets-for-dashboard-visibility",level:4},{value:"Alternative Authentication Methods",id:"alternative-authentication-methods",level:2},{value:"Reverse proxy auth",id:"reverse-proxy-auth",level:3},{value:"Zero-trust tunnels",id:"zero-trust-tunnels",level:3},{value:"VPN",id:"vpn",level:3},{value:"IP-based access",id:"ip-based-access",level:3},{value:"Web server authentication",id:"web-server-authentication",level:3},{value:"SSO / OAuth providers",id:"sso--oauth-providers",level:3},{value:"Cloud hosting providers",id:"cloud-hosting-providers",level:3}];function d(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"authentication",children:"Authentication"})}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#built-in-auth",children:"Basic Auth"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#setting-up-authentication",children:"Setting Up Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#hash-password",children:"Hash Password"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#logging-in-and-out",children:"Logging In and Out"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#enabling-guest-access",children:"Guest Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#granular-access",children:"Granular Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#permissions",children:"Permissions"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-environment-variables-for-passwords",children:"Using Environment Variables for Passwords"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#security",children:"Security"})}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#http-auth",children:"HTTP Auth"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-config-file-users-recommended",children:"Using Config-File Users"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-static-credentials",children:"Using Static Credentials"})}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#authentik",children:"authentik"})}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#alternative-authentication-methods",children:"Alternative Authentication Methods"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#reverse-proxy-auth",children:"Reverse Proxy Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#zero-trust-tunnels",children:"Zero-Trust Tunnels"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#vpn",children:"VPN"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#ip-based-access",children:"IP-Based Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#web-server-authentication",children:"Web Server Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#sso--oauth-providers",children:"SSO / OAuth Providers"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#cloud-hosting-providers",children:"Cloud Hosting Providers"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.admonition,{type:"important",children:[(0,t.jsx)(n.p,{children:"Dashy's built-in auth is not intended to protect a publicly hosted instance against unauthorized access. Instead you should use an auth provider compatible with your reverse proxy, or access Dashy via your VPN, or implement your own SSO logic."}),(0,t.jsxs)(n.p,{children:["If Dashy is only accessible within your home network and you just want a login page, then the built-in auth may be sufficient. To also protect server-side endpoints and config files: with built-in auth set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," (",(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"details"}),"). (Or, consider setting up",(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC"}),", ",(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak"}),", or ",(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Auth"}),", where the server-side enforcement is on automatically)."]})]}),"\n",(0,t.jsx)(n.h2,{id:"built-in-auth",children:"Built-In Auth"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy has a basic login page included, and frontend authentication. You can enable this by adding users to the ",(0,t.jsx)(n.code,{children:"auth"})," section under ",(0,t.jsx)(n.code,{children:"appConfig"})," in your ",(0,t.jsx)(n.code,{children:"conf.yml"}),". If this section is not specified, then no authentication will be required to access the app, and the homepage will resolve to your dashboard. To also enable HTTP Authorization, set the ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"})," env var to ",(0,t.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"setting-up-authentication",children:"Setting Up Authentication"}),"\n",(0,t.jsxs)(n.p,{children:["The ",(0,t.jsx)(n.code,{children:"auth"})," property takes an array of users. Each user needs to include a username, hash and optional user type (",(0,t.jsx)(n.code,{children:"admin"})," or ",(0,t.jsx)(n.code,{children:"normal"}),"). The hash property is a ",(0,t.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/SHA-2",children:"SHA-256 Hash"})," of your desired password."]}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n auth:\n users:\n - user: alicia\n hash: 4D1E58C90B3B94BCAD9848ECCACD6D2A8C9FBC5CA913304BBA5CDEAB36FEEFA3\n type: admin\n - user: bob\n hash: 5E884898DA28047151D0E56F8DC6292773603D0D6AABBDD62A11EF721D1542D8\n"})}),"\n",(0,t.jsx)(n.h3,{id:"hash-password",children:"Hash Password"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy uses ",(0,t.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Sha-256",children:"SHA-256 Hash"}),", a 64-character string, which you can generate by running ",(0,t.jsx)(n.code,{children:'echo -n "my-super-secure-password" | sha256sum'}),", or using an online tool, such as ",(0,t.jsx)(n.a,{href:"https://passwordsgenerator.net/sha256-hash-generator/",children:"this one"})," or ",(0,t.jsx)(n.a,{href:"https://gchq.github.io/CyberChef/",children:"CyberChef"})," (which can be self-hosted/ ran locally)."]}),"\n",(0,t.jsx)(n.p,{children:"A hash is a one-way cryptographic function, meaning that it is easy to generate a hash for a given password, but very hard to determine the original password for a given hash. This means, that so long as your password is long, strong and unique, it is safe to store its hash in the clear. Having said that, you should never reuse passwords, hashes can be cracked by iterating over known password lists, generating a hash of each."}),"\n",(0,t.jsx)(n.h3,{id:"logging-in-and-out",children:"Logging In and Out"}),"\n",(0,t.jsx)(n.p,{children:"Once authentication is enabled, so long as there is no valid token in cookie storage, the application will redirect the user to the login page. When the user enters credentials in the login page, they will be checked, and if valid, then a token will be generated, and they can be redirected to the home page. If credentials are invalid, then an error message will be shown, and they will remain on the login page. Once in the application, to log out: the user can click the logout button (in the top-right), which will clear cookie storage, causing them to be redirected back to the login page."}),"\n",(0,t.jsx)(n.h3,{id:"enabling-guest-access",children:"Enabling Guest Access"}),"\n",(0,t.jsxs)(n.p,{children:["With authentication set up, by default no access is allowed to your dashboard without first logging in with valid credentials. Guest mode can be enabled to allow for read-only access to a secured dashboard by any user, without the need to log in. A guest user cannot write any changes to the config file, but can apply modifications locally (stored in their browser). You can enable guest access, by setting ",(0,t.jsx)(n.code,{children:"appConfig.auth.enableGuestAccess: true"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"granular-access",children:"Granular Access"}),"\n",(0,t.jsx)(n.p,{children:"You can use the following properties to make certain pages, sections or items only visible to some users, or hide pages, sections and items from guests."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"hideForUsers"})," - Page, Section or Item will be visible to all users, except for those specified in this list"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"showForUsers"})," - Page, Section or Item will be hidden from all users, except for those specified in this list"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"hideForGuests"})," - Page, Section or Item will be visible for logged in users, but not for guests"]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"For Example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n - name: Home Lab\n path: home-lab.yml\n displayData:\n showForUsers: [admin]\n - name: Intranet\n path: intranet.yml\n displayData:\n hideForGuests: true\n hideForUsers: [alicia, bob]\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Code Analysis & Monitoring\n icon: fas fa-code\n displayData:\n cols: 2\n hideForUsers: [alicia, bob]\n items:\n ...\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Deployment Pipelines\n icon: fas fa-rocket\n displayData:\n hideForGuests: true\n items:\n - title: Hide Me\n displayData:\n hideForUsers: [alicia, bob]\n"})}),"\n",(0,t.jsx)(n.h3,{id:"permissions",children:"Permissions"}),"\n",(0,t.jsxs)(n.p,{children:["Any user who is not an admin (with ",(0,t.jsx)(n.code,{children:"type: admin"}),") will not be able to write changes to disk."]}),"\n",(0,t.jsxs)(n.p,{children:["You can also prevent any user from writing changes to disk, using ",(0,t.jsx)(n.code,{children:"preventWriteToDisk"}),". Or prevent any changes from being saved locally in browser storage, using ",(0,t.jsx)(n.code,{children:"preventLocalSave"}),". Both properties can be found under ",(0,t.jsx)(n.a,{href:"/docs/configuring#appconfig-optional",children:(0,t.jsx)(n.code,{children:"appConfig"})}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["To disable all UI config features, including View Config, set ",(0,t.jsx)(n.code,{children:"disableConfiguration"}),". Alternatively you can disable UI config features for all non admin users by setting ",(0,t.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})," to true."]}),"\n",(0,t.jsx)(n.h3,{id:"using-environment-variables-for-passwords",children:"Using Environment Variables for Passwords"}),"\n",(0,t.jsxs)(n.p,{children:["If you don't want to hash your password, you can instead leave out the ",(0,t.jsx)(n.code,{children:"hash"})," attribute, and replace it with ",(0,t.jsx)(n.code,{children:"password"})," which should have the value of an environmental variable name you wish to use."]}),"\n",(0,t.jsxs)(n.p,{children:["Note that env var must begin with ",(0,t.jsx)(n.code,{children:"VITE_APP_"}),", and you must set this variable before building the app."]}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:" auth:\n users:\n - user: bob\n password: VITE_APP_BOB\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Just be sure to set ",(0,t.jsx)(n.code,{children:"VITE_APP_BOB='my super secret password'"})," before build-time."]}),"\n",(0,t.jsx)(n.h3,{id:"adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"}),"\n",(0,t.jsxs)(n.p,{children:["Without this, the built-in auth is just a client-side login page \u2014 your config and API endpoints can still be accessed directly. Set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," to protect them."]}),"\n",(0,t.jsx)(n.admonition,{type:"note",children:(0,t.jsxs)(n.p,{children:["HTTP Auth and guest access (",(0,t.jsx)(n.code,{children:"enableGuestAccess"}),") are incompatible. Guests have no credentials, so they can't fetch the config file when HTTP auth is active."]})}),"\n",(0,t.jsxs)(n.p,{children:["This uses the same users you've already defined in ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"})," to authenticate all server-side requests (config files, status checks, system info, CORS proxy, etc.) via HTTP Basic Auth."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"How it works:"})," When a user logs in through the Dashy UI, a session token is stored in a cookie. The frontend automatically includes this token in requests to local API endpoints. On the server side, the token is validated against your configured users. If someone tries to access an endpoint directly (e.g. with curl), the server will respond with a ",(0,t.jsx)(n.code,{children:"401"})," and a Basic Auth challenge \u2014 they'll need to provide a valid username and password."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:"Setup:"})}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Make sure you have users configured in your ",(0,t.jsx)(n.code,{children:"conf.yml"})," (see ",(0,t.jsx)(n.a,{href:"#setting-up-authentication",children:"Setting Up Authentication"})," above)"]}),"\n",(0,t.jsxs)(n.li,{children:["Set the ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," environment variable (e.g. in your ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," or ",(0,t.jsx)(n.code,{children:".env"})," file)"]}),"\n",(0,t.jsx)(n.li,{children:"Restart the container - the auth mode is determined at startup, so env var changes need a restart"}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["Adding or removing users in ",(0,t.jsx)(n.code,{children:"conf.yml"})," takes effect immediately without a restart, since the user list is read from disk on each request."]}),"\n",(0,t.jsxs)(n.p,{children:["For full protection, you'll want both the client-side login page (via ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"}),") and server-side auth (via ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"}),")."]}),"\n",(0,t.jsx)(n.h3,{id:"security",children:"Security"}),"\n",(0,t.jsx)(n.p,{children:"With basic auth (and without HTTP auth), the login logic runs on the client-side. A technical user could inspect the code and view parts of your configuration, including password hashes. If the SHA-256 hash is of a common password, it may be possible to determine it using a lookup table, and then use that to generate a valid auth token. Therefore, you should always use a long, strong and unique password."}),"\n",(0,t.jsxs)(n.p,{children:["If your instance is exposed to the internet, the built-in auth alone is not sufficient - use a reverse proxy with its own authentication layer (see ",(0,t.jsx)(n.a,{href:"#alternative-authentication-methods",children:"Alternative Authentication Methods"}),"), or access Dashy over a VPN. See the ",(0,t.jsx)(n.a,{href:"/docs/management#network-exposure",children:"Network Exposure"})," section in the management docs for more on this."]}),"\n",(0,t.jsx)(n.p,{children:"The built-in login page prevents casual unauthorized access on a private network. It's not a security perimeter."}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"http-auth",children:"HTTP Auth"}),"\n",(0,t.jsx)(n.p,{children:"If you'd like to protect server-side endpoints with HTTP Basic Auth, there are two approaches. They protect the same endpoints but use different credential sources, so pick one - don't combine them."}),"\n",(0,t.jsx)(n.h3,{id:"using-config-file-users-recommended",children:"Using config-file users (recommended)"}),"\n",(0,t.jsxs)(n.p,{children:["This is the approach described in ",(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"})," above. Set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," and it uses the same ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"})," from your ",(0,t.jsx)(n.code,{children:"conf.yml"}),". The frontend handles authentication automatically using the session token from the login page, so no extra setup is needed."]}),"\n",(0,t.jsx)(n.p,{children:"This is the recommended approach because it keeps credentials in one place and works together with the client-side login page. But the drawback is that your credentials will be stored in your config file."}),"\n",(0,t.jsx)(n.h3,{id:"using-static-credentials",children:"Using static credentials"}),"\n",(0,t.jsxs)(n.p,{children:["If you don't have users in your ",(0,t.jsx)(n.code,{children:"conf.yml"})," (e.g. you handle user management externally, or just want a single shared password for server-side access), you can set the ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"})," and ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," environmental variables instead."]}),"\n",(0,t.jsxs)(n.p,{children:["With this approach, there is no Dashy login page. When the browser first requests the config file, the server responds with a ",(0,t.jsx)(n.code,{children:"401"})," and the browser shows its native HTTP auth prompt. Once the user enters the correct credentials, the browser caches them for the session and all subsequent requests work."]}),"\n",(0,t.jsxs)(n.p,{children:["To skip the browser prompt and have the frontend authenticate automatically, also set ",(0,t.jsx)(n.code,{children:"VITE_APP_BASIC_AUTH_USERNAME"})," and ",(0,t.jsx)(n.code,{children:"VITE_APP_BASIC_AUTH_PASSWORD"})," to the same values. These are baked in at build time, so a rebuild is required, and you should only do this on a trusted network."]}),"\n",(0,t.jsx)(n.admonition,{type:"warning",children:(0,t.jsxs)(n.p,{children:["Do not combine ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"}),"/",(0,t.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," with conf.yml users. If both are present, the server will log a warning at startup. With ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"})," set, config-file users take priority and the static credentials are ignored. Without it, the static credentials protect the server but the Dashy login page will use conf.yml credentials, and the frontend will send the wrong credentials to server endpoints. Pick one approach or the other."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"keycloak",children:"Keycloak"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy also supports using a ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/",children:"Keycloak"})," authentication server. The setup for this is a bit more involved, but it gives you greater security overall, useful for if your instance is exposed to the internet."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://www.keycloak.org/about.html",children:"Keycloak"})," is a Java-based ",(0,t.jsx)(n.a,{href:"https://github.com/keycloak/keycloak",children:"open source"}),", high-performance, secure authentication system, supported by ",(0,t.jsx)(n.a,{href:"https://www.redhat.com/en",children:"RedHat"}),". It is easy to setup (",(0,t.jsx)(n.a,{href:"https://quay.io/repository/keycloak/keycloak",children:"with Docker"}),"), and enables you to secure multiple self-hosted applications with single-sign-on using standard protocols (OpenID Connect, OAuth 2.0, SAML 2.0 and social login). It's also very customizable, you can write or use custom ",(0,t.jsx)(n.a,{href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_development/topics/themes.html",children:"themes"}),", ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/extensions.html",children:"plugins"}),", ",(0,t.jsx)(n.a,{href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_admin/topics/authentication/password-policies.html",children:"password policies"})," and more.\nThe following guide will walk you through setting up Keycloak with Dashy. If you already have a Keycloak instance configured, then skip to Step 3."]}),"\n",(0,t.jsx)(n.h3,{id:"1-deploy-keycloak",children:"1. Deploy Keycloak"}),"\n",(0,t.jsxs)(n.p,{children:["First thing to do is to spin up a new instance of Keycloak. You will need ",(0,t.jsx)(n.a,{href:"https://docs.docker.com/engine/install/",children:"Docker installed"}),", and can then choose a tag, and pull the container from ",(0,t.jsx)(n.a,{href:"https://quay.io/repository/keycloak/keycloak",children:"quay.io/keycloak/keycloak"})]}),"\n",(0,t.jsx)(n.p,{children:"Use the following run command, replacing the attributes (default credentials, port and name), or incorporate this into your docker-compose file."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8081:8080 \\\n --name auth-server \\\n -e KEYCLOAK_ADMIN=admin \\\n -e KEYCLOAK_ADMIN_PASSWORD=admin \\\n quay.io/keycloak/keycloak:25.0 start-dev\n"})}),"\n",(0,t.jsxs)(n.p,{children:["(The ",(0,t.jsx)(n.code,{children:"KEYCLOAK_USER"})," / ",(0,t.jsx)(n.code,{children:"KEYCLOAK_PASSWORD"})," env vars and the ",(0,t.jsx)(n.code,{children:"/auth"})," URL prefix from Keycloak 16 and older have been replaced. If you are still on 17 or older, set ",(0,t.jsx)(n.code,{children:"legacySupport: true"})," in your Dashy config later on.)"]}),"\n",(0,t.jsxs)(n.p,{children:["If you need to pull from DockerHub, a non-official image is available ",(0,t.jsx)(n.a,{href:"https://registry.hub.docker.com/r/jboss/keycloak",children:"here"}),". Or if you would prefer not to use Docker, you can also directly install Keycloak from source, following ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/docs/latest/getting_started/index.html",children:"this guide"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["You should now be able to access the Keycloak web interface, using the port specified above (e.g. ",(0,t.jsx)(n.code,{children:"http://127.0.0.1:8081"}),"), login with the default credentials, and when prompted create a new password."]}),"\n",(0,t.jsx)(n.h3,{id:"2-setup-keycloak-users",children:"2. Setup Keycloak Users"}),"\n",(0,t.jsx)(n.p,{children:"Before we can use Keycloak, we must first set it up with some users. Keycloak uses Realms (similar to tenants) to create isolated groups of users. You must create a Realm before you will be able to add your first user."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Head over to the admin console"}),"\n",(0,t.jsx)(n.li,{children:"In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm'"}),"\n",(0,t.jsx)(n.li,{children:"Give your realm a name, and hit 'Create'"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"You can now create your first user."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"In the left-hand menu, click 'Users', then 'Add User'"}),"\n",(0,t.jsxs)(n.li,{children:["Fill in the form. On Keycloak 25 and newer, ",(0,t.jsx)(n.em,{children:"First name"})," and ",(0,t.jsx)(n.em,{children:"Last name"}),' are required by the default user-profile schema. If you skip them the user can sign in but login will then fail with "Account is not fully set up"']}),"\n",(0,t.jsx)(n.li,{children:"Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"Next, create a new client for Dashy."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right"}),"\n",(0,t.jsxs)(n.li,{children:["Choose a 'Client ID' (e.g. ",(0,t.jsx)(n.code,{children:"dashy"}),"), set 'Client Protocol' to 'openid-connect'"]}),"\n",(0,t.jsxs)(n.li,{children:["Turn ",(0,t.jsx)(n.em,{children:"Client authentication"})," OFF and leave ",(0,t.jsx)(n.em,{children:"Standard flow"})," enabled. Dashy is a SPA, so it acts as an OAuth public client with PKCE. A confidential client requires a client_secret that a browser app can't safely hold"]}),"\n",(0,t.jsxs)(n.li,{children:["For 'Valid Redirect URIs' put the URL where you host Dashy (e.g. ",(0,t.jsx)(n.code,{children:"https://dashy.example.com/*"}),", or just ",(0,t.jsx)(n.code,{children:"*"})," while testing locally). Do the same for the 'Web Origins' field"]}),"\n",(0,t.jsx)(n.li,{children:"Make note of your client-id, and click 'Save'"}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["For the ",(0,t.jsx)(n.code,{children:"adminRole"})," check to work, the role must appear in the id_token (Keycloak's default mapper only adds it to the access token):"]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Open your ",(0,t.jsx)(n.code,{children:"dashy"})," client, go to the ",(0,t.jsx)(n.em,{children:"Client scopes"})," tab, click the dedicated scope row (",(0,t.jsx)(n.code,{children:"dashy-dedicated"}),")"]}),"\n",(0,t.jsxs)(n.li,{children:["Add a new mapper of type ",(0,t.jsx)(n.em,{children:"User Realm Role"}),", name it (e.g. ",(0,t.jsx)(n.code,{children:"realm_roles"}),"), claim name ",(0,t.jsx)(n.code,{children:"realm_access.roles"}),", multivalued ON, ",(0,t.jsx)(n.em,{children:"Add to ID token"})," ON, ",(0,t.jsx)(n.em,{children:"Add to access token"})," ON"]}),"\n",(0,t.jsxs)(n.li,{children:["(Optional, for ",(0,t.jsx)(n.code,{children:"adminGroup"})," instead of ",(0,t.jsx)(n.code,{children:"adminRole"}),") Add a second mapper of type ",(0,t.jsx)(n.em,{children:"Group Membership"}),", claim name ",(0,t.jsx)(n.code,{children:"groups"})]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"To create the admin role itself and grant it to a user:"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.em,{children:"Realm roles"})," in the left-hand menu, ",(0,t.jsx)(n.em,{children:"Create role"}),", name it (e.g. ",(0,t.jsx)(n.code,{children:"dashy-admin"}),")"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.em,{children:"Users"})," \u2192 pick your admin user \u2192 ",(0,t.jsx)(n.em,{children:"Role mapping"})," \u2192 ",(0,t.jsx)(n.em,{children:"Assign role"})," \u2192 select ",(0,t.jsx)(n.code,{children:"dashy-admin"})]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"3-enable-keycloak-in-dashy-config-file",children:"3. Enable Keycloak in Dashy Config File"}),"\n",(0,t.jsxs)(n.p,{children:["Now that your Keycloak instance is up and running, all that's left to do is to configure Dashy to use it. Under ",(0,t.jsx)(n.code,{children:"appConfig"}),", set ",(0,t.jsx)(n.code,{children:"auth.enableKeycloak: true"}),", then fill in the details in ",(0,t.jsx)(n.code,{children:"auth.keycloak"}),", including: ",(0,t.jsx)(n.code,{children:"serverUrl"})," - the URL where your Keycloak instance is hosted, ",(0,t.jsx)(n.code,{children:"realm"})," - the name you gave your Realm, and ",(0,t.jsx)(n.code,{children:"clientId"})," - the Client ID you chose.\nFor example:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n # ...\n disableConfigurationForNonAdmin: true\n auth:\n enableKeycloak: true\n keycloak:\n serverUrl: 'http://localhost:8081'\n realm: 'alicia-homelab'\n clientId: 'dashy'\n adminRole: 'dashy-admin' # role name that grants admin privileges\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Note that if you are using Keycloak V 17 or older, you will also need to set ",(0,t.jsx)(n.code,{children:"legacySupport: true"})," (also under ",(0,t.jsx)(n.code,{children:"appConfig.auth.keycloak"}),"). This is because the API endpoint was updated in later versions."]}),"\n",(0,t.jsxs)(n.p,{children:["If you use Keycloak with an external Identity Provier, you can set the ",(0,t.jsx)(n.code,{children:"idpHint: 'alias-of-kc-idp'"})," option to allow the IdP Hint to be passed to Keycloak. This will cause Keycloak to skip its login page and redirect the user directly to the specified IdP's login page. Set to the value of the 'Alias' field of the desired IdP as defined in Keycloak under 'Identity Providers'."]}),"\n",(0,t.jsx)(n.h3,{id:"4-add-groups-and-roles-optional",children:"4. Add groups and roles (Optional)"}),"\n",(0,t.jsxs)(n.p,{children:["Keycloak allows you to assign users roles and groups. You can use these values to configure who can access various sections or items in Dashy.\nKeycloak server administration and configuration is a deep topic; please refer to the ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/docs/latest/server_admin/index.html#assigning-permissions-and-access-using-roles-and-groups",children:"server admin guide"})," to see details about creating and assigning roles and groups.\nOnce you have groups or roles assigned to users you can configure access under each section or item ",(0,t.jsx)(n.code,{children:"displayData.showForKeycloakUser"})," and ",(0,t.jsx)(n.code,{children:"displayData.hideForKeycloakUser"}),".\nBoth show and hide configurations accept a list of ",(0,t.jsx)(n.code,{children:"groups"})," and ",(0,t.jsx)(n.code,{children:"roles"})," that limit access. If a users data matches one or more items in these lists they will be allowed or excluded as defined."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"sections:\n - name: DeveloperResources\n displayData:\n showForKeycloakUsers:\n roles: ['canViewDevResources']\n hideForKeycloakUsers:\n groups: ['ProductTeam']\n items:\n - title: Not Visible for developers\n displayData:\n hideForKeycloakUsers:\n groups: ['DevelopmentTeam']\n"})}),"\n",(0,t.jsx)(n.p,{children:"Your app is now secured :) When you load Dashy, it will redirect to your Keycloak login page, and any user without valid credentials will be prevented from accessing your dashboard."}),"\n",(0,t.jsxs)(n.p,{children:["From within the Keycloak console, you can then configure things like time-outs, password policies, etc. You can also backup your full Keycloak config, and it is recommended to do this, along with your Dashy config. You can spin up both Dashy and Keycloak simultaneously and restore both applications configs using a ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," file, and this is recommended."]}),"\n",(0,t.jsx)(n.h3,{id:"cors-headers",children:"CORS Headers"}),"\n",(0,t.jsxs)(n.p,{children:["If Dashy and Keycloak run on different origins (typical when testing locally on different ",(0,t.jsx)(n.code,{children:"localhost:"})," ports), Keycloak's default ",(0,t.jsx)(n.code,{children:"Content-Security-Policy: frame-ancestors 'self'"})," and ",(0,t.jsx)(n.code,{children:"X-Frame-Options: SAMEORIGIN"})," block the hidden iframe ",(0,t.jsx)(n.code,{children:"keycloak-js"}),' uses to check your session. Symptom: a generic "Authentication failed (Keycloak)" toast on first load. To allow the iframe, open ',(0,t.jsx)(n.em,{children:"Realm settings \u2192 Security defenses \u2192 Browser headers"}),", clear ",(0,t.jsx)(n.code,{children:"X-Frame-Options"}),", and change ",(0,t.jsx)(n.code,{children:"Content-Security-Policy"})," to ",(0,t.jsx)(n.code,{children:"frame-src 'self' ; frame-ancestors 'self' ; object-src 'none';"}),". Same-origin production deployments don't hit this."]}),"\n",(0,t.jsx)(n.h3,{id:"how-server-side-enforcement-works",children:"How server-side enforcement works"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy's server reads ",(0,t.jsx)(n.code,{children:"auth.keycloak"})," from ",(0,t.jsx)(n.code,{children:"conf.yml"})," at boot, lazily fetches your Keycloak realm's OIDC discovery doc + JWKS, then verifies the ",(0,t.jsx)(n.code,{children:"id_token"})," the SPA attaches to every API call as ",(0,t.jsx)(n.code,{children:"Authorization: Bearer "}),". Tokens that fail signature / issuer / audience / expiry verification are rejected with ",(0,t.jsx)(n.code,{children:"401"}),". Write endpoints (",(0,t.jsx)(n.code,{children:"POST /config-manager/save"}),") additionally require the ",(0,t.jsx)(n.code,{children:"adminRole"})," (or ",(0,t.jsx)(n.code,{children:"adminGroup"}),") to be present in the token claims, and non-admins receive ",(0,t.jsx)(n.code,{children:"403"}),". Note that the ",(0,t.jsx)(n.code,{children:"/conf.yml"})," remains anonymously readable still (so the SPA can bootstrap before login)."]}),"\n",(0,t.jsxs)(n.p,{children:["The admin check reads the role / group claim from the id_token, so the client mapper from Step 2 above (the one with ",(0,t.jsx)(n.em,{children:"Add to ID token"})," on) is what makes ",(0,t.jsx)(n.code,{children:"adminRole"})," / ",(0,t.jsx)(n.code,{children:"adminGroup"})," work. Without it the server gets a token with no roles claim and treats everyone as non-admin."]}),"\n",(0,t.jsx)(n.h3,{id:"troubleshooting-keycloak",children:"Troubleshooting Keycloak"}),"\n",(0,t.jsx)(n.p,{children:"If you encounter issues with your Keycloak setup, follow these steps to troubleshoot and resolve common problems."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsx)(n.p,{children:'Client Authentication Issue\nProblem: Redirect loop, if client authentication is enabled.\nSolution: Switch off "client authentication" in "TC clients" -> "Advanced" settings.'}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsxs)(n.p,{children:['Double URL\nProblem: If you get redirected to "',(0,t.jsx)(n.a,{href:"https://dashy.my.domain/#iss=https://keycloak.my.domain/realms/my-realm",children:"https://dashy.my.domain/#iss=https://keycloak.my.domain/realms/my-realm"}),'"\nSolution: Make sure to turn on "Exclude Issuer From Authentication Response" in "TC clients" -> "Advanced" -> "OpenID Connect Compatibility Modes"']}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsxs)(n.p,{children:['Problems with mutiple Dashy Pages\nProblem: Refreshing or logging out of dashy results in an "invalid_redirect_uri" error.\nSolution: In "TC clients" -> "Access settings" -> "Root URL" ',(0,t.jsx)(n.a,{href:"https://dashy.my.domain/",children:"https://dashy.my.domain/"}),", valid redirect URIs must be /*"]}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"header-authentication",children:"Header Authentication"}),"\n",(0,t.jsxs)(n.p,{children:["Header authentication allows Dashy to trust an upstream reverse proxy to handle authentication. The proxy authenticates users and forwards their identity to Dashy via a configurable HTTP header (e.g. ",(0,t.jsx)(n.code,{children:"REMOTE_USER"}),"). This is the standard pattern used by ",(0,t.jsx)(n.a,{href:"https://www.authelia.com/",children:"Authelia"}),", ",(0,t.jsx)(n.a,{href:"https://goauthentik.io/",children:"Authentik"}),", Traefik's ",(0,t.jsx)(n.code,{children:"forwardAuth"}),", Caddy's ",(0,t.jsx)(n.code,{children:"forward_auth"}),", and Nginx's ",(0,t.jsx)(n.code,{children:"auth_request"}),"."]}),"\n",(0,t.jsx)(n.p,{children:"This is useful when you already have a central authentication layer in front of your self-hosted services and want Dashy to automatically pick up the authenticated user without requiring a separate login."}),"\n",(0,t.jsx)(n.h3,{id:"configuration",children:"Configuration"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n auth:\n enableHeaderAuth: true\n users:\n - user: alice\n hash: 0a7b1d4c2e... # SHA-256 hash of password\n type: admin\n - user: bob\n hash: 3f8e2b1a9d...\n type: normal\n headerAuth:\n userHeader: Remote-User\n proxyWhitelist:\n - 172.18.0.2\n - 127.0.0.1\n"})}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"userHeader"})})," - The HTTP header name containing the authenticated username. Defaults to ",(0,t.jsx)(n.code,{children:"Remote-User"})," if not specified. Common values: ",(0,t.jsx)(n.code,{children:"Remote-User"})," (Authelia), ",(0,t.jsx)(n.code,{children:"X-authentik-username"})," (Authentik), or whatever your proxy forwards. Header matching is case-insensitive."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"proxyWhitelist"})})," - Required. An array of IP addresses that Dashy will accept the header from. Only requests originating from these IPs will be trusted. This prevents clients from spoofing the header directly."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"users"})})," - Required. The header username is matched against this list to determine the user's role (",(0,t.jsx)(n.code,{children:"admin"})," or ",(0,t.jsx)(n.code,{children:"normal"}),") and to generate the session token. Users must be defined here even though authentication is handled externally."]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"how-it-works",children:"How it Works"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"User visits Dashy, which is behind a reverse proxy (e.g. Authelia)"}),"\n",(0,t.jsxs)(n.li,{children:["The proxy authenticates the user and forwards the request with a header like ",(0,t.jsx)(n.code,{children:"Remote-User: alice"})]}),"\n",(0,t.jsxs)(n.li,{children:["Dashy's server checks that the request comes from a whitelisted proxy IP, then returns the username via the ",(0,t.jsx)(n.code,{children:"/get-user"})," endpoint"]}),"\n",(0,t.jsx)(n.li,{children:"The client matches the username against the configured users, generates a session token, and sets the auth cookie"}),"\n",(0,t.jsxs)(n.li,{children:["From this point, standard Dashy auth applies - ",(0,t.jsx)(n.code,{children:"isLoggedIn()"}),", admin checks, and granular access controls all work as normal"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"notes",children:"Notes"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["The ",(0,t.jsx)(n.code,{children:"proxyWhitelist"})," checks ",(0,t.jsx)(n.code,{children:"req.socket.remoteAddress"}),", which is the direct connection source. If your proxy connects through Docker networking, use the container's internal IP (e.g. ",(0,t.jsx)(n.code,{children:"172.18.0.2"}),"), not the external IP"]}),"\n",(0,t.jsx)(n.li,{children:"Logout clears Dashy's session cookie, but the user remains authenticated at the proxy level. Revisiting the page will re-authenticate automatically"}),"\n",(0,t.jsxs)(n.li,{children:["When header auth is enabled, server-side API endpoints are also protected by the proxy whitelist. Requests not from a whitelisted IP will be rejected. Admin enforcement applies - only users with ",(0,t.jsx)(n.code,{children:"type: admin"})," can access write endpoints (config save)"]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"oidc",children:"OIDC"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy also supports using a general ",(0,t.jsx)(n.a,{href:"https://openid.net/connect/",children:"OIDC compatible"})," authentication server. In order to use it, the authentication section needs to be configured:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n disableConfigurationForNonAdmin: true # Prevent authenticated non-admins using editor\n auth:\n enableOidc: true\n oidc:\n clientId: 'registered-client-id'\n endpoint: 'https://your-oidc-provider.example.com'\n scope: 'openid profile email'\n adminGroup: admin\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Because Dashy is a SPA, a ",(0,t.jsx)(n.a,{href:"https://datatracker.ietf.org/doc/html/rfc6749#section-2.1",children:"public client"})," registration with PKCE is needed."]}),"\n",(0,t.jsxs)(n.p,{children:["If you set ",(0,t.jsx)(n.code,{children:"adminGroup"}),", include ",(0,t.jsx)(n.code,{children:"groups"})," in ",(0,t.jsx)(n.code,{children:"scope"})," (e.g. ",(0,t.jsx)(n.code,{children:"scope: 'openid profile email groups'"}),") so your IdP actually returns the claim in the id_token. Same goes for ",(0,t.jsx)(n.code,{children:"adminRole"})," and a ",(0,t.jsx)(n.code,{children:"roles"})," scope if your IdP needs one."]}),"\n",(0,t.jsxs)(n.p,{children:["Note, that if your ",(0,t.jsx)(n.code,{children:"clientId"})," is numeric, you must place it in quotes. Otherwise it will be interpreted as a number and truncated to 64 chars!"]}),"\n",(0,t.jsx)(n.p,{children:"An example for Authelia is shared below, but other OIDC systems can be used:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"identity_providers:\n oidc:\n clients:\n - client_id: dashy\n client_name: dashy\n public: true\n authorization_policy: 'one_factor'\n require_pkce: true\n pkce_challenge_method: 'S256'\n redirect_uris:\n - https://dashy.local # should point to your dashy endpoint\n grant_types:\n - authorization_code\n scopes:\n - 'openid'\n - 'profile'\n - 'roles'\n - 'email'\n - 'groups'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Groups and roles will be populated and available for controlling display similar to ",(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak"})," above."]}),"\n",(0,t.jsx)(n.h3,{id:"how-server-side-enforcement-works-1",children:"How server-side enforcement works"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy's server reads ",(0,t.jsx)(n.code,{children:"auth.oidc"})," from ",(0,t.jsx)(n.code,{children:"conf.yml"})," at boot, lazily fetches the OIDC discovery doc + JWKS from your ",(0,t.jsx)(n.code,{children:"endpoint"}),", then verifies the ",(0,t.jsx)(n.code,{children:"id_token"})," the SPA attaches to every API call as ",(0,t.jsx)(n.code,{children:"Authorization: Bearer "}),". Tokens that fail signature / issuer / audience / expiry verification are rejected with ",(0,t.jsx)(n.code,{children:"401"}),". Write endpoints (",(0,t.jsx)(n.code,{children:"POST /config-manager/save"}),") additionally require the ",(0,t.jsx)(n.code,{children:"adminGroup"})," (or ",(0,t.jsx)(n.code,{children:"adminRole"}),") to be present in the token's ",(0,t.jsx)(n.code,{children:"groups"})," / ",(0,t.jsx)(n.code,{children:"roles"})," claims, and non-admins receive ",(0,t.jsx)(n.code,{children:"403"}),". Note that the ",(0,t.jsx)(n.code,{children:"/conf.yml"})," remains anonymously readable still"]}),"\n",(0,t.jsxs)(n.p,{children:["Your IdP must include ",(0,t.jsx)(n.code,{children:"groups"})," / ",(0,t.jsx)(n.code,{children:"roles"})," in the id_token, not only the access token, for the admin check to work (most IdPs do this when the ",(0,t.jsx)(n.code,{children:"groups"})," scope is requested)."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"authentik",children:"authentik"}),"\n",(0,t.jsxs)(n.p,{children:["This documentation is specific to ",(0,t.jsx)(n.code,{children:"authentik"}),", however it may be useful in getting other idP's working with ",(0,t.jsx)(n.code,{children:"Dashy"}),"."]}),"\n",(0,t.jsx)(n.p,{children:"This guide will only walk through the following:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Creating and configuring an OIDC provider"}),"\n",(0,t.jsx)(n.li,{children:"Creating and configuring an application"}),"\n",(0,t.jsx)(n.li,{children:"Assigning groups"}),"\n",(0,t.jsxs)(n.li,{children:["Configuring ",(0,t.jsx)(n.code,{children:"Dashy"})," to use the OIDC client"]}),"\n",(0,t.jsxs)(n.li,{children:["Show quick examples of how to hide/show ",(0,t.jsx)(n.code,{children:"pages"}),", ",(0,t.jsx)(n.code,{children:"items"}),", and ",(0,t.jsx)(n.code,{children:"sections"})," using OIDC groups"]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"This guide assumes the following:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["You have a working instance of ",(0,t.jsx)(n.code,{children:"authentik"})," terminated with SSL"]}),"\n",(0,t.jsxs)(n.li,{children:["You have a working instance of ",(0,t.jsx)(n.code,{children:"Dashy"})," terminated with SSL"]}),"\n",(0,t.jsx)(n.li,{children:"Users and groups are provisioned"}),"\n",(0,t.jsxs)(n.li,{children:["You are familiar with how ",(0,t.jsx)(n.code,{children:"authentik"})," works in case you need to do further troubleshooting that is outside the scope of this guide."]}),"\n"]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["It it recommended that you create groups specific for ",(0,t.jsx)(n.code,{children:"Dashy"}),". Groups will allow you to display content based on group membership as well as limiting user access to ",(0,t.jsx)(n.code,{children:"Dashy"}),". If you do not need this functionality, then you can forgo creating specific groups."]})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsx)(n.p,{children:"You can use the application wizard to create the provider and application at one time. This is the recommended route, but only the manual process will be outlined in this guide."})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/72e45162-6c86-4d6f-a1ae-724ac503c00c",alt:"image"})}),"\n",(0,t.jsx)(n.h4,{id:"1-create-an-oidc-provider",children:"1. Create an OIDC provider"}),"\n",(0,t.jsxs)(n.p,{children:["Login to the admin console for ",(0,t.jsx)(n.code,{children:"authentik"}),". Go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Providers"}),". Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/c1f7f45d-469c-4bf1-a825-34658341a83e",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["A dialog box will pop-up, select the ",(0,t.jsx)(n.code,{children:"OAuth2/OpenID Provider"}),". Click ",(0,t.jsx)(n.code,{children:"Next"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ea84fe57-b813-404d-8dad-5e221b440bdb",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["On the next page of the wizard, set the ",(0,t.jsx)(n.code,{children:"Name"}),", ",(0,t.jsx)(n.code,{children:"Authentication flow"}),", ",(0,t.jsx)(n.code,{children:"Authorization flow"}),", and ",(0,t.jsx)(n.code,{children:"Invalidation flow"}),". See example below. Using the ",(0,t.jsx)(n.code,{children:"default-provider-authorization-implicit-consent"})," authorization flow on internal services and ",(0,t.jsx)(n.code,{children:"default-provider-authorization-explicit-consent"})," on external services is a common practice. However, it is fully up to you on how you would like to configure this option. ",(0,t.jsx)(n.code,{children:"Implicit"})," will login directly without user consent, ",(0,t.jsx)(n.code,{children:"explicit"})," will ask if the user approves the service being logged into with their user credentials. For the invalidation flow (required on Authentik 2023.10 and later) the built-in ",(0,t.jsx)(n.code,{children:"default-provider-invalidation-flow"})," is fine."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/e600aeaf-08d1-49aa-b304-11e90e5c89cd",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Scroll down and configure the ",(0,t.jsx)(n.code,{children:"Protocol settings"}),". Set the ",(0,t.jsx)(n.code,{children:"Client type"})," to ",(0,t.jsx)(n.code,{children:"Public"}),". Add the ",(0,t.jsx)(n.code,{children:"Redirect URIs/Origins (RegEx)"}),". If the site is hosted at ",(0,t.jsx)(n.code,{children:"dashy.lan.domain.com"}),", then you would enter as the example below."]}),"\n",(0,t.jsx)(n.admonition,{type:"note",children:(0,t.jsxs)(n.p,{children:["If you have an internal and external domain for ",(0,t.jsx)(n.code,{children:"Dashy"}),", enter both URI's. Enter each URI on a new line."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/4a289d7e-d7b4-4ff6-af5d-3e5202fae84e",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Scroll down to set the ",(0,t.jsx)(n.code,{children:"Signing Key"}),". It is recommended to use the built in ",(0,t.jsx)(n.code,{children:"authentik Self-signed Certificate"})," here unless you have special needs for your own custom cert."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/386c0750-9d2b-4482-8938-8b301b489b38",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["If you plan to use ",(0,t.jsx)(n.code,{children:"adminGroup"})," in your Dashy config, you need a ",(0,t.jsx)(n.code,{children:"groups"})," scope mapping first. Authentik does not ship one by default. Open ",(0,t.jsx)(n.em,{children:"Customisation > Property Mappings"})," in a new tab, click ",(0,t.jsx)(n.em,{children:"Create > Scope Mapping"}),", set ",(0,t.jsx)(n.em,{children:"Name"})," to ",(0,t.jsx)(n.code,{children:"groups"}),", ",(0,t.jsx)(n.em,{children:"Scope name"})," to ",(0,t.jsx)(n.code,{children:"groups"}),", and ",(0,t.jsx)(n.em,{children:"Expression"})," to:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-python",children:'return {"groups": [g.name for g in request.user.ak_groups.all()]}\n'})}),"\n",(0,t.jsx)(n.p,{children:"Save it, then come back to the provider wizard."}),"\n",(0,t.jsxs)(n.p,{children:["Expand ",(0,t.jsx)(n.code,{children:"Advanced protocol settings"})," then verify the ",(0,t.jsx)(n.code,{children:"Scopes"})," are set to what is highlighted in ",(0,t.jsx)(n.code,{children:"white"})," below (including the ",(0,t.jsx)(n.code,{children:"groups"})," mapping you just created, if you want ",(0,t.jsx)(n.code,{children:"adminGroup"})," to work). Set the ",(0,t.jsx)(n.code,{children:"Subject mode"})," to ",(0,t.jsx)(n.code,{children:"Based on the Users's Email"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ae5e87b8-1ad6-41dd-b6e1-9665623f842a",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Lastly, toggle ",(0,t.jsx)(n.code,{children:"Include claims in id_token"})," to on. Click ",(0,t.jsx)(n.code,{children:"Finish"})," to complete creating the provider."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/25353b3c-3f54-47cf-bd47-b5023f86d7cf",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Grab the generated ",(0,t.jsx)(n.code,{children:"Client ID"})," and ",(0,t.jsx)(n.code,{children:"OpenID Configuration Issuer"})," URL by clicking the newly created provider as this will use this later when ",(0,t.jsx)(n.code,{children:"Dashy"})," is configured to use the OIDC auth mechanism. In this tutorial, what was generated is used below. Obviously adjust the ",(0,t.jsx)(n.code,{children:"Client ID"})," that was generated and use your domain here for the ",(0,t.jsx)(n.code,{children:"issuer"}),"."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"Client ID: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15\nOpenID Configuration Issuer: https://auth.domain.com/application/o/dashy/\n"})}),"\n",(0,t.jsx)(n.h4,{id:"2-create-an-application",children:"2. Create an application"}),"\n",(0,t.jsxs)(n.p,{children:["Make sure you are still in the ",(0,t.jsx)(n.code,{children:"authentik"})," admin console then go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Applications"}),". Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/fd225936-15a1-409f-83c8-e24a43047df0",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Next, it is required to give a user facing ",(0,t.jsx)(n.code,{children:"Name"}),", ",(0,t.jsx)(n.code,{children:"Slug"})," and assign the newly created provider. Use the example below if you have been following the guide. If you have used your own naming, then adjust accordingly. Click ",(0,t.jsx)(n.code,{children:"Create"})," once you are done."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/e6574d7d-6b22-4e7d-b388-45341b98746b",alt:"image"})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["Open the application in a new tab from the ",(0,t.jsx)(n.code,{children:"authentik"})," user portal and upload a custom icon. You can also enter a user facing ",(0,t.jsx)(n.code,{children:"Description"})," that the user would see."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/20561387-549f-49de-98e6-30330dcdc734",alt:"image"})}),"\n",(0,t.jsxs)(n.h4,{id:"3-optional-limiting-access-via-authentik-with-groups",children:["3. ",(0,t.jsx)(n.em,{children:"(Optional)"})," Limiting access via ",(0,t.jsx)(n.code,{children:"authentik"})," with groups"]}),"\n",(0,t.jsxs)(n.p,{children:["If you would like to deny ",(0,t.jsx)(n.code,{children:"Dashy"})," access from specific users who are not within ",(0,t.jsx)(n.code,{children:"authentik"})," based groups, you bind them to the application you just created now. ",(0,t.jsx)(n.code,{children:"authentik"})," will deny access to those who are not members of this group or groups. If you want to allow everyone access from your ",(0,t.jsx)(n.code,{children:"authentik"})," instance, skip this step."]}),"\n",(0,t.jsxs)(n.p,{children:["Make sure you are still in the ",(0,t.jsx)(n.code,{children:"authentik"})," admin console then go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Applications"}),". Click the newly created ",(0,t.jsx)(n.code,{children:"Dashy"})," application."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/613fafe7-881f-4664-a903-945854ac65e2",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Click the ",(0,t.jsx)(n.code,{children:"Policy/Group/User Bindings"})," tab at the top, then click ",(0,t.jsx)(n.code,{children:"Bind existing policy"}),". This assumes you have already created the groups you want to use for ",(0,t.jsx)(n.code,{children:"Dashy"})," and populated users in those groups."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/10fca15b-e77d-4624-ae03-0ece3910904c",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Click ",(0,t.jsx)(n.code,{children:"Group"})," for the binding type. Under ",(0,t.jsx)(n.code,{children:"Group"})," select the appropriate group you would like to bind. Make sure ",(0,t.jsx)(n.code,{children:"Enabled"})," is toggeled on. Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ebf680ab-696f-4c08-ae89-d73fe92b398f",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"Dashy"})," will now be scoped only to users within the assigned groups you have bound the application to. Keep adding groups if you would like to adjust the dashboard visibilty based on group membership."]}),"\n",(0,t.jsxs)(n.h4,{id:"4-configure-dashy-to-use-oidc-client",children:["4. Configure ",(0,t.jsx)(n.code,{children:"Dashy"})," to use OIDC client"]}),"\n",(0,t.jsx)(n.admonition,{type:"important",children:(0,t.jsxs)(n.p,{children:["It is highly recommended to edit your ",(0,t.jsx)(n.code,{children:"conf.yml"})," directly for this step."]})}),"\n",(0,t.jsx)(n.admonition,{type:"caution",children:(0,t.jsxs)(n.p,{children:["Do not make the same mistake many have made here by including the fully qualified address for the ",(0,t.jsx)(n.code,{children:"OpenID Configuration URL"}),". ",(0,t.jsx)(n.code,{children:"Dashy"})," will append the ",(0,t.jsx)(n.code,{children:".well-known"})," configuration automatically. If the ",(0,t.jsx)(n.code,{children:".well-known"})," URI is included the app will get redirect loops and ",(0,t.jsx)(n.code,{children:"400"})," errors."]})}),"\n",(0,t.jsxs)(n.p,{children:["Enter the ",(0,t.jsx)(n.code,{children:"Client ID"})," in the ",(0,t.jsx)(n.code,{children:"clientId"})," field and ",(0,t.jsx)(n.code,{children:"OpenID Configuration Issuer"})," in the ",(0,t.jsx)(n.code,{children:"endpoint"})," field."]}),"\n",(0,t.jsxs)(n.p,{children:["Below is how to configure the ",(0,t.jsx)(n.code,{children:"auth"})," section in the yaml syntax. Once this is enabled, when an attempt to access ",(0,t.jsx)(n.code,{children:"Dashy"})," is made it will now redirect you to the ",(0,t.jsx)(n.code,{children:"authentik"})," login page moving forward."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n theme: glass\n layout: auto\n iconSize: medium\n disableConfigurationForNonAdmin: true # Prevent logged-in, non-admins using the view/edit config features\n auth:\n enableOidc: true\n oidc:\n clientId: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15\n endpoint: https://auth.domain.com/application/o/dashy/\n"})}),"\n",(0,t.jsxs)(n.h4,{id:"5-optional-example-snippets-for-dashboard-visibility",children:["5. ",(0,t.jsx)(n.em,{children:"(OPTIONAL)"})," Example snippets for dashboard visibility"]}),"\n",(0,t.jsxs)(n.p,{children:["Using the ",(0,t.jsx)(n.code,{children:"hideForKeycloakUsers"})," configuration option is needed to use the ",(0,t.jsx)(n.code,{children:"authentik"})," groups that were created previously."]}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"pages"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"pages:\n - name: App Management\n path: appmgmt.yml\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n - name: Network Management\n path: network.yml\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"items"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:" items:\n - title: Authentik Admin\n icon: authentik.svg\n url: https://auth.domain.com/if/admin/\n target: newtab\n id: 0_1472_authentikadmin\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n - title: Authentik User\n icon: authentik-light.png\n url: https://auth.domain.com/if/user/\n target: newtab\n id: 1_1472_authentikuser\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"sections"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"sections:\n - name: Authentication\n displayData:\n sortBy: default\n cols: 1\n collapsed: false\n hideForGuests: false\n hideForKeycloakUsers:\n groups: \n - Dashy Users\n"})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"alternative-authentication-methods",children:"Alternative Authentication Methods"}),"\n",(0,t.jsx)(n.p,{children:"These are alternatives to Dashy's built-in auth, Keycloak, and OIDC. Most of them sit in front of Dashy at the network or reverse proxy level, which is generally the better approach for anything internet-facing."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#reverse-proxy-auth",children:"Reverse Proxy Auth"})," - Authelia, Authentik, or similar sitting in front of Dashy"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#zero-trust-tunnels",children:"Zero-Trust Tunnels"})," - Cloudflare Tunnel, Tailscale Funnel"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#vpn",children:"VPN"})," - Keep Dashy off the internet entirely"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#ip-based-access",children:"IP-Based Access"})," - Restrict by source IP in your web server"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#web-server-authentication",children:"Web Server Authentication"})," - HTTP basic auth at the proxy level"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#sso--oauth-providers",children:"SSO / OAuth Providers"})," - Cloud-hosted identity providers"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#cloud-hosting-providers",children:"Cloud Hosting Providers"})," - Built-in auth on hosting platforms"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"reverse-proxy-auth",children:"Reverse proxy auth"}),"\n",(0,t.jsx)(n.p,{children:"The most common setup for self-hosters running multiple services. You put an auth server in front of your reverse proxy, and it handles login, 2FA, and sessions for everything behind it. You configure it once, and all your apps get protected."}),"\n",(0,t.jsxs)(n.p,{children:["Dashy has ",(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Authentication"})," support, so when your proxy authenticates a user and forwards their identity via a header, Dashy picks up the username and maps it to a configured user automatically. No separate Dashy login needed."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Authelia"})," is lightweight and Docker-friendly. It supports 2FA, per-path access rules, and multiple user backends. To get started quickly:"]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"git clone https://github.com/authelia/authelia.git"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"cd authelia/examples/compose/lite"})}),"\n",(0,t.jsxs)(n.li,{children:["Edit ",(0,t.jsx)(n.code,{children:"users_database.yml"}),", ",(0,t.jsx)(n.code,{children:"configuration.yml"}),", and ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," for your domain and users"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"docker compose up -d"})}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["See the ",(0,t.jsx)(n.a,{href:"https://www.authelia.com/docs/",children:"Authelia docs"})," for the full setup guide."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Authentik"})," is heavier but gives you a proper admin UI, built-in OIDC/SAML support, and user self-service (password resets, enrollment flows, etc). Good if you want a single identity provider across many apps. See the ",(0,t.jsx)(n.a,{href:"https://docs.goauthentik.io/docs/installation/docker-compose",children:"authentik Docker Compose install"})," to get started, and the ",(0,t.jsx)(n.a,{href:"#authentik",children:"authentik section"})," above for Dashy-specific OIDC config."]}),"\n",(0,t.jsx)(n.h3,{id:"zero-trust-tunnels",children:"Zero-trust tunnels"}),"\n",(0,t.jsx)(n.p,{children:"These let you expose Dashy to the internet without opening inbound ports or configuring port forwarding. Auth is handled by the tunnel provider before traffic ever reaches your server."}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Cloudflare Tunnel"})," connects Dashy to Cloudflare's edge network via an outbound-only ",(0,t.jsx)(n.code,{children:"cloudflared"})," daemon (runs nicely as a Docker sidecar). Cloudflare handles DNS, TLS, and DDoS protection. Pair it with Cloudflare Access to require identity provider login before anyone reaches Dashy. The free tier covers most home setups. See the ",(0,t.jsx)(n.a,{href:"https://developers.cloudflare.com/cloudflare-one/connections/connect-networks/",children:"Cloudflare Tunnel docs"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Tailscale Funnel"})," exposes Dashy through your Tailscale mesh to the public internet, with automatic TLS. Simpler to set up than Cloudflare but you get less control over access policies. See the ",(0,t.jsx)(n.a,{href:"https://tailscale.com/kb/1223/funnel",children:"Funnel docs"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"vpn",children:"VPN"}),"\n",(0,t.jsx)(n.p,{children:"A VPN keeps Dashy off the public internet entirely. You connect to your home network remotely and access Dashy like you're on the LAN. No auth to configure, no attack surface to worry about. The downside: you need the VPN running to see anything, and some networks (corporate WiFi, hotels) block VPN traffic."}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://www.wireguard.com/",children:"WireGuard"})," is fast and minimal. Most self-hosters run it through a UI like ",(0,t.jsx)(n.a,{href:"https://github.com/wg-easy/wg-easy",children:"wg-easy"}),", which gives you a web interface for managing peers and generating QR codes for mobile."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://tailscale.com/",children:"Tailscale"})," wraps WireGuard and takes care of NAT traversal, key exchange, and device management. No port forwarding needed, works across networks with zero config. There's a generous free tier. ",(0,t.jsx)(n.a,{href:"https://github.com/juanfont/headscale",children:"Headscale"})," is a self-hosted coordination server if you want to keep everything on your own infrastructure."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://openvpn.net/",children:"OpenVPN"})," still works fine if you already have it running, but for a new setup WireGuard or Tailscale are easier to get going."]}),"\n",(0,t.jsx)(n.h3,{id:"ip-based-access",children:"IP-based access"}),"\n",(0,t.jsx)(n.p,{children:"If you have a static IP or are already on a VPN, you can restrict access to Dashy by source IP at the web server level. This works well as an extra layer on top of other auth methods."}),"\n",(0,t.jsx)(n.p,{children:"NGINX:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"location / {\n proxy_pass http://dashy:8080;\n allow 192.168.1.0/24;\n allow 203.0.113.50;\n deny all;\n}\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Caddy (",(0,t.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/matchers",children:"request matchers docs"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'dashy.example.com {\n @blocked not remote_ip 192.168.1.0/24 203.0.113.50\n respond @blocked "Access denied" 403\n reverse_proxy dashy:8080\n}\n'})}),"\n",(0,t.jsx)(n.p,{children:"Apache (2.4+):"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"\n Require ip 192.168.1.0/24\n Require ip 203.0.113.50\n\n"})}),"\n",(0,t.jsx)(n.h3,{id:"web-server-authentication",children:"Web server authentication"}),"\n",(0,t.jsx)(n.p,{children:"Your reverse proxy can handle HTTP basic auth directly, no extra services needed. This gives you a browser login prompt in front of Dashy. Make sure you're using HTTPS, as basic auth sends credentials base64-encoded (not encrypted) with every request."}),"\n",(0,t.jsxs)(n.p,{children:["NGINX (",(0,t.jsx)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_auth_basic_module.html",children:"auth module docs"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'location / {\n auth_basic "Dashy";\n auth_basic_user_file /etc/nginx/conf.d/.htpasswd;\n proxy_pass http://dashy:8080;\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Generate the password file with ",(0,t.jsx)(n.code,{children:"htpasswd -c /etc/nginx/conf.d/.htpasswd alicia"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["Caddy (",(0,t.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/basicauth",children:"basicauth directive"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"dashy.example.com {\n basicauth {\n alicia $2a$14$... # generate with: caddy hash-password\n }\n reverse_proxy dashy:8080\n}\n"})}),"\n",(0,t.jsx)(n.p,{children:"Apache:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'AuthType Basic\nAuthName "Dashy"\nAuthUserFile /path/to/.htpasswd\nRequire valid-user\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Generate the password file with ",(0,t.jsx)(n.code,{children:"htpasswd -c /path/to/.htpasswd alicia"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"sso--oauth-providers",children:"SSO / OAuth providers"}),"\n",(0,t.jsxs)(n.p,{children:["Cloud identity providers like ",(0,t.jsx)(n.a,{href:"https://auth0.com/",children:"Auth0"}),", ",(0,t.jsx)(n.a,{href:"https://developer.okta.com/",children:"Okta"}),", ",(0,t.jsx)(n.a,{href:"https://www.ory.sh/",children:"Ory"}),", and ",(0,t.jsx)(n.a,{href:"https://cloud.google.com/identity",children:"Google Cloud Identity"})," can work with Dashy through its ",(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC support"}),". If your provider speaks OIDC (most do), just configure it as described in the OIDC section and you're set."]}),"\n",(0,t.jsx)(n.p,{children:"For providers that only support OAuth2 or SAML without an OIDC layer, you'll need something in between to translate. Authentik, Keycloak, and Authelia can all bridge from SAML/OAuth2 to OIDC."}),"\n",(0,t.jsx)(n.h3,{id:"cloud-hosting-providers",children:"Cloud hosting providers"}),"\n",(0,t.jsxs)(n.p,{children:["If you're running Dashy on a cloud platform, most have their own auth options you can enable without touching Dashy's config. See your provider's docs: ",(0,t.jsx)(n.a,{href:"https://www.cloudflare.com/teams/access/",children:"Cloudflare Access"}),", ",(0,t.jsx)(n.a,{href:"https://docs.netlify.com/visitor-access/password-protection/",children:"Netlify Password Protection"}),", ",(0,t.jsx)(n.a,{href:"https://aws.amazon.com/cognito/",children:"AWS Cognito"}),", ",(0,t.jsx)(n.a,{href:"https://learn.microsoft.com/en-us/azure/app-service/overview-authentication-authorization",children:"Azure App Service Authentication"}),", and ",(0,t.jsx)(n.a,{href:"https://vercel.com/docs/security/password-protection",children:"Vercel Password Protection"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>r,x:()=>a});var i=s(6540);const t={},o=i.createContext(t);function r(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[1831],{5159(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"authentication","title":"Authentication","description":"- Basic Auth","source":"@site/docs/authentication.md","sourceDirName":".","slug":"/authentication","permalink":"/docs/authentication","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/authentication.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Status Indicators","permalink":"/docs/status-indicators"},"next":{"title":"Keyboard Shortcuts","permalink":"/docs/searching"}}');var t=s(4848),o=s(8453);const r={},a="Authentication",c={},l=[{value:"Built-In Auth",id:"built-in-auth",level:2},{value:"Setting Up Authentication",id:"setting-up-authentication",level:3},{value:"Hash Password",id:"hash-password",level:3},{value:"Logging In and Out",id:"logging-in-and-out",level:3},{value:"Enabling Guest Access",id:"enabling-guest-access",level:3},{value:"Granular Access",id:"granular-access",level:3},{value:"Permissions",id:"permissions",level:3},{value:"Using Environment Variables for Passwords",id:"using-environment-variables-for-passwords",level:3},{value:"Adding HTTP Auth to Configuration",id:"adding-http-auth-to-configuration",level:3},{value:"Security",id:"security",level:3},{value:"HTTP Auth",id:"http-auth",level:2},{value:"Using config-file users (recommended)",id:"using-config-file-users-recommended",level:3},{value:"Using static credentials",id:"using-static-credentials",level:3},{value:"Keycloak",id:"keycloak",level:2},{value:"1. Deploy Keycloak",id:"1-deploy-keycloak",level:3},{value:"2. Setup Keycloak Users",id:"2-setup-keycloak-users",level:3},{value:"3. Enable Keycloak in Dashy Config File",id:"3-enable-keycloak-in-dashy-config-file",level:3},{value:"4. Add groups and roles (Optional)",id:"4-add-groups-and-roles-optional",level:3},{value:"CORS Headers",id:"cors-headers",level:3},{value:"How server-side enforcement works",id:"how-server-side-enforcement-works",level:3},{value:"Troubleshooting Keycloak",id:"troubleshooting-keycloak",level:3},{value:"Header Authentication",id:"header-authentication",level:2},{value:"Configuration",id:"configuration",level:3},{value:"How it Works",id:"how-it-works",level:3},{value:"Notes",id:"notes",level:3},{value:"OIDC",id:"oidc",level:2},{value:"How server-side enforcement works",id:"how-server-side-enforcement-works-1",level:3},{value:"authentik",id:"authentik",level:2},{value:"1. Create an OIDC provider",id:"1-create-an-oidc-provider",level:4},{value:"2. Create an application",id:"2-create-an-application",level:4},{value:"3. (Optional) Limiting access via authentik with groups",id:"3-optional-limiting-access-via-authentik-with-groups",level:4},{value:"4. Configure Dashy to use OIDC client",id:"4-configure-dashy-to-use-oidc-client",level:4},{value:"5. (OPTIONAL) Example snippets for dashboard visibility",id:"5-optional-example-snippets-for-dashboard-visibility",level:4},{value:"Alternative Authentication Methods",id:"alternative-authentication-methods",level:2},{value:"Reverse proxy auth",id:"reverse-proxy-auth",level:3},{value:"Zero-trust tunnels",id:"zero-trust-tunnels",level:3},{value:"VPN",id:"vpn",level:3},{value:"IP-based access",id:"ip-based-access",level:3},{value:"Web server authentication",id:"web-server-authentication",level:3},{value:"SSO / OAuth providers",id:"sso--oauth-providers",level:3},{value:"Cloud hosting providers",id:"cloud-hosting-providers",level:3}];function d(e){const n={a:"a",admonition:"admonition",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"authentication",children:"Authentication"})}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#built-in-auth",children:"Basic Auth"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#setting-up-authentication",children:"Setting Up Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#hash-password",children:"Hash Password"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#logging-in-and-out",children:"Logging In and Out"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#enabling-guest-access",children:"Guest Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#granular-access",children:"Granular Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#permissions",children:"Permissions"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-environment-variables-for-passwords",children:"Using Environment Variables for Passwords"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#security",children:"Security"})}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#http-auth",children:"HTTP Auth"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-config-file-users-recommended",children:"Using Config-File Users"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#using-static-credentials",children:"Using Static Credentials"})}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#authentik",children:"authentik"})}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#alternative-authentication-methods",children:"Alternative Authentication Methods"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#reverse-proxy-auth",children:"Reverse Proxy Auth"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#zero-trust-tunnels",children:"Zero-Trust Tunnels"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#vpn",children:"VPN"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#ip-based-access",children:"IP-Based Access"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#web-server-authentication",children:"Web Server Authentication"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#sso--oauth-providers",children:"SSO / OAuth Providers"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#cloud-hosting-providers",children:"Cloud Hosting Providers"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.admonition,{type:"important",children:[(0,t.jsx)(n.p,{children:"Dashy's built-in auth is not intended to protect a publicly hosted instance against unauthorized access. Instead you should use an auth provider compatible with your reverse proxy, or access Dashy via your VPN, or implement your own SSO logic."}),(0,t.jsxs)(n.p,{children:["If Dashy is only accessible within your home network and you just want a login page, then the built-in auth may be sufficient. To also protect server-side endpoints and config files: with built-in auth set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," (",(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"details"}),"). (Or, consider setting up",(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC"}),", ",(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak"}),", or ",(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Auth"}),", where the server-side enforcement is on automatically)."]})]}),"\n",(0,t.jsx)(n.h2,{id:"built-in-auth",children:"Built-In Auth"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy has a basic login page included, and frontend authentication. You can enable this by adding users to the ",(0,t.jsx)(n.code,{children:"auth"})," section under ",(0,t.jsx)(n.code,{children:"appConfig"})," in your ",(0,t.jsx)(n.code,{children:"conf.yml"}),". If this section is not specified, then no authentication will be required to access the app, and the homepage will resolve to your dashboard. To also enable HTTP Authorization, set the ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"})," env var to ",(0,t.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"setting-up-authentication",children:"Setting Up Authentication"}),"\n",(0,t.jsxs)(n.p,{children:["The ",(0,t.jsx)(n.code,{children:"auth"})," property takes an array of users. Each user needs to include a username, hash and optional user type (",(0,t.jsx)(n.code,{children:"admin"})," or ",(0,t.jsx)(n.code,{children:"normal"}),"). The hash property is a ",(0,t.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/SHA-2",children:"SHA-256 Hash"})," of your desired password."]}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n auth:\n users:\n - user: alicia\n hash: 4D1E58C90B3B94BCAD9848ECCACD6D2A8C9FBC5CA913304BBA5CDEAB36FEEFA3\n type: admin\n - user: bob\n hash: 5E884898DA28047151D0E56F8DC6292773603D0D6AABBDD62A11EF721D1542D8\n"})}),"\n",(0,t.jsx)(n.h3,{id:"hash-password",children:"Hash Password"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy uses ",(0,t.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/Sha-256",children:"SHA-256 Hash"}),", a 64-character string, which you can generate by running ",(0,t.jsx)(n.code,{children:'echo -n "my-super-secure-password" | sha256sum'}),", or using an online tool, such as ",(0,t.jsx)(n.a,{href:"https://passwordsgenerator.net/sha256-hash-generator/",children:"this one"})," or ",(0,t.jsx)(n.a,{href:"https://gchq.github.io/CyberChef/",children:"CyberChef"})," (which can be self-hosted/ ran locally)."]}),"\n",(0,t.jsx)(n.p,{children:"A hash is a one-way cryptographic function, meaning that it is easy to generate a hash for a given password, but very hard to determine the original password for a given hash. This means, that so long as your password is long, strong and unique, it is safe to store its hash in the clear. Having said that, you should never reuse passwords, hashes can be cracked by iterating over known password lists, generating a hash of each."}),"\n",(0,t.jsx)(n.h3,{id:"logging-in-and-out",children:"Logging In and Out"}),"\n",(0,t.jsx)(n.p,{children:"Once authentication is enabled, so long as there is no valid token in cookie storage, the application will redirect the user to the login page. When the user enters credentials in the login page, they will be checked, and if valid, then a token will be generated, and they can be redirected to the home page. If credentials are invalid, then an error message will be shown, and they will remain on the login page. Once in the application, to log out: the user can click the logout button (in the top-right), which will clear cookie storage, causing them to be redirected back to the login page."}),"\n",(0,t.jsx)(n.h3,{id:"enabling-guest-access",children:"Enabling Guest Access"}),"\n",(0,t.jsxs)(n.p,{children:["With authentication set up, by default no access is allowed to your dashboard without first logging in with valid credentials. Guest mode can be enabled to allow for read-only access to a secured dashboard by any user, without the need to log in. A guest user cannot write any changes to the config file, but can apply modifications locally (stored in their browser). You can enable guest access, by setting ",(0,t.jsx)(n.code,{children:"appConfig.auth.enableGuestAccess: true"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"granular-access",children:"Granular Access"}),"\n",(0,t.jsx)(n.p,{children:"You can use the following properties to make certain pages, sections or items only visible to some users, or hide pages, sections and items from guests."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"hideForUsers"})," - Page, Section or Item will be visible to all users, except for those specified in this list"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"showForUsers"})," - Page, Section or Item will be hidden from all users, except for those specified in this list"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"hideForGuests"})," - Page, Section or Item will be visible for logged in users, but not for guests"]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"For Example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"pages:\n - name: Home Lab\n path: home-lab.yml\n displayData:\n showForUsers: [admin]\n - name: Intranet\n path: intranet.yml\n displayData:\n hideForGuests: true\n hideForUsers: [alicia, bob]\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Code Analysis & Monitoring\n icon: fas fa-code\n displayData:\n cols: 2\n hideForUsers: [alicia, bob]\n items:\n ...\n"})}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"- name: Deployment Pipelines\n icon: fas fa-rocket\n displayData:\n hideForGuests: true\n items:\n - title: Hide Me\n displayData:\n hideForUsers: [alicia, bob]\n"})}),"\n",(0,t.jsx)(n.h3,{id:"permissions",children:"Permissions"}),"\n",(0,t.jsxs)(n.p,{children:["Any user who is not an admin (with ",(0,t.jsx)(n.code,{children:"type: admin"}),") will not be able to write changes to disk."]}),"\n",(0,t.jsxs)(n.p,{children:["You can also prevent any user from writing changes to disk, using ",(0,t.jsx)(n.code,{children:"preventWriteToDisk"}),". Or prevent any changes from being saved locally in browser storage, using ",(0,t.jsx)(n.code,{children:"preventLocalSave"}),". Both properties can be found under ",(0,t.jsx)(n.a,{href:"/docs/configuring#appconfig-optional",children:(0,t.jsx)(n.code,{children:"appConfig"})}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["To disable all UI config features, including View Config, set ",(0,t.jsx)(n.code,{children:"disableConfiguration"}),". Alternatively you can disable UI config features for all non admin users by setting ",(0,t.jsx)(n.code,{children:"disableConfigurationForNonAdmin"})," to true."]}),"\n",(0,t.jsx)(n.h3,{id:"using-environment-variables-for-passwords",children:"Using Environment Variables for Passwords"}),"\n",(0,t.jsxs)(n.p,{children:["If you don't want to hash your password, you can instead leave out the ",(0,t.jsx)(n.code,{children:"hash"})," attribute, and replace it with ",(0,t.jsx)(n.code,{children:"password"})," which should have the value of an environmental variable name you wish to use."]}),"\n",(0,t.jsxs)(n.p,{children:["Note that env var must begin with ",(0,t.jsx)(n.code,{children:"VITE_APP_"}),", and you must set this variable before building the app."]}),"\n",(0,t.jsx)(n.p,{children:"For example:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:" auth:\n users:\n - user: bob\n password: VITE_APP_BOB\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Just be sure to set ",(0,t.jsx)(n.code,{children:"VITE_APP_BOB='my super secret password'"})," before build-time."]}),"\n",(0,t.jsx)(n.h3,{id:"adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"}),"\n",(0,t.jsxs)(n.p,{children:["Without this, the built-in auth is just a client-side login page \u2014 your config and API endpoints can still be accessed directly. Set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," to protect them."]}),"\n",(0,t.jsx)(n.admonition,{type:"note",children:(0,t.jsxs)(n.p,{children:["HTTP Auth and guest access (",(0,t.jsx)(n.code,{children:"enableGuestAccess"}),") are incompatible. Guests have no credentials, so they can't fetch the config file when HTTP auth is active."]})}),"\n",(0,t.jsxs)(n.p,{children:["This uses the same users you've already defined in ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"})," to authenticate all server-side requests (config files, status checks, system info, CORS proxy, etc.) via HTTP Basic Auth."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"How it works:"})," When a user logs in through the Dashy UI, a session token is stored in a cookie. The frontend automatically includes this token in requests to local API endpoints. On the server side, the token is validated against your configured users. If someone tries to access an endpoint directly (e.g. with curl), the server will respond with a ",(0,t.jsx)(n.code,{children:"401"})," and a Basic Auth challenge \u2014 they'll need to provide a valid username and password."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:"Setup:"})}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Make sure you have users configured in your ",(0,t.jsx)(n.code,{children:"conf.yml"})," (see ",(0,t.jsx)(n.a,{href:"#setting-up-authentication",children:"Setting Up Authentication"})," above)"]}),"\n",(0,t.jsxs)(n.li,{children:["Set the ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," environment variable (e.g. in your ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," or ",(0,t.jsx)(n.code,{children:".env"})," file)"]}),"\n",(0,t.jsx)(n.li,{children:"Restart the container - the auth mode is determined at startup, so env var changes need a restart"}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["Adding or removing users in ",(0,t.jsx)(n.code,{children:"conf.yml"})," takes effect immediately without a restart, since the user list is read from disk on each request."]}),"\n",(0,t.jsxs)(n.p,{children:["For full protection, you'll want both the client-side login page (via ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"}),") and server-side auth (via ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"}),")."]}),"\n",(0,t.jsx)(n.h3,{id:"security",children:"Security"}),"\n",(0,t.jsx)(n.p,{children:"With basic auth (and without HTTP auth), the login logic runs on the client-side. A technical user could inspect the code and view parts of your configuration, including password hashes. If the SHA-256 hash is of a common password, it may be possible to determine it using a lookup table, and then use that to generate a valid auth token. Therefore, you should always use a long, strong and unique password."}),"\n",(0,t.jsxs)(n.p,{children:["If your instance is exposed to the internet, the built-in auth alone is not sufficient - use a reverse proxy with its own authentication layer (see ",(0,t.jsx)(n.a,{href:"#alternative-authentication-methods",children:"Alternative Authentication Methods"}),"), or access Dashy over a VPN. See the ",(0,t.jsx)(n.a,{href:"/docs/management#network-exposure",children:"Network Exposure"})," section in the management docs for more on this."]}),"\n",(0,t.jsx)(n.p,{children:"The built-in login page prevents casual unauthorized access on a private network. It's not a security perimeter."}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"http-auth",children:"HTTP Auth"}),"\n",(0,t.jsx)(n.p,{children:"If you'd like to protect server-side endpoints with HTTP Basic Auth, there are two approaches. They protect the same endpoints but use different credential sources, so pick one - don't combine them."}),"\n",(0,t.jsx)(n.h3,{id:"using-config-file-users-recommended",children:"Using config-file users (recommended)"}),"\n",(0,t.jsxs)(n.p,{children:["This is the approach described in ",(0,t.jsx)(n.a,{href:"#adding-http-auth-to-configuration",children:"Adding HTTP Auth to Configuration"})," above. Set ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH=true"})," and it uses the same ",(0,t.jsx)(n.code,{children:"appConfig.auth.users"})," from your ",(0,t.jsx)(n.code,{children:"conf.yml"}),". The frontend handles authentication automatically using the session token from the login page, so no extra setup is needed."]}),"\n",(0,t.jsx)(n.p,{children:"This is the recommended approach because it keeps credentials in one place and works together with the client-side login page. But the drawback is that your credentials will be stored in your config file."}),"\n",(0,t.jsx)(n.h3,{id:"using-static-credentials",children:"Using static credentials"}),"\n",(0,t.jsxs)(n.p,{children:["If you don't have users in your ",(0,t.jsx)(n.code,{children:"conf.yml"})," (e.g. you handle user management externally, or just want a single shared password for server-side access), you can set the ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"})," and ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," environmental variables instead."]}),"\n",(0,t.jsxs)(n.p,{children:["With this approach, there is no Dashy login page. When the browser first requests the config file, the server responds with a ",(0,t.jsx)(n.code,{children:"401"})," and the browser shows its native HTTP auth prompt. Once the user enters the correct credentials, the browser caches them for the session and all subsequent requests work."]}),"\n",(0,t.jsxs)(n.p,{children:["To skip the browser prompt and have the frontend authenticate automatically, also set ",(0,t.jsx)(n.code,{children:"VITE_APP_BASIC_AUTH_USERNAME"})," and ",(0,t.jsx)(n.code,{children:"VITE_APP_BASIC_AUTH_PASSWORD"})," to the same values. These are baked in at build time, so a rebuild is required, and you should only do this on a trusted network."]}),"\n",(0,t.jsx)(n.admonition,{type:"warning",children:(0,t.jsxs)(n.p,{children:["Do not combine ",(0,t.jsx)(n.code,{children:"BASIC_AUTH_USERNAME"}),"/",(0,t.jsx)(n.code,{children:"BASIC_AUTH_PASSWORD"})," with conf.yml users. If both are present, the server will log a warning at startup. With ",(0,t.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"})," set, config-file users take priority and the static credentials are ignored. Without it, the static credentials protect the server but the Dashy login page will use conf.yml credentials, and the frontend will send the wrong credentials to server endpoints. Pick one approach or the other."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"keycloak",children:"Keycloak"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy also supports using a ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/",children:"Keycloak"})," authentication server. The setup for this is a bit more involved, but it gives you greater security overall, useful for if your instance is exposed to the internet."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://www.keycloak.org/about.html",children:"Keycloak"})," is a Java-based ",(0,t.jsx)(n.a,{href:"https://github.com/keycloak/keycloak",children:"open source"}),", high-performance, secure authentication system, supported by ",(0,t.jsx)(n.a,{href:"https://www.redhat.com/en",children:"RedHat"}),". It is easy to setup (",(0,t.jsx)(n.a,{href:"https://quay.io/repository/keycloak/keycloak",children:"with Docker"}),"), and enables you to secure multiple self-hosted applications with single-sign-on using standard protocols (OpenID Connect, OAuth 2.0, SAML 2.0 and social login). It's also very customizable, you can write or use custom ",(0,t.jsx)(n.a,{href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_development/topics/themes.html",children:"themes"}),", ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/extensions.html",children:"plugins"}),", ",(0,t.jsx)(n.a,{href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_admin/topics/authentication/password-policies.html",children:"password policies"})," and more.\nThe following guide will walk you through setting up Keycloak with Dashy. If you already have a Keycloak instance configured, then skip to Step 3."]}),"\n",(0,t.jsx)(n.h3,{id:"1-deploy-keycloak",children:"1. Deploy Keycloak"}),"\n",(0,t.jsxs)(n.p,{children:["First thing to do is to spin up a new instance of Keycloak. You will need ",(0,t.jsx)(n.a,{href:"https://docs.docker.com/engine/install/",children:"Docker installed"}),", and can then choose a tag, and pull the container from ",(0,t.jsx)(n.a,{href:"https://quay.io/repository/keycloak/keycloak",children:"quay.io/keycloak/keycloak"})]}),"\n",(0,t.jsx)(n.p,{children:"Use the following run command, replacing the attributes (default credentials, port and name), or incorporate this into your docker-compose file."}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-bash",children:"docker run -d \\\n -p 8081:8080 \\\n --name auth-server \\\n -e KEYCLOAK_ADMIN=admin \\\n -e KEYCLOAK_ADMIN_PASSWORD=admin \\\n quay.io/keycloak/keycloak:25.0 start-dev\n"})}),"\n",(0,t.jsxs)(n.p,{children:["(The ",(0,t.jsx)(n.code,{children:"KEYCLOAK_USER"})," / ",(0,t.jsx)(n.code,{children:"KEYCLOAK_PASSWORD"})," env vars and the ",(0,t.jsx)(n.code,{children:"/auth"})," URL prefix from Keycloak 16 and older have been replaced. If you are still on 17 or older, set ",(0,t.jsx)(n.code,{children:"legacySupport: true"})," in your Dashy config later on.)"]}),"\n",(0,t.jsxs)(n.p,{children:["If you need to pull from DockerHub, a non-official image is available ",(0,t.jsx)(n.a,{href:"https://registry.hub.docker.com/r/jboss/keycloak",children:"here"}),". Or if you would prefer not to use Docker, you can also directly install Keycloak from source, following ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/docs/latest/getting_started/index.html",children:"this guide"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["You should now be able to access the Keycloak web interface, using the port specified above (e.g. ",(0,t.jsx)(n.code,{children:"http://127.0.0.1:8081"}),"), login with the default credentials, and when prompted create a new password."]}),"\n",(0,t.jsx)(n.h3,{id:"2-setup-keycloak-users",children:"2. Setup Keycloak Users"}),"\n",(0,t.jsx)(n.p,{children:"Before we can use Keycloak, we must first set it up with some users. Keycloak uses Realms (similar to tenants) to create isolated groups of users. You must create a Realm before you will be able to add your first user."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Head over to the admin console"}),"\n",(0,t.jsx)(n.li,{children:"In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm'"}),"\n",(0,t.jsx)(n.li,{children:"Give your realm a name, and hit 'Create'"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"You can now create your first user."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"In the left-hand menu, click 'Users', then 'Add User'"}),"\n",(0,t.jsxs)(n.li,{children:["Fill in the form. On Keycloak 25 and newer, ",(0,t.jsx)(n.em,{children:"First name"})," and ",(0,t.jsx)(n.em,{children:"Last name"}),' are required by the default user-profile schema. If you skip them the user can sign in but login will then fail with "Account is not fully set up"']}),"\n",(0,t.jsx)(n.li,{children:"Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"Next, create a new client for Dashy."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right"}),"\n",(0,t.jsxs)(n.li,{children:["Choose a 'Client ID' (e.g. ",(0,t.jsx)(n.code,{children:"dashy"}),"), set 'Client Protocol' to 'openid-connect'"]}),"\n",(0,t.jsxs)(n.li,{children:["Turn ",(0,t.jsx)(n.em,{children:"Client authentication"})," OFF and leave ",(0,t.jsx)(n.em,{children:"Standard flow"})," enabled. Dashy is a SPA, so it acts as an OAuth public client with PKCE. A confidential client requires a client_secret that a browser app can't safely hold"]}),"\n",(0,t.jsxs)(n.li,{children:["For 'Valid Redirect URIs' put the URL where you host Dashy (e.g. ",(0,t.jsx)(n.code,{children:"https://dashy.example.com/*"}),", or just ",(0,t.jsx)(n.code,{children:"*"})," while testing locally). Do the same for the 'Web Origins' field"]}),"\n",(0,t.jsx)(n.li,{children:"Make note of your client-id, and click 'Save'"}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["For the ",(0,t.jsx)(n.code,{children:"adminRole"})," check to work, the role must appear in the id_token (Keycloak's default mapper only adds it to the access token):"]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Open your ",(0,t.jsx)(n.code,{children:"dashy"})," client, go to the ",(0,t.jsx)(n.em,{children:"Client scopes"})," tab, click the dedicated scope row (",(0,t.jsx)(n.code,{children:"dashy-dedicated"}),")"]}),"\n",(0,t.jsxs)(n.li,{children:["Add a new mapper of type ",(0,t.jsx)(n.em,{children:"User Realm Role"}),", name it (e.g. ",(0,t.jsx)(n.code,{children:"realm_roles"}),"), claim name ",(0,t.jsx)(n.code,{children:"realm_access.roles"}),", multivalued ON, ",(0,t.jsx)(n.em,{children:"Add to ID token"})," ON, ",(0,t.jsx)(n.em,{children:"Add to access token"})," ON"]}),"\n",(0,t.jsxs)(n.li,{children:["(Optional, for ",(0,t.jsx)(n.code,{children:"adminGroup"})," instead of ",(0,t.jsx)(n.code,{children:"adminRole"}),") Add a second mapper of type ",(0,t.jsx)(n.em,{children:"Group Membership"}),", claim name ",(0,t.jsx)(n.code,{children:"groups"})]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"To create the admin role itself and grant it to a user:"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.em,{children:"Realm roles"})," in the left-hand menu, ",(0,t.jsx)(n.em,{children:"Create role"}),", name it (e.g. ",(0,t.jsx)(n.code,{children:"dashy-admin"}),")"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.em,{children:"Users"})," \u2192 pick your admin user \u2192 ",(0,t.jsx)(n.em,{children:"Role mapping"})," \u2192 ",(0,t.jsx)(n.em,{children:"Assign role"})," \u2192 select ",(0,t.jsx)(n.code,{children:"dashy-admin"})]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"3-enable-keycloak-in-dashy-config-file",children:"3. Enable Keycloak in Dashy Config File"}),"\n",(0,t.jsxs)(n.p,{children:["Now that your Keycloak instance is up and running, all that's left to do is to configure Dashy to use it. Under ",(0,t.jsx)(n.code,{children:"appConfig"}),", set ",(0,t.jsx)(n.code,{children:"auth.enableKeycloak: true"}),", then fill in the details in ",(0,t.jsx)(n.code,{children:"auth.keycloak"}),", including: ",(0,t.jsx)(n.code,{children:"serverUrl"})," - the URL where your Keycloak instance is hosted, ",(0,t.jsx)(n.code,{children:"realm"})," - the name you gave your Realm, and ",(0,t.jsx)(n.code,{children:"clientId"})," - the Client ID you chose.\nFor example:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n # ...\n disableConfigurationForNonAdmin: true\n auth:\n enableKeycloak: true\n keycloak:\n serverUrl: 'http://localhost:8081'\n realm: 'alicia-homelab'\n clientId: 'dashy'\n adminRole: 'dashy-admin' # role name that grants admin privileges\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Note that if you are using Keycloak V 17 or older, you will also need to set ",(0,t.jsx)(n.code,{children:"legacySupport: true"})," (also under ",(0,t.jsx)(n.code,{children:"appConfig.auth.keycloak"}),"). This is because the API endpoint was updated in later versions."]}),"\n",(0,t.jsxs)(n.p,{children:["If you use Keycloak with an external Identity Provier, you can set the ",(0,t.jsx)(n.code,{children:"idpHint: 'alias-of-kc-idp'"})," option to allow the IdP Hint to be passed to Keycloak. This will cause Keycloak to skip its login page and redirect the user directly to the specified IdP's login page. Set to the value of the 'Alias' field of the desired IdP as defined in Keycloak under 'Identity Providers'."]}),"\n",(0,t.jsx)(n.h3,{id:"4-add-groups-and-roles-optional",children:"4. Add groups and roles (Optional)"}),"\n",(0,t.jsxs)(n.p,{children:["Keycloak allows you to assign users roles and groups. You can use these values to configure who can access various sections or items in Dashy.\nKeycloak server administration and configuration is a deep topic; please refer to the ",(0,t.jsx)(n.a,{href:"https://www.keycloak.org/docs/latest/server_admin/index.html#assigning-permissions-and-access-using-roles-and-groups",children:"server admin guide"})," to see details about creating and assigning roles and groups.\nOnce you have groups or roles assigned to users you can configure access under each section or item ",(0,t.jsx)(n.code,{children:"displayData.showForKeycloakUser"})," and ",(0,t.jsx)(n.code,{children:"displayData.hideForKeycloakUser"}),".\nBoth show and hide configurations accept a list of ",(0,t.jsx)(n.code,{children:"groups"})," and ",(0,t.jsx)(n.code,{children:"roles"})," that limit access. If a users data matches one or more items in these lists they will be allowed or excluded as defined."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"sections:\n - name: DeveloperResources\n displayData:\n showForKeycloakUsers:\n roles: ['canViewDevResources']\n hideForKeycloakUsers:\n groups: ['ProductTeam']\n items:\n - title: Not Visible for developers\n displayData:\n hideForKeycloakUsers:\n groups: ['DevelopmentTeam']\n"})}),"\n",(0,t.jsx)(n.p,{children:"Your app is now secured :) When you load Dashy, it will redirect to your Keycloak login page, and any user without valid credentials will be prevented from accessing your dashboard."}),"\n",(0,t.jsxs)(n.p,{children:["From within the Keycloak console, you can then configure things like time-outs, password policies, etc. You can also backup your full Keycloak config, and it is recommended to do this, along with your Dashy config. You can spin up both Dashy and Keycloak simultaneously and restore both applications configs using a ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," file, and this is recommended."]}),"\n",(0,t.jsx)(n.h3,{id:"cors-headers",children:"CORS Headers"}),"\n",(0,t.jsxs)(n.p,{children:["If Dashy and Keycloak run on different origins (typical when testing locally on different ",(0,t.jsx)(n.code,{children:"localhost:"})," ports), Keycloak's default ",(0,t.jsx)(n.code,{children:"Content-Security-Policy: frame-ancestors 'self'"})," and ",(0,t.jsx)(n.code,{children:"X-Frame-Options: SAMEORIGIN"})," block the hidden iframe ",(0,t.jsx)(n.code,{children:"keycloak-js"}),' uses to check your session. Symptom: a generic "Authentication failed (Keycloak)" toast on first load. To allow the iframe, open ',(0,t.jsx)(n.em,{children:"Realm settings \u2192 Security defenses \u2192 Browser headers"}),", clear ",(0,t.jsx)(n.code,{children:"X-Frame-Options"}),", and change ",(0,t.jsx)(n.code,{children:"Content-Security-Policy"})," to ",(0,t.jsx)(n.code,{children:"frame-src 'self' ; frame-ancestors 'self' ; object-src 'none';"}),". Same-origin production deployments don't hit this."]}),"\n",(0,t.jsx)(n.h3,{id:"how-server-side-enforcement-works",children:"How server-side enforcement works"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy's server reads ",(0,t.jsx)(n.code,{children:"auth.keycloak"})," from ",(0,t.jsx)(n.code,{children:"conf.yml"})," at boot, lazily fetches your Keycloak realm's OIDC discovery doc + JWKS, then verifies the ",(0,t.jsx)(n.code,{children:"id_token"})," the SPA attaches to every API call as ",(0,t.jsx)(n.code,{children:"Authorization: Bearer "}),". Tokens that fail signature / issuer / audience / expiry verification are rejected with ",(0,t.jsx)(n.code,{children:"401"}),". Write endpoints (",(0,t.jsx)(n.code,{children:"POST /config-manager/save"}),") additionally require the ",(0,t.jsx)(n.code,{children:"adminRole"})," (or ",(0,t.jsx)(n.code,{children:"adminGroup"}),") to be present in the token claims, and non-admins receive ",(0,t.jsx)(n.code,{children:"403"}),". Note that the ",(0,t.jsx)(n.code,{children:"/conf.yml"})," remains anonymously readable still (so the SPA can bootstrap before login)."]}),"\n",(0,t.jsxs)(n.p,{children:["The admin check reads the role / group claim from the id_token, so the client mapper from Step 2 above (the one with ",(0,t.jsx)(n.em,{children:"Add to ID token"})," on) is what makes ",(0,t.jsx)(n.code,{children:"adminRole"})," / ",(0,t.jsx)(n.code,{children:"adminGroup"})," work. Without it the server gets a token with no roles claim and treats everyone as non-admin."]}),"\n",(0,t.jsx)(n.h3,{id:"troubleshooting-keycloak",children:"Troubleshooting Keycloak"}),"\n",(0,t.jsx)(n.p,{children:"If you encounter issues with your Keycloak setup, follow these steps to troubleshoot and resolve common problems."}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsx)(n.p,{children:'Client Authentication Issue\nProblem: Redirect loop, if client authentication is enabled.\nSolution: Switch off "client authentication" in "TC clients" -> "Advanced" settings.'}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsxs)(n.p,{children:['Double URL\nProblem: If you get redirected to "',(0,t.jsx)(n.a,{href:"https://dashy.my.domain/#iss=https://keycloak.my.domain/realms/my-realm",children:"https://dashy.my.domain/#iss=https://keycloak.my.domain/realms/my-realm"}),'"\nSolution: Make sure to turn on "Exclude Issuer From Authentication Response" in "TC clients" -> "Advanced" -> "OpenID Connect Compatibility Modes"']}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:["\n",(0,t.jsxs)(n.p,{children:['Problems with mutiple Dashy Pages\nProblem: Refreshing or logging out of dashy results in an "invalid_redirect_uri" error.\nSolution: In "TC clients" -> "Access settings" -> "Root URL" ',(0,t.jsx)(n.a,{href:"https://dashy.my.domain/",children:"https://dashy.my.domain/"}),", valid redirect URIs must be /*"]}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"header-authentication",children:"Header Authentication"}),"\n",(0,t.jsxs)(n.p,{children:["Header authentication allows Dashy to trust an upstream reverse proxy to handle authentication. The proxy authenticates users and forwards their identity to Dashy via a configurable HTTP header (e.g. ",(0,t.jsx)(n.code,{children:"REMOTE_USER"}),"). This is the standard pattern used by ",(0,t.jsx)(n.a,{href:"https://www.authelia.com/",children:"Authelia"}),", ",(0,t.jsx)(n.a,{href:"https://goauthentik.io/",children:"Authentik"}),", Traefik's ",(0,t.jsx)(n.code,{children:"forwardAuth"}),", Caddy's ",(0,t.jsx)(n.code,{children:"forward_auth"}),", and Nginx's ",(0,t.jsx)(n.code,{children:"auth_request"}),"."]}),"\n",(0,t.jsx)(n.p,{children:"This is useful when you already have a central authentication layer in front of your self-hosted services and want Dashy to automatically pick up the authenticated user without requiring a separate login."}),"\n",(0,t.jsx)(n.h3,{id:"configuration",children:"Configuration"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n auth:\n enableHeaderAuth: true\n users:\n - user: alice\n hash: 0a7b1d4c2e... # SHA-256 hash of password\n type: admin\n - user: bob\n hash: 3f8e2b1a9d...\n type: normal\n headerAuth:\n userHeader: Remote-User\n proxyWhitelist:\n - 172.18.0.2\n - 127.0.0.1\n"})}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"userHeader"})})," - The HTTP header name containing the authenticated username. Defaults to ",(0,t.jsx)(n.code,{children:"Remote-User"})," if not specified. Common values: ",(0,t.jsx)(n.code,{children:"Remote-User"})," (Authelia), ",(0,t.jsx)(n.code,{children:"X-authentik-username"})," (Authentik), or whatever your proxy forwards. Header matching is case-insensitive."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"proxyWhitelist"})})," - Required. An array of IP addresses that Dashy will accept the header from. Only requests originating from these IPs will be trusted. This prevents clients from spoofing the header directly."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"users"})})," - Required. The header username is matched against this list to determine the user's role (",(0,t.jsx)(n.code,{children:"admin"})," or ",(0,t.jsx)(n.code,{children:"normal"}),") and to generate the session token. Users must be defined here even though authentication is handled externally."]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"how-it-works",children:"How it Works"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"User visits Dashy, which is behind a reverse proxy (e.g. Authelia)"}),"\n",(0,t.jsxs)(n.li,{children:["The proxy authenticates the user and forwards the request with a header like ",(0,t.jsx)(n.code,{children:"Remote-User: alice"})]}),"\n",(0,t.jsxs)(n.li,{children:["Dashy's server checks that the request comes from a whitelisted proxy IP, then returns the username via the ",(0,t.jsx)(n.code,{children:"/get-user"})," endpoint"]}),"\n",(0,t.jsx)(n.li,{children:"The client matches the username against the configured users, generates a session token, and sets the auth cookie"}),"\n",(0,t.jsxs)(n.li,{children:["From this point, standard Dashy auth applies - ",(0,t.jsx)(n.code,{children:"isLoggedIn()"}),", admin checks, and granular access controls all work as normal"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"notes",children:"Notes"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["The ",(0,t.jsx)(n.code,{children:"proxyWhitelist"})," checks ",(0,t.jsx)(n.code,{children:"req.socket.remoteAddress"}),", which is the direct connection source. If your proxy connects through Docker networking, use the container's internal IP (e.g. ",(0,t.jsx)(n.code,{children:"172.18.0.2"}),"), not the external IP"]}),"\n",(0,t.jsx)(n.li,{children:"Logout clears Dashy's session cookie, but the user remains authenticated at the proxy level. Revisiting the page will re-authenticate automatically"}),"\n",(0,t.jsxs)(n.li,{children:["When header auth is enabled, server-side API endpoints are also protected by the proxy whitelist. Requests not from a whitelisted IP will be rejected. Admin enforcement applies - only users with ",(0,t.jsx)(n.code,{children:"type: admin"})," can access write endpoints (config save)"]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"oidc",children:"OIDC"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy also supports using a general ",(0,t.jsx)(n.a,{href:"https://openid.net/connect/",children:"OIDC compatible"})," authentication server. In order to use it, the authentication section needs to be configured:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n disableConfigurationForNonAdmin: true # Prevent authenticated non-admins using editor\n auth:\n enableOidc: true\n oidc:\n clientId: 'registered-client-id'\n endpoint: 'https://your-oidc-provider.example.com'\n scope: 'openid profile email'\n adminGroup: admin\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Because Dashy is a SPA, a ",(0,t.jsx)(n.a,{href:"https://datatracker.ietf.org/doc/html/rfc6749#section-2.1",children:"public client"})," registration with PKCE is needed."]}),"\n",(0,t.jsxs)(n.p,{children:["If you set ",(0,t.jsx)(n.code,{children:"adminGroup"}),", include ",(0,t.jsx)(n.code,{children:"groups"})," in ",(0,t.jsx)(n.code,{children:"scope"})," (e.g. ",(0,t.jsx)(n.code,{children:"scope: 'openid profile email groups'"}),") so your IdP actually returns the claim in the id_token. Same goes for ",(0,t.jsx)(n.code,{children:"adminRole"})," and a ",(0,t.jsx)(n.code,{children:"roles"})," scope if your IdP needs one."]}),"\n",(0,t.jsxs)(n.p,{children:["Note, that if your ",(0,t.jsx)(n.code,{children:"clientId"})," is numeric, you must place it in quotes. Otherwise it will be interpreted as a number and truncated to 64 chars!"]}),"\n",(0,t.jsx)(n.p,{children:"An example for Authelia is shared below, but other OIDC systems can be used:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"identity_providers:\n oidc:\n clients:\n - client_id: dashy\n client_name: dashy\n public: true\n authorization_policy: 'one_factor'\n require_pkce: true\n pkce_challenge_method: 'S256'\n redirect_uris:\n - https://dashy.local # should point to your dashy endpoint\n grant_types:\n - authorization_code\n scopes:\n - 'openid'\n - 'profile'\n - 'roles'\n - 'email'\n - 'groups'\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Groups and roles will be populated and available for controlling display similar to ",(0,t.jsx)(n.a,{href:"#keycloak",children:"Keycloak"})," above."]}),"\n",(0,t.jsx)(n.h3,{id:"how-server-side-enforcement-works-1",children:"How server-side enforcement works"}),"\n",(0,t.jsxs)(n.p,{children:["Dashy's server reads ",(0,t.jsx)(n.code,{children:"auth.oidc"})," from ",(0,t.jsx)(n.code,{children:"conf.yml"})," at boot, lazily fetches the OIDC discovery doc + JWKS from your ",(0,t.jsx)(n.code,{children:"endpoint"}),", then verifies the ",(0,t.jsx)(n.code,{children:"id_token"})," the SPA attaches to every API call as ",(0,t.jsx)(n.code,{children:"Authorization: Bearer "}),". Tokens that fail signature / issuer / audience / expiry verification are rejected with ",(0,t.jsx)(n.code,{children:"401"}),". Write endpoints (",(0,t.jsx)(n.code,{children:"POST /config-manager/save"}),") additionally require the ",(0,t.jsx)(n.code,{children:"adminGroup"})," (or ",(0,t.jsx)(n.code,{children:"adminRole"}),") to be present in the token's ",(0,t.jsx)(n.code,{children:"groups"})," / ",(0,t.jsx)(n.code,{children:"roles"})," claims, and non-admins receive ",(0,t.jsx)(n.code,{children:"403"}),". Note that the ",(0,t.jsx)(n.code,{children:"/conf.yml"})," remains anonymously readable still"]}),"\n",(0,t.jsxs)(n.p,{children:["Your IdP must include ",(0,t.jsx)(n.code,{children:"groups"})," / ",(0,t.jsx)(n.code,{children:"roles"})," in the id_token, not only the access token, for the admin check to work (most IdPs do this when the ",(0,t.jsx)(n.code,{children:"groups"})," scope is requested)."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"authentik",children:"authentik"}),"\n",(0,t.jsxs)(n.p,{children:["This documentation is specific to ",(0,t.jsx)(n.code,{children:"authentik"}),", however it may be useful in getting other idP's working with ",(0,t.jsx)(n.code,{children:"Dashy"}),"."]}),"\n",(0,t.jsx)(n.p,{children:"This guide will only walk through the following:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Creating and configuring an OIDC provider"}),"\n",(0,t.jsx)(n.li,{children:"Creating and configuring an application"}),"\n",(0,t.jsx)(n.li,{children:"Assigning groups"}),"\n",(0,t.jsxs)(n.li,{children:["Configuring ",(0,t.jsx)(n.code,{children:"Dashy"})," to use the OIDC client"]}),"\n",(0,t.jsxs)(n.li,{children:["Show quick examples of how to hide/show ",(0,t.jsx)(n.code,{children:"pages"}),", ",(0,t.jsx)(n.code,{children:"items"}),", and ",(0,t.jsx)(n.code,{children:"sections"})," using OIDC groups"]}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"This guide assumes the following:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["You have a working instance of ",(0,t.jsx)(n.code,{children:"authentik"})," terminated with SSL"]}),"\n",(0,t.jsxs)(n.li,{children:["You have a working instance of ",(0,t.jsx)(n.code,{children:"Dashy"})," terminated with SSL"]}),"\n",(0,t.jsx)(n.li,{children:"Users and groups are provisioned"}),"\n",(0,t.jsxs)(n.li,{children:["You are familiar with how ",(0,t.jsx)(n.code,{children:"authentik"})," works in case you need to do further troubleshooting that is outside the scope of this guide."]}),"\n"]}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["It it recommended that you create groups specific for ",(0,t.jsx)(n.code,{children:"Dashy"}),". Groups will allow you to display content based on group membership as well as limiting user access to ",(0,t.jsx)(n.code,{children:"Dashy"}),". If you do not need this functionality, then you can forgo creating specific groups."]})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsx)(n.p,{children:"You can use the application wizard to create the provider and application at one time. This is the recommended route, but only the manual process will be outlined in this guide."})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/72e45162-6c86-4d6f-a1ae-724ac503c00c",alt:"image"})}),"\n",(0,t.jsx)(n.h4,{id:"1-create-an-oidc-provider",children:"1. Create an OIDC provider"}),"\n",(0,t.jsxs)(n.p,{children:["Login to the admin console for ",(0,t.jsx)(n.code,{children:"authentik"}),". Go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Providers"}),". Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/c1f7f45d-469c-4bf1-a825-34658341a83e",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["A dialog box will pop-up, select the ",(0,t.jsx)(n.code,{children:"OAuth2/OpenID Provider"}),". Click ",(0,t.jsx)(n.code,{children:"Next"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ea84fe57-b813-404d-8dad-5e221b440bdb",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["On the next page of the wizard, set the ",(0,t.jsx)(n.code,{children:"Name"}),", ",(0,t.jsx)(n.code,{children:"Authentication flow"}),", ",(0,t.jsx)(n.code,{children:"Authorization flow"}),", and ",(0,t.jsx)(n.code,{children:"Invalidation flow"}),". See example below. Using the ",(0,t.jsx)(n.code,{children:"default-provider-authorization-implicit-consent"})," authorization flow on internal services and ",(0,t.jsx)(n.code,{children:"default-provider-authorization-explicit-consent"})," on external services is a common practice. However, it is fully up to you on how you would like to configure this option. ",(0,t.jsx)(n.code,{children:"Implicit"})," will login directly without user consent, ",(0,t.jsx)(n.code,{children:"explicit"})," will ask if the user approves the service being logged into with their user credentials. For the invalidation flow (required on Authentik 2023.10 and later) the built-in ",(0,t.jsx)(n.code,{children:"default-provider-invalidation-flow"})," is fine."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/e600aeaf-08d1-49aa-b304-11e90e5c89cd",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Scroll down and configure the ",(0,t.jsx)(n.code,{children:"Protocol settings"}),". Set the ",(0,t.jsx)(n.code,{children:"Client type"})," to ",(0,t.jsx)(n.code,{children:"Public"}),". Add the ",(0,t.jsx)(n.code,{children:"Redirect URIs/Origins (RegEx)"}),". If the site is hosted at ",(0,t.jsx)(n.code,{children:"dashy.lan.domain.com"}),", then you would enter as the example below."]}),"\n",(0,t.jsx)(n.admonition,{type:"note",children:(0,t.jsxs)(n.p,{children:["If you have an internal and external domain for ",(0,t.jsx)(n.code,{children:"Dashy"}),", enter both URI's. Enter each URI on a new line."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/4a289d7e-d7b4-4ff6-af5d-3e5202fae84e",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Scroll down to set the ",(0,t.jsx)(n.code,{children:"Signing Key"}),". It is recommended to use the built in ",(0,t.jsx)(n.code,{children:"authentik Self-signed Certificate"})," here unless you have special needs for your own custom cert."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/386c0750-9d2b-4482-8938-8b301b489b38",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["If you plan to use ",(0,t.jsx)(n.code,{children:"adminGroup"})," in your Dashy config, you need a ",(0,t.jsx)(n.code,{children:"groups"})," scope mapping first. Authentik does not ship one by default. Open ",(0,t.jsx)(n.em,{children:"Customisation > Property Mappings"})," in a new tab, click ",(0,t.jsx)(n.em,{children:"Create > Scope Mapping"}),", set ",(0,t.jsx)(n.em,{children:"Name"})," to ",(0,t.jsx)(n.code,{children:"groups"}),", ",(0,t.jsx)(n.em,{children:"Scope name"})," to ",(0,t.jsx)(n.code,{children:"groups"}),", and ",(0,t.jsx)(n.em,{children:"Expression"})," to:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-python",children:'return {"groups": [g.name for g in request.user.ak_groups.all()]}\n'})}),"\n",(0,t.jsx)(n.p,{children:"Save it, then come back to the provider wizard."}),"\n",(0,t.jsxs)(n.p,{children:["Expand ",(0,t.jsx)(n.code,{children:"Advanced protocol settings"})," then verify the ",(0,t.jsx)(n.code,{children:"Scopes"})," are set to what is highlighted in ",(0,t.jsx)(n.code,{children:"white"})," below (including the ",(0,t.jsx)(n.code,{children:"groups"})," mapping you just created, if you want ",(0,t.jsx)(n.code,{children:"adminGroup"})," to work). Set the ",(0,t.jsx)(n.code,{children:"Subject mode"})," to ",(0,t.jsx)(n.code,{children:"Based on the Users's Email"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ae5e87b8-1ad6-41dd-b6e1-9665623f842a",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Lastly, toggle ",(0,t.jsx)(n.code,{children:"Include claims in id_token"})," to on. Click ",(0,t.jsx)(n.code,{children:"Finish"})," to complete creating the provider."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/25353b3c-3f54-47cf-bd47-b5023f86d7cf",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Grab the generated ",(0,t.jsx)(n.code,{children:"Client ID"})," and ",(0,t.jsx)(n.code,{children:"OpenID Configuration Issuer"})," URL by clicking the newly created provider as this will use this later when ",(0,t.jsx)(n.code,{children:"Dashy"})," is configured to use the OIDC auth mechanism. In this tutorial, what was generated is used below. Obviously adjust the ",(0,t.jsx)(n.code,{children:"Client ID"})," that was generated and use your domain here for the ",(0,t.jsx)(n.code,{children:"issuer"}),"."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"Client ID: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15\nOpenID Configuration Issuer: https://auth.domain.com/application/o/dashy/\n"})}),"\n",(0,t.jsx)(n.h4,{id:"2-create-an-application",children:"2. Create an application"}),"\n",(0,t.jsxs)(n.p,{children:["Make sure you are still in the ",(0,t.jsx)(n.code,{children:"authentik"})," admin console then go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Applications"}),". Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/fd225936-15a1-409f-83c8-e24a43047df0",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Next, it is required to give a user facing ",(0,t.jsx)(n.code,{children:"Name"}),", ",(0,t.jsx)(n.code,{children:"Slug"})," and assign the newly created provider. Use the example below if you have been following the guide. If you have used your own naming, then adjust accordingly. Click ",(0,t.jsx)(n.code,{children:"Create"})," once you are done."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/e6574d7d-6b22-4e7d-b388-45341b98746b",alt:"image"})}),"\n",(0,t.jsx)(n.admonition,{type:"tip",children:(0,t.jsxs)(n.p,{children:["Open the application in a new tab from the ",(0,t.jsx)(n.code,{children:"authentik"})," user portal and upload a custom icon. You can also enter a user facing ",(0,t.jsx)(n.code,{children:"Description"})," that the user would see."]})}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/20561387-549f-49de-98e6-30330dcdc734",alt:"image"})}),"\n",(0,t.jsxs)(n.h4,{id:"3-optional-limiting-access-via-authentik-with-groups",children:["3. ",(0,t.jsx)(n.em,{children:"(Optional)"})," Limiting access via ",(0,t.jsx)(n.code,{children:"authentik"})," with groups"]}),"\n",(0,t.jsxs)(n.p,{children:["If you would like to deny ",(0,t.jsx)(n.code,{children:"Dashy"})," access from specific users who are not within ",(0,t.jsx)(n.code,{children:"authentik"})," based groups, you bind them to the application you just created now. ",(0,t.jsx)(n.code,{children:"authentik"})," will deny access to those who are not members of this group or groups. If you want to allow everyone access from your ",(0,t.jsx)(n.code,{children:"authentik"})," instance, skip this step."]}),"\n",(0,t.jsxs)(n.p,{children:["Make sure you are still in the ",(0,t.jsx)(n.code,{children:"authentik"})," admin console then go to ",(0,t.jsx)(n.code,{children:"Applications"})," > ",(0,t.jsx)(n.code,{children:"Applications"}),". Click the newly created ",(0,t.jsx)(n.code,{children:"Dashy"})," application."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/613fafe7-881f-4664-a903-945854ac65e2",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Click the ",(0,t.jsx)(n.code,{children:"Policy/Group/User Bindings"})," tab at the top, then click ",(0,t.jsx)(n.code,{children:"Bind existing policy"}),". This assumes you have already created the groups you want to use for ",(0,t.jsx)(n.code,{children:"Dashy"})," and populated users in those groups."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/10fca15b-e77d-4624-ae03-0ece3910904c",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:["Click ",(0,t.jsx)(n.code,{children:"Group"})," for the binding type. Under ",(0,t.jsx)(n.code,{children:"Group"})," select the appropriate group you would like to bind. Make sure ",(0,t.jsx)(n.code,{children:"Enabled"})," is toggeled on. Click ",(0,t.jsx)(n.code,{children:"Create"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://github.com/user-attachments/assets/ebf680ab-696f-4c08-ae89-d73fe92b398f",alt:"image"})}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"Dashy"})," will now be scoped only to users within the assigned groups you have bound the application to. Keep adding groups if you would like to adjust the dashboard visibilty based on group membership."]}),"\n",(0,t.jsxs)(n.h4,{id:"4-configure-dashy-to-use-oidc-client",children:["4. Configure ",(0,t.jsx)(n.code,{children:"Dashy"})," to use OIDC client"]}),"\n",(0,t.jsx)(n.admonition,{type:"important",children:(0,t.jsxs)(n.p,{children:["It is highly recommended to edit your ",(0,t.jsx)(n.code,{children:"conf.yml"})," directly for this step."]})}),"\n",(0,t.jsx)(n.admonition,{type:"caution",children:(0,t.jsxs)(n.p,{children:["Do not make the same mistake many have made here by including the fully qualified address for the ",(0,t.jsx)(n.code,{children:"OpenID Configuration URL"}),". ",(0,t.jsx)(n.code,{children:"Dashy"})," will append the ",(0,t.jsx)(n.code,{children:".well-known"})," configuration automatically. If the ",(0,t.jsx)(n.code,{children:".well-known"})," URI is included the app will get redirect loops and ",(0,t.jsx)(n.code,{children:"400"})," errors."]})}),"\n",(0,t.jsxs)(n.p,{children:["Enter the ",(0,t.jsx)(n.code,{children:"Client ID"})," in the ",(0,t.jsx)(n.code,{children:"clientId"})," field and ",(0,t.jsx)(n.code,{children:"OpenID Configuration Issuer"})," in the ",(0,t.jsx)(n.code,{children:"endpoint"})," field."]}),"\n",(0,t.jsxs)(n.p,{children:["Below is how to configure the ",(0,t.jsx)(n.code,{children:"auth"})," section in the yaml syntax. Once this is enabled, when an attempt to access ",(0,t.jsx)(n.code,{children:"Dashy"})," is made it will now redirect you to the ",(0,t.jsx)(n.code,{children:"authentik"})," login page moving forward."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-yaml",children:"appConfig:\n theme: glass\n layout: auto\n iconSize: medium\n disableConfigurationForNonAdmin: true # Prevent logged-in, non-admins using the view/edit config features\n auth:\n enableOidc: true\n oidc:\n clientId: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15\n endpoint: https://auth.domain.com/application/o/dashy/\n"})}),"\n",(0,t.jsxs)(n.h4,{id:"5-optional-example-snippets-for-dashboard-visibility",children:["5. ",(0,t.jsx)(n.em,{children:"(OPTIONAL)"})," Example snippets for dashboard visibility"]}),"\n",(0,t.jsxs)(n.p,{children:["Using the ",(0,t.jsx)(n.code,{children:"hideForKeycloakUsers"})," configuration option is needed to use the ",(0,t.jsx)(n.code,{children:"authentik"})," groups that were created previously."]}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"pages"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"pages:\n - name: App Management\n path: appmgmt.yml\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n - name: Network Management\n path: network.yml\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"items"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:" items:\n - title: Authentik Admin\n icon: authentik.svg\n url: https://auth.domain.com/if/admin/\n target: newtab\n id: 0_1472_authentikadmin\n displayData:\n hideForKeycloakUsers:\n groups:\n - Dashy Users\n - title: Authentik User\n icon: authentik-light.png\n url: https://auth.domain.com/if/user/\n target: newtab\n id: 1_1472_authentikuser\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Adjusting ",(0,t.jsx)(n.code,{children:"sections"})," visibility:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{children:"sections:\n - name: Authentication\n displayData:\n sortBy: default\n cols: 1\n collapsed: false\n hideForGuests: false\n hideForKeycloakUsers:\n groups: \n - Dashy Users\n"})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"alternative-authentication-methods",children:"Alternative Authentication Methods"}),"\n",(0,t.jsx)(n.p,{children:"These are alternatives to Dashy's built-in auth, Keycloak, and OIDC. Most of them sit in front of Dashy at the network or reverse proxy level, which is generally the better approach for anything internet-facing."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#reverse-proxy-auth",children:"Reverse Proxy Auth"})," - Authelia, Authentik, or similar sitting in front of Dashy"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#zero-trust-tunnels",children:"Zero-Trust Tunnels"})," - Cloudflare Tunnel, Tailscale Funnel"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#vpn",children:"VPN"})," - Keep Dashy off the internet entirely"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#ip-based-access",children:"IP-Based Access"})," - Restrict by source IP in your web server"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#web-server-authentication",children:"Web Server Authentication"})," - HTTP basic auth at the proxy level"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#sso--oauth-providers",children:"SSO / OAuth Providers"})," - Cloud-hosted identity providers"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#cloud-hosting-providers",children:"Cloud Hosting Providers"})," - Built-in auth on hosting platforms"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"reverse-proxy-auth",children:"Reverse proxy auth"}),"\n",(0,t.jsx)(n.p,{children:"The most common setup for self-hosters running multiple services. You put an auth server in front of your reverse proxy, and it handles login, 2FA, and sessions for everything behind it. You configure it once, and all your apps get protected."}),"\n",(0,t.jsxs)(n.p,{children:["Dashy has ",(0,t.jsx)(n.a,{href:"#header-authentication",children:"Header Authentication"})," support, so when your proxy authenticates a user and forwards their identity via a header, Dashy picks up the username and maps it to a configured user automatically. No separate Dashy login needed."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Authelia"})," is lightweight and Docker-friendly. It supports 2FA, per-path access rules, and multiple user backends. To get started quickly:"]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"git clone https://github.com/authelia/authelia.git"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"cd authelia/examples/compose/lite"})}),"\n",(0,t.jsxs)(n.li,{children:["Edit ",(0,t.jsx)(n.code,{children:"users_database.yml"}),", ",(0,t.jsx)(n.code,{children:"configuration.yml"}),", and ",(0,t.jsx)(n.code,{children:"docker-compose.yml"})," for your domain and users"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.code,{children:"docker compose up -d"})}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["See the ",(0,t.jsx)(n.a,{href:"https://www.authelia.com/docs/",children:"Authelia docs"})," for the full setup guide."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Authentik"})," is heavier but gives you a proper admin UI, built-in OIDC/SAML support, and user self-service (password resets, enrollment flows, etc). Good if you want a single identity provider across many apps. See the ",(0,t.jsx)(n.a,{href:"https://docs.goauthentik.io/docs/installation/docker-compose",children:"authentik Docker Compose install"})," to get started, and the ",(0,t.jsx)(n.a,{href:"#authentik",children:"authentik section"})," above for Dashy-specific OIDC config."]}),"\n",(0,t.jsx)(n.h3,{id:"zero-trust-tunnels",children:"Zero-trust tunnels"}),"\n",(0,t.jsx)(n.p,{children:"These let you expose Dashy to the internet without opening inbound ports or configuring port forwarding. Auth is handled by the tunnel provider before traffic ever reaches your server."}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Cloudflare Tunnel"})," connects Dashy to Cloudflare's edge network via an outbound-only ",(0,t.jsx)(n.code,{children:"cloudflared"})," daemon (runs nicely as a Docker sidecar). Cloudflare handles DNS, TLS, and DDoS protection. Pair it with Cloudflare Access to require identity provider login before anyone reaches Dashy. The free tier covers most home setups. See the ",(0,t.jsx)(n.a,{href:"https://developers.cloudflare.com/cloudflare-one/connections/connect-networks/",children:"Cloudflare Tunnel docs"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.strong,{children:"Tailscale Funnel"})," exposes Dashy through your Tailscale mesh to the public internet, with automatic TLS. Simpler to set up than Cloudflare but you get less control over access policies. See the ",(0,t.jsx)(n.a,{href:"https://tailscale.com/kb/1223/funnel",children:"Funnel docs"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"vpn",children:"VPN"}),"\n",(0,t.jsx)(n.p,{children:"A VPN keeps Dashy off the public internet entirely. You connect to your home network remotely and access Dashy like you're on the LAN. No auth to configure, no attack surface to worry about. The downside: you need the VPN running to see anything, and some networks (corporate WiFi, hotels) block VPN traffic."}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://www.wireguard.com/",children:"WireGuard"})," is fast and minimal. Most self-hosters run it through a UI like ",(0,t.jsx)(n.a,{href:"https://github.com/wg-easy/wg-easy",children:"wg-easy"}),", which gives you a web interface for managing peers and generating QR codes for mobile."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://tailscale.com/",children:"Tailscale"})," wraps WireGuard and takes care of NAT traversal, key exchange, and device management. No port forwarding needed, works across networks with zero config. There's a generous free tier. ",(0,t.jsx)(n.a,{href:"https://github.com/juanfont/headscale",children:"Headscale"})," is a self-hosted coordination server if you want to keep everything on your own infrastructure."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://openvpn.net/",children:"OpenVPN"})," still works fine if you already have it running, but for a new setup WireGuard or Tailscale are easier to get going."]}),"\n",(0,t.jsx)(n.h3,{id:"ip-based-access",children:"IP-based access"}),"\n",(0,t.jsx)(n.p,{children:"If you have a static IP or are already on a VPN, you can restrict access to Dashy by source IP at the web server level. This works well as an extra layer on top of other auth methods."}),"\n",(0,t.jsx)(n.p,{children:"NGINX:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"location / {\n proxy_pass http://dashy:8080;\n allow 192.168.1.0/24;\n allow 203.0.113.50;\n deny all;\n}\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Caddy (",(0,t.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/matchers",children:"request matchers docs"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'dashy.example.com {\n @blocked not remote_ip 192.168.1.0/24 203.0.113.50\n respond @blocked "Access denied" 403\n reverse_proxy dashy:8080\n}\n'})}),"\n",(0,t.jsx)(n.p,{children:"Apache (2.4+):"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"\n Require ip 192.168.1.0/24\n Require ip 203.0.113.50\n\n"})}),"\n",(0,t.jsx)(n.h3,{id:"web-server-authentication",children:"Web server authentication"}),"\n",(0,t.jsx)(n.p,{children:"Your reverse proxy can handle HTTP basic auth directly, no extra services needed. This gives you a browser login prompt in front of Dashy. Make sure you're using HTTPS, as basic auth sends credentials base64-encoded (not encrypted) with every request."}),"\n",(0,t.jsxs)(n.p,{children:["NGINX (",(0,t.jsx)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_auth_basic_module.html",children:"auth module docs"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'location / {\n auth_basic "Dashy";\n auth_basic_user_file /etc/nginx/conf.d/.htpasswd;\n proxy_pass http://dashy:8080;\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Generate the password file with ",(0,t.jsx)(n.code,{children:"htpasswd -c /etc/nginx/conf.d/.htpasswd alicia"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["Caddy (",(0,t.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/basicauth",children:"basicauth directive"}),"):"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"dashy.example.com {\n basicauth {\n alicia $2a$14$... # generate with: caddy hash-password\n }\n reverse_proxy dashy:8080\n}\n"})}),"\n",(0,t.jsx)(n.p,{children:"Apache:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:'AuthType Basic\nAuthName "Dashy"\nAuthUserFile /path/to/.htpasswd\nRequire valid-user\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Generate the password file with ",(0,t.jsx)(n.code,{children:"htpasswd -c /path/to/.htpasswd alicia"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"sso--oauth-providers",children:"SSO / OAuth providers"}),"\n",(0,t.jsxs)(n.p,{children:["Cloud identity providers like ",(0,t.jsx)(n.a,{href:"https://auth0.com/",children:"Auth0"}),", ",(0,t.jsx)(n.a,{href:"https://developer.okta.com/",children:"Okta"}),", ",(0,t.jsx)(n.a,{href:"https://www.ory.sh/",children:"Ory"}),", and ",(0,t.jsx)(n.a,{href:"https://cloud.google.com/identity",children:"Google Cloud Identity"})," can work with Dashy through its ",(0,t.jsx)(n.a,{href:"#oidc",children:"OIDC support"}),". If your provider speaks OIDC (most do), just configure it as described in the OIDC section and you're set."]}),"\n",(0,t.jsx)(n.p,{children:"For providers that only support OAuth2 or SAML without an OIDC layer, you'll need something in between to translate. Authentik, Keycloak, and Authelia can all bridge from SAML/OAuth2 to OIDC."}),"\n",(0,t.jsx)(n.h3,{id:"cloud-hosting-providers",children:"Cloud hosting providers"}),"\n",(0,t.jsxs)(n.p,{children:["If you're running Dashy on a cloud platform, most have their own auth options you can enable without touching Dashy's config. See your provider's docs: ",(0,t.jsx)(n.a,{href:"https://www.cloudflare.com/teams/access/",children:"Cloudflare Access"}),", ",(0,t.jsx)(n.a,{href:"https://docs.netlify.com/visitor-access/password-protection/",children:"Netlify Password Protection"}),", ",(0,t.jsx)(n.a,{href:"https://aws.amazon.com/cognito/",children:"AWS Cognito"}),", ",(0,t.jsx)(n.a,{href:"https://learn.microsoft.com/en-us/azure/app-service/overview-authentication-authorization",children:"Azure App Service Authentication"}),", and ",(0,t.jsx)(n.a,{href:"https://vercel.com/docs/security/password-protection",children:"Vercel Password Protection"}),"."]}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"#",children:"\u2b06\ufe0f Back to Top"})})})})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,n,s){s.d(n,{R:()=>r,x:()=>a});var i=s(6540);const t={},o=i.createContext(t);function r(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/68d50eef.2a1a5db7.js b/assets/js/68d50eef.3438a0b3.js similarity index 99% rename from assets/js/68d50eef.2a1a5db7.js rename to assets/js/68d50eef.3438a0b3.js index 57acb743..7eb8b1d1 100644 --- a/assets/js/68d50eef.2a1a5db7.js +++ b/assets/js/68d50eef.3438a0b3.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5992],{2745(e,n,i){i.r(n),i.d(n,{assets:()=>a,contentTitle:()=>l,default:()=>h,frontMatter:()=>r,metadata:()=>s,toc:()=>d});const s=JSON.parse('{"id":"developing","title":"Developing","description":"This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture.","source":"@site/docs/developing.md","sourceDirName":".","slug":"/developing","permalink":"/docs/developing","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/developing.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Contributing","permalink":"/docs/contributing"},"next":{"title":"Development Guides","permalink":"/docs/development-guides"}}');var t=i(4848),o=i(8453);const r={},l="Developing",a={},d=[{value:"Setting up the Dev Environment",id:"setting-up-the-dev-environment",level:2},{value:"Prerequisites",id:"prerequisites",level:3},{value:"Running the Project",id:"running-the-project",level:3},{value:"Project Commands",id:"project-commands",level:3},{value:"Basics",id:"basics",level:4},{value:"Development",id:"development",level:4},{value:"Utils and Checks",id:"utils-and-checks",level:4},{value:"Alternate Start Commands",id:"alternate-start-commands",level:4},{value:"Notes",id:"notes",level:4},{value:"Environmental Variables",id:"environmental-variables",level:3},{value:"Environment Modes",id:"environment-modes",level:3},{value:"Git Strategy",id:"git-strategy",level:2},{value:"Git Flow",id:"git-flow",level:3},{value:"Git Branch Naming",id:"git-branch-naming",level:3},{value:"Commit Emojis",id:"commit-emojis",level:3},{value:"PR Guidelines",id:"pr-guidelines",level:3},{value:"Resources for Beginners",id:"resources-for-beginners",level:2},{value:"App Info",id:"app-info",level:2},{value:"Style Guide",id:"style-guide",level:3},{value:"Application Structure",id:"application-structure",level:3},{value:"Files in the Root: ./",id:"files-in-the-root-",level:4},{value:"Frontend Source: ./src/",id:"frontend-source-src",level:4},{value:"Visualisation of Source Directory",id:"visualisation-of-source-directory",level:4},{value:"Development Tools",id:"development-tools",level:2},{value:"Performance - Lighthouse",id:"performance---lighthouse",level:3},{value:"Dependencies - BundlePhobia",id:"dependencies---bundlephobia",level:3},{value:"Notes",id:"notes-1",level:2},{value:"Known Warnings",id:"known-warnings",level:3}];function c(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"developing",children:"Developing"})}),"\n",(0,t.jsxs)(n.p,{children:["This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture.\nIf you're adding new features, you may want to check out the ",(0,t.jsx)(n.a,{href:"/docs/development-guides",children:"Development Guides"})," docs, for tutorials covering basic tasks."]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#setting-up-the-dev-environment",children:"Setting up the Development Environment"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#prerequisites",children:"Prerequisites"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#running-the-project",children:"Running the App"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#project-commands",children:"Project Commands"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#environmental-variables",children:"Environmental Variables"})}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#git-strategy",children:"Git Strategy"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#git-flow",children:"Flow"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#git-branch-naming",children:"Branches"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#commit-emojis",children:"Commit emojis"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#pr-guidelines",children:"PR Guidelines"})}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#resources-for-beginners",children:"Resources for Beginners"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#app-info",children:"App Info"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#style-guide",children:"Code Style Guide"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#application-structure",children:"Application Structure"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#development-tools",children:"Development Tools"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#notes",children:"Misc / Notes"})}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"setting-up-the-dev-environment",children:"Setting up the Dev Environment"}),"\n",(0,t.jsx)(n.h3,{id:"prerequisites",children:"Prerequisites"}),"\n",(0,t.jsxs)(n.p,{children:["You will need either the latest or LTS version of ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://nodejs.org/",children:"Node.js"})})," to build and serve the application and ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://git-scm.com/downloads",children:"Git"})})," to easily fetch the code, and push any changes. If you plan on running or deploying the container, you'll also need ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://docs.docker.com/get-docker/",children:"Docker"})}),". To avoid any unexpected issues, ensure you've got at least ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://www.npmjs.com/get-npm",children:"NPM"})})," V 7.5 or ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://classic.yarnpkg.com/en/docs/install/#windows-stable",children:"Yarn"})})," 1.22 (you may find ",(0,t.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm",children:"NVM"})," helpful for switching/ managing versions)."]}),"\n",(0,t.jsx)(n.h3,{id:"running-the-project",children:"Running the Project"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Get Code: ",(0,t.jsx)(n.code,{children:"git clone https://github.com/Lissy93/dashy.git"})]}),"\n",(0,t.jsxs)(n.li,{children:["Navigate into the directory: ",(0,t.jsx)(n.code,{children:"cd dashy"})]}),"\n",(0,t.jsxs)(n.li,{children:["Install dependencies: ",(0,t.jsx)(n.code,{children:"yarn"})]}),"\n",(0,t.jsxs)(n.li,{children:["Start dev server: ",(0,t.jsx)(n.code,{children:"yarn dev"})]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["Dashy should now be being served on ",(0,t.jsx)(n.a,{href:"http://localhost:8080/",children:"http://localhost:8080/"}),". Hot reload is enabled, so making changes to any of the files will trigger them to be rebuilt and the page refreshed."]}),"\n",(0,t.jsx)(n.h3,{id:"project-commands",children:"Project Commands"}),"\n",(0,t.jsx)(n.h4,{id:"basics",children:"Basics"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn build"})})," - Builds the production bundle into ",(0,t.jsx)(n.code,{children:"./dist"}),". The Vite dev/runtime server serves ",(0,t.jsx)(n.code,{children:"user-data/conf.yml"})," on each request, so a rebuild is only needed when source code or assets change, not when config changes"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn start"})})," - Starts the Node server, which serves the built site from ",(0,t.jsx)(n.code,{children:"./dist"})," and the live config from ",(0,t.jsx)(n.code,{children:"user-data/"}),". Run ",(0,t.jsx)(n.code,{children:"yarn build"})," first"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"development",children:"Development"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn dev"})})," - Starts the development server with hot reloading"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn lint"})})," - Lints code to ensure it follows a consistent, neat style"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn test"})})," - Runs tests, and outputs results"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"utils-and-checks",children:"Utils and Checks"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn validate-config"})})," - If you have quite a long configuration file, you may wish to check that it's all good to go, before deploying the app. This can be done with ",(0,t.jsx)(n.code,{children:"yarn validate-config"})," or ",(0,t.jsx)(n.code,{children:"docker exec -it [container-id] yarn validate-config"}),". Your config file needs to be in ",(0,t.jsx)(n.code,{children:"/user-data/conf.yml"})," (or within your Docker container at ",(0,t.jsx)(n.code,{children:"/app/user-data/conf.yml"}),"). This will first check that your YAML is valid, and then validates it against Dashy's ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"schema"}),"."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn health-check"})})," - Checks that the application is up and running on it's specified port, and outputs current status and response times. Useful for integrating into your monitoring service, if you need to maintain high system availability"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"alternate-start-commands",children:"Alternate Start Commands"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn build && yarn start"})})," - Builds the app, then starts the production Node server. Use this for a manual production-style run on bare metal. With Vite, ",(0,t.jsx)(n.code,{children:"conf.yml"})," is served from ",(0,t.jsx)(n.code,{children:"user-data/"})," at runtime, so config changes only require a page refresh"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn pm2-start"})})," - Starts the Node server using ",(0,t.jsx)(n.a,{href:"https://pm2.keymetrics.io/",children:"PM2"}),", a process manager for Node.js applications, that helps them stay alive. PM2 has some built-in basic monitoring features, and an optional ",(0,t.jsx)(n.a,{href:"https://pm2.io/",children:"management solution"}),". If you are running the app on bare metal, it is recommended to use this start command"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"notes",children:"Notes"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["If you are using NPM, replace ",(0,t.jsx)(n.code,{children:"yarn"})," with ",(0,t.jsx)(n.code,{children:"npm run"})]}),"\n",(0,t.jsxs)(n.li,{children:["If you are using Docker, precede each command with ",(0,t.jsx)(n.code,{children:"docker exec -it [container-id]"}),". Container ID can be found by running ",(0,t.jsx)(n.code,{children:"docker ps"})]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"environmental-variables",children:"Environmental Variables"}),"\n",(0,t.jsxs)(n.p,{children:["All environmental variables are optional. Currently there are not many environmental variables used, as most of the user preferences are stored under ",(0,t.jsx)(n.code,{children:"appConfig"})," in the ",(0,t.jsx)(n.code,{children:"conf.yml"})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["You can set variables either in your environment, or using the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.env",children:(0,t.jsx)(n.code,{children:".env"})})," file."]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"NODE_ENV"})," - Current environment, can be either development, production or test"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"PORT"})," - The port to expose the running application on"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"HOST"})," - The host that Dashy is running on, domain or IP"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"BASE_URL"})," - The default base path for serving up static assets"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"VITE_APP_DOMAIN"})," - Usually the same as BASE_URL, but accessible in frontend"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"INTEGRITY"})," - Should enable SRI for build script and link resources"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"IS_DOCKER"})," - Computed automatically on build. Indicates if running in container"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"VITE_APP_VERSION"})," - Again, set automatically using package.json during build time"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"BACKUP_DIR"})," - Directory for conf.yml backups"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS"})," - Set to 'true' to skip the pre-save backup step (useful on read-only filesystems or where permissions don't allow it)"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"environment-modes",children:"Environment Modes"}),"\n",(0,t.jsxs)(n.p,{children:["You can set the environment using the ",(0,t.jsx)(n.code,{children:"NODE_ENV"})," variable. By default, the correct environment should be selected based on the script you run to start the app. The following environments are supported: ",(0,t.jsx)(n.code,{children:"production"}),", ",(0,t.jsx)(n.code,{children:"development"})," and ",(0,t.jsx)(n.code,{children:"test"}),". For more info, see ",(0,t.jsx)(n.a,{href:"https://cli.vuejs.org/guide/mode-and-env.html#modes",children:"Vue CLI Environment Modes"}),"."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"git-strategy",children:"Git Strategy"}),"\n",(0,t.jsx)(n.h3,{id:"git-flow",children:"Git Flow"}),"\n",(0,t.jsxs)(n.p,{children:["Like most Git repos, we are following the ",(0,t.jsx)(n.a,{href:"https://guides.github.com/introduction/flow",children:"Github Flow"})," standard."]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Create a branch (or fork if you don'd have write access)"}),"\n",(0,t.jsx)(n.li,{children:"Code some awesome stuff \ud83e\uddd1\u200d\ud83d\udcbb"}),"\n",(0,t.jsx)(n.li,{children:"Add, commit and push your changes to your branch/ fork"}),"\n",(0,t.jsx)(n.li,{children:"Head over to GitHub and create a Pull Request"}),"\n",(0,t.jsx)(n.li,{children:"Fill in the required sections in the template, and hit submit"}),"\n",(0,t.jsx)(n.li,{children:"Follow up with any reviews on your code"}),"\n",(0,t.jsx)(n.li,{children:"Merge \ud83c\udf89"}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"git-branch-naming",children:"Git Branch Naming"}),"\n",(0,t.jsxs)(n.p,{children:["The format of your branch name should be something similar to: ",(0,t.jsx)(n.code,{children:"[TYPE]/[TICKET]_[TITLE]"}),"\nFor example, ",(0,t.jsx)(n.code,{children:"FEATURE/420_Awesome-feature"})," or ",(0,t.jsx)(n.code,{children:"FIX/690_login-server-error"})]}),"\n",(0,t.jsx)(n.h3,{id:"commit-emojis",children:"Commit Emojis"}),"\n",(0,t.jsx)(n.p,{children:"Using a single emoji at the start of each commit message, to indicate the type task, makes the commit ledger easier to understand, plus it looks cool."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["\ud83c\udfa8 ",(0,t.jsx)(n.code,{children:":art:"})," - Improve structure / format of the code."]}),"\n",(0,t.jsxs)(n.li,{children:["\u26a1\ufe0f ",(0,t.jsx)(n.code,{children:":zap:"})," - Improve performance."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd25 ",(0,t.jsx)(n.code,{children:":fire:"})," - Remove code or files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc1b ",(0,t.jsx)(n.code,{children:":bug:"})," - Fix a bug."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\ude91\ufe0f ",(0,t.jsx)(n.code,{children:":ambulance:"})," - Critical hotfix"]}),"\n",(0,t.jsxs)(n.li,{children:["\u2728 ",(0,t.jsx)(n.code,{children:":sparkles:"})," - Introduce new features."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udcdd ",(0,t.jsx)(n.code,{children:":memo:"})," - Add or update documentation."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\ude80 ",(0,t.jsx)(n.code,{children:":rocket:"})," - Deploy stuff."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc84 ",(0,t.jsx)(n.code,{children:":lipstick:"})," - Add or update the UI and style files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf89 ",(0,t.jsx)(n.code,{children:":tada:"})," - Begin a project."]}),"\n",(0,t.jsxs)(n.li,{children:["\u2705 ",(0,t.jsx)(n.code,{children:":white_check_mark:"})," - Add, update, or pass tests."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd12\ufe0f ",(0,t.jsx)(n.code,{children:":lock:"})," - Fix security issues."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd16 ",(0,t.jsx)(n.code,{children:":bookmark:"})," - Make a Release or Version tag."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udea8 ",(0,t.jsx)(n.code,{children:":rotating_light:"})," - Fix compiler / linter warnings."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udea7 ",(0,t.jsx)(n.code,{children:":construction:"})," - Work in progress."]}),"\n",(0,t.jsxs)(n.li,{children:["\u2b06\ufe0f ",(0,t.jsx)(n.code,{children:":arrow_up:"})," - Upgrade dependencies."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc77 ",(0,t.jsx)(n.code,{children:":construction_worker:"})," - Add or update CI build system."]}),"\n",(0,t.jsxs)(n.li,{children:["\u267b\ufe0f ",(0,t.jsx)(n.code,{children:":recycle:"})," - Refactor code."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83e\ude79 ",(0,t.jsx)(n.code,{children:":adhesive_bandage:"})," - Simple fix for a non-critical issue."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd27 ",(0,t.jsx)(n.code,{children:":wrench:"})," - Add or update configuration files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf71 ",(0,t.jsx)(n.code,{children:":bento:"})," - Add or update assets."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\uddc3\ufe0f ",(0,t.jsx)(n.code,{children:":card_file_box:"})," - Perform database schema related changes."]}),"\n",(0,t.jsxs)(n.li,{children:["\u270f\ufe0f ",(0,t.jsx)(n.code,{children:":pencil2:"})," - Fix typos."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf10 ",(0,t.jsx)(n.code,{children:":globe_with_meridians:"})," - Internationalization and translations."]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["For a full list of options, see ",(0,t.jsx)(n.a,{href:"https://gitmoji.dev/",children:"gitmoji.dev"})]}),"\n",(0,t.jsx)(n.h3,{id:"pr-guidelines",children:"PR Guidelines"}),"\n",(0,t.jsx)(n.p,{children:"Once you've made your changes, and pushed them to your fork or branch, you're ready to open a pull request!"}),"\n",(0,t.jsx)(n.p,{children:"For a pull request to be merged, it must:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Must be backwards compatible"}),"\n",(0,t.jsx)(n.li,{children:"The build, lint and tests (run by GH actions) must pass"}),"\n",(0,t.jsx)(n.li,{children:"There must not be any merge conflicts"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"When you submit your PR, include the required info, by filling out the PR template. Including:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"A brief description of your changes"}),"\n",(0,t.jsx)(n.li,{children:"The issue, ticket or discussion number (if applicable)"}),"\n",(0,t.jsx)(n.li,{children:"For UI relate updates include a screenshot"}),"\n",(0,t.jsx)(n.li,{children:"If any dependencies were added, explain why it was needed, state the cost associated, and confirm it does not introduce any security issues"}),"\n",(0,t.jsx)(n.li,{children:"Finally, check the checkboxes, to confirm that the standards are met, and hit submit!"}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"resources-for-beginners",children:"Resources for Beginners"}),"\n",(0,t.jsx)(n.p,{children:"New to Web Development? Glad you're here! Dashy is a pretty simple app, so it should make a good candidate for your first PR. Presuming that you already have a basic knowledge of JavaScript, the following articles should point you in the right direction for getting up to speed with the technologies used in this project:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://opensource.guide/how-to-contribute/",children:"Open Source for Beginners"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://v3.vuejs.org/guide/introduction.html",children:"Introduction to Vue.js"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.taniarascia.com/getting-started-with-vue/",children:"Vue.js Walkthrough"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://github.com/lukehoban/es6features",children:"ES6 Features"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://blog.logrocket.com/the-definitive-guide-to-scss/",children:"Definitive guide to SCSS"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://docker-curriculum.com/",children:"Complete beginners guide to Docker"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://training.play-with-docker.com/",children:"Docker Classroom - Interactive Tutorials"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.freecodecamp.org/news/learn-typescript-in-5-minutes-13eda868daeb/",children:"Quick start TypeScript guide"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.typescripttutorial.net/",children:"Complete TypeScript tutorial series"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://blog.logrocket.com/vue-typescript-tutorial-examples/",children:"Using TypeScript with Vue.js"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"http://git-cheatsheet.com/",children:"Git cheat sheet"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.freecodecamp.org/news/what-is-npm-a-node-package-manager-tutorial-for-beginners/",children:"Basics of using NPM"})}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["As well as Node, Git and Docker- you'll also need an IDE (e.g. ",(0,t.jsx)(n.a,{href:"https://code.visualstudio.com/",children:"VS Code"})," or ",(0,t.jsx)(n.a,{href:"https://www.vim.org/",children:"Vim"}),") and a terminal (Windows users may find ",(0,t.jsx)(n.a,{href:"https://docs.microsoft.com/en-us/windows/wsl/",children:"WSL"})," more convenient)."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"app-info",children:"App Info"}),"\n",(0,t.jsx)(n.h3,{id:"style-guide",children:"Style Guide"}),"\n",(0,t.jsxs)(n.p,{children:["Linting is done using ",(0,t.jsx)(n.a,{href:"https://eslint.org/",children:"ESLint"}),", and using the ",(0,t.jsx)(n.a,{href:"https://github.com/vuejs/eslint-config-standard",children:"Vue.js Styleguide"}),", which is very similar to the ",(0,t.jsx)(n.a,{href:"https://github.com/airbnb/javascript",children:"AirBnB Styleguide"}),". You can run ",(0,t.jsx)(n.code,{children:"yarn lint"})," to report and fix issues. While the dev server is running, issues will be reported to the console automatically, and any lint errors will trigger the build to fail. Note that all lint checks must pass before any PR can be merged. Linting is also run as a git pre-commit hook"]}),"\n",(0,t.jsx)(n.p,{children:"The most significant things to note are:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Indentation should be done with two spaces"}),"\n",(0,t.jsx)(n.li,{children:"Strings should use single quotes"}),"\n",(0,t.jsx)(n.li,{children:"All statements must end in a semi-colon"}),"\n",(0,t.jsx)(n.li,{children:"The final element in all objects must be preceded with a comma"}),"\n",(0,t.jsx)(n.li,{children:"Maximum line length is 100"}),"\n",(0,t.jsx)(n.li,{children:"There must be exactly one blank line between sections, before function names, and at the end of the file"}),"\n",(0,t.jsx)(n.li,{children:"With conditionals, put else on the same line as your if block's closing brace"}),"\n",(0,t.jsx)(n.li,{children:"All multiline blocks must use braces"}),"\n",(0,t.jsx)(n.li,{children:"Avoid console statements in the frontend"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"Styleguides:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["Vue: ",(0,t.jsx)(n.a,{href:"https://vuejs.org/v2/style-guide/",children:"Vue styleguide"})]}),"\n",(0,t.jsxs)(n.li,{children:["JavaScript: ",(0,t.jsx)(n.a,{href:"https://github.com/airbnb/javascript",children:"github.com/airbnb/javascript"})]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h3,{id:"application-structure",children:"Application Structure"}),"\n",(0,t.jsxs)(n.h4,{id:"files-in-the-root-",children:["Files in the Root: ",(0,t.jsx)(n.code,{children:"./"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"\u256e\n\u251c\u2500\u2500 package.json # Project meta-data, dependencies and paths to scripts\n\u251c\u2500\u2500 src/ # Project front-end source code\n\u251c\u2500\u2500 server.js # A Node.js server to serve up the /dist directory\n\u251c\u2500\u2500 services/ # All server-side endpoints and utilities\n\u251c\u2500\u2500 vue.config.js # Vue.js configuration\n\u251c\u2500\u2500 Dockerfile # The blueprint for building the Docker container\n\u251c\u2500\u2500 docker-compose.yml # A Docker run command\n\u251c\u2500\u2500 .env # Location for any environmental variables\n\u251c\u2500\u2500 yarn.lock # Auto-generated list of current packages and version numbers\n\u251c\u2500\u2500 docs/ # Markdown documentation\n\u251c\u2500\u2500 README.md # Readme, basic info for getting started\n\u251c\u2500\u2500 LICENSE.md # License for use\n\u256f\n"})}),"\n",(0,t.jsxs)(n.h4,{id:"frontend-source-src",children:["Frontend Source: ",(0,t.jsx)(n.code,{children:"./src/"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"./src\n\u251c\u2500\u2500 App.vue # Vue.js starting file\n\u251c\u2500\u2500 assets # Static non-compiled assets\n\u2502 \u251c\u2500\u2500 fonts # .ttf font files\n\u2502 \u251c\u2500\u2500 locales # All app text, each language in a separate JSON file\n\u2502 \u2570\u2500\u2500 interface-icons # SVG icons used in the app\n\u251c\u2500\u2500 components # All front-end Vue web components\n\u2502 \u251c\u2500\u2500 Charts # Charting components for dynamically displaying widget data\n\u2502 \u2502 \u251c\u2500\u2500 Gauge.vue # A speed-dial style chart for showing 0 - 100 values\n\u2502 \u2502 \u2570\u2500\u2500 PercentageChart.vue # A horizontal bar for showing percentage breakdowns\n\u2502 \u251c\u2500\u2500 Configuration # Components relating to the user config pop-up\n\u2502 \u2502 \u251c\u2500\u2500 AppInfoModal.vue # A modal showing core app info, like version, language, etc\n\u2502 \u2502 \u251c\u2500\u2500 AppVersion.vue # Shows current version from package.json, compares with GitHub\n\u2502 \u2502 \u251c\u2500\u2500 CloudBackupRestore.vue # Form where the user manages cloud sync options\n\u2502 \u2502 \u251c\u2500\u2500 ConfigContainer.vue # Main container, wrapping all other config components\n\u2502 \u2502 \u251c\u2500\u2500 CustomCss.vue # Form where the user can input custom CSS\n\u2502 \u2502 \u2570\u2500\u2500 JsonEditor.vue # JSON editor, where the user can modify the main config file\n\u2502 \u251c\u2500\u2500 FormElements # Basic form elements used throughout the app\n\u2502 \u2502 \u251c\u2500\u2500 Button.vue # Standard button component\n\u2502 \u2502 \u251c\u2500\u2500 Radio.vue # Standard radio button input\n\u2502 \u2502 \u251c\u2500\u2500 Select.vue # Standard dropdown input selector\n\u2502 \u2502 \u251c\u2500\u2500 Input.vue # Standard text field input component\n\u2502 \u2502 \u2570\u2500\u2500 Toggle.vue # Standard on / off toggle switch\n\u2502 \u251c\u2500\u2500 InteractiveEditor # Components for the interactive UI config editor\n\u2502 \u2502 \u251c\u2500\u2500 AddNewSectionLauncher # Button that launches the EditSection form, used for adding new section\n\u2502 \u2502 \u251c\u2500\u2500 EditAppConfig.vue # Form for editing appConfig\n\u2502 \u2502 \u251c\u2500\u2500 EditPageInfo.vue # Form for editing pageInfo\n\u2502 \u2502 \u251c\u2500\u2500 EditSection.vue # Form for adding / editing sections\n\u2502 \u2502 \u251c\u2500\u2500 EditItem.vue # Form for adding or editing items\n\u2502 \u2502 \u251c\u2500\u2500 EditModeSaveMenu.vue # The bar at the bottom of screen in edit mode, containing save buttons\n\u2502 \u2502 \u251c\u2500\u2500 EditModeTopBanner.vue # The bar at the top of screen in edit mode\n\u2502 \u2502 \u251c\u2500\u2500 ExportConfigMenu.vue # Modal for viewing / exporting edited config\n\u2502 \u2502 \u251c\u2500\u2500 MoveItemTo.vue # Form for moving / copying items to other sections\n\u2502 \u2502 \u2570\u2500\u2500 SaveCancelButtons.vue # Buttons visible in all the edit menus, to save or cancel changes\n\u2502 \u251c\u2500\u2500 LinkItems # Components for Sections and Link Items\n\u2502 \u2502 \u251c\u2500\u2500 Collapsable.vue # The collapsible functionality of sections\n\u2502 \u2502 \u251c\u2500\u2500 IframeModal.vue # Pop-up iframe modal, for viewing websites within the app\n\u2502 \u2502 \u251c\u2500\u2500 Item.vue # Main link item, which is displayed within an item group\n\u2502 \u2502 \u251c\u2500\u2500 ItemGroup.vue # Item group is a section containing icons\n\u2502 \u2502 \u251c\u2500\u2500 ItemIcon.vue # The icon used by both items and sections\n\u2502 \u2502 \u251c\u2500\u2500 ItemOpenMethodIcon.vue # A small icon, visible on hover, indicating opening method\n\u2502 \u2502 \u251c\u2500\u2500 ItemContextMenu.vue # The right-click menu, for showing Item opening methods and info\n\u2502 \u2502 \u251c\u2500\u2500 SectionContextMenu.vue # The right-click menu, for showing Section edit/ open options\n\u2502 \u2502 \u2570\u2500\u2500 StatusIndicator.vue # Traffic light dot, showing if app is online or down\n\u2502 \u251c\u2500\u2500 Minimal View # Components used for the startpage / minimal alternative view\n\u2502 \u2502 \u251c\u2500\u2500 MinimalHeading.vue # Title part of minimal view\n\u2502 \u2502 \u251c\u2500\u2500 MinimalSearch.vue # Search bar for minimal view\n\u2502 \u2502 \u2570\u2500\u2500 MinimalSection.vue # Tabbed-Item section for minimal view\n\u2502 \u251c\u2500\u2500 PageStrcture # Components relating the main structure of the page\n\u2502 \u2502 \u251c\u2500\u2500 Footer.vue # Footer, visible at the bottom of all pages\n\u2502 \u2502 \u251c\u2500\u2500 Header.vue # Header, visible at the top of pages, and includes title and nav\n\u2502 \u2502 \u251c\u2500\u2500 LoadingScreen.vue # Splash screen shown on first load\n\u2502 \u2502 \u251c\u2500\u2500 Nav.vue # Navigation bar, includes a list of links\n\u2502 \u2502 \u2570\u2500\u2500 PageTitle.vue # Page title and sub-title, visible within the Header\n\u2502 \u251c\u2500\u2500 Workspace # Components used for the multi-tasking/ Workspace view\n\u2502 \u2502 \u251c\u2500\u2500 MultiTaskingWeb.vue # When multi-tasking enabled, generates new iframe\n\u2502 \u2502 \u251c\u2500\u2500 SideBar.vue # The left sidebar for the workspace view\n\u2502 \u2502 \u251c\u2500\u2500 SideBarItem.vue # App item for the sidebar view\n\u2502 \u2502 \u251c\u2500\u2500 SideBarSection.vue # Collapsible collection of items within workspace sidebar\n\u2502 \u2502 \u251c\u2500\u2500 WebContent.vue # Workspace iframe view, displays content of current app\n\u2502 \u2502 \u2570\u2500\u2500 WidgetView.vue # Workspace container for displaying widgets in main content\n\u2502 \u251c\u2500\u2500 Widgets # Directory contains all custom widget components\n\u2502 \u2502 \u2570\u2500\u2500 .... # Too many to list, see widget docs instead\n\u2502 \u2570\u2500\u2500 Settings # Components relating to the quick-settings, in the top-right\n\u2502 \u251c\u2500\u2500 AuthButtons.vue # Logout button and other app info\n\u2502 \u251c\u2500\u2500 ConfigLauncher.vue # Icon that when clicked will launch the Configuration component\n\u2502 \u251c\u2500\u2500 CustomThemeMaker.vue # Color pickers for letting user build their own theme\n\u2502 \u251c\u2500\u2500 ItemSizeSelector.vue # Set of buttons used to set and save item size\n\u2502 \u251c\u2500\u2500 KeyboardShortcutInfo.vue# Small pop-up displaying the available keyboard shortcuts\n\u2502 \u251c\u2500\u2500 LanguageSwitcher.vue # Dropdown in a modal for changing app language\n\u2502 \u251c\u2500\u2500 LayoutSelector.vue # Set of buttons, letting the user select their desired layout\n\u2502 \u251c\u2500\u2500 SearchBar.vue # The input field in the header, used for searching the app\n\u2502 \u251c\u2500\u2500 SettingsContainer.vue # Container that wraps all the quick-settings components\n\u2502 \u2570\u2500\u2500 ThemeSelector.vue # Drop-down menu enabling the user to select and change themes\n\u251c\u2500\u2500 main.js # Main front-end entry point\n\u251c\u2500\u2500 registerServiceWorker.js # Registers and manages service workers, for PWA apps\n\u251c\u2500\u2500 router.js # Defines all available application routes\n\u251c\u2500\u2500 styles # Directory of all globally used common SCSS styles\n\u2502 \u251c\u2500\u2500 color-palette.scss # All color variable names and default values\n\u2502 \u251c\u2500\u2500 color-themes.scss # All variable values for built-in themes\n\u2502 \u251c\u2500\u2500 dimensions.scss # Dimensions and sizes as variables\n\u2502 \u251c\u2500\u2500 global-styles.scss # Basics and style resets used globally\n\u2502 \u251c\u2500\u2500 media-queries.scss # Screen sizes and media queries\n\u2502 \u251c\u2500\u2500 style-helpers.scss # SCSS functions used for modifying values\n\u2502 \u251c\u2500\u2500 typography.scss # Font and text styles used globally\n\u2502 \u2570\u2500\u2500 user-defined-themes.scss # Empty, put any custom styles or themes here\n\u251c\u2500\u2500 mixins # Reusable component bases, extended by other views / components\n\u2502 \u251c\u2500\u2500 ChartingMixin.js # Functions for rendering charts in widget components\n\u2502 \u251c\u2500\u2500 GlancesMixin.js # Functions for fetching system info from Glances for widgets\n\u2502 \u251c\u2500\u2500 HomeMixin.js # Functions for homepage, used by default, minimal and workspace views\n\u2502 \u2570\u2500\u2500 WidgetMixin.js # Functions for all widgets, like data fetching, updating and error handling\n\u251c\u2500\u2500 utils # Directory of re-used helper functions\n\u2502 \u251c\u2500\u2500 ArrowKeyNavigation.js # Functionality for arrow-key navigation\n\u2502 \u251c\u2500\u2500 Auth.js # Handles all authentication related actions\n\u2502 \u251c\u2500\u2500 CheckSectionVisibility.js # Checks which parts of the page should be visible/ hidden based on config\n\u2502 \u251c\u2500\u2500 ClickOutside.js # A directive for detecting click, used to hide dropdown, modal or context menu\n\u2502 \u251c\u2500\u2500 ConfigHelpers.js # Helper functions for managing configuration\n\u2502 \u251c\u2500\u2500 CloudBackup.js # Functionality for encrypting, processing and network calls\n\u2502 \u251c\u2500\u2500 ConfigSchema.json # The schema, used to validate the users conf.yml file\n\u2502 \u251c\u2500\u2500 ConfigAccumulator.js # Central place for managing and combining config\n\u2502 \u251c\u2500\u2500 ConfigHelpers.js # Collection of helper functions to process config using accumulator\n\u2502 \u251c\u2500\u2500 ConfigValidator.js # A helper script that validates the config file against schema\n\u2502 \u251c\u2500\u2500 CoolConsole.js # Prints info, warning and error messages to browser console, with a cool style\n\u2502 \u251c\u2500\u2500 defaults.js # Global constants and their default values\n\u2502 \u251c\u2500\u2500 emojis.json # List of emojis with unicode and shortcode, used for emoji icon feature\n\u2502 \u251c\u2500\u2500 EmojiUnicodeRegex.js # Regular expression to validate emoji unicode format, for emoji icons\n\u2502 \u251c\u2500\u2500 ErrorHandler.js # Helper function called when an error is returned\n\u2502 \u251c\u2500\u2500 InitServiceWorker.js # Initializes and manages service worker, if enabled\n\u2502 \u251c\u2500\u2500 Search.js # Helper functions for searching/ filtering items in all views\n\u2502 \u251c\u2500\u2500 JsonToYaml.js # Function that parses and converts raw JSON into valid YAML\n\u2502 \u251c\u2500\u2500 KeycloakAuth.js # Singleton class to manage Keycloak authentication\n\u2502 \u251c\u2500\u2500 OidcAuth.js # Manages OIDC authentication with external providers\n\u2502 \u251c\u2500\u2500 HeaderAuth.js # Handles header-based authentication via reverse proxy\n\u2502 \u251c\u2500\u2500 languages.js # Handles fetching, switching and validating languages\n\u2502 \u2570\u2500\u2500 ThemeHelper.js # Function that handles the fetching and setting of user themes\n\u2570\u2500\u2500 views # Directory of available pages, corresponding to available routes\n \u251c\u2500\u2500 Home.vue # The home page container\n \u251c\u2500\u2500 About.vue # About page\n \u251c\u2500\u2500 Login.vue # Authentication page\n \u251c\u2500\u2500 Minimal.vue # The minimal view\n \u251c\u2500\u2500 Workspace.vue # The workspace view with apps in sidebar\n \u251c\u2500\u2500 DownloadConfig.vue # Config export page\n \u2570\u2500\u2500 404.vue # Not found page\n"})}),"\n",(0,t.jsx)(n.h4,{id:"visualisation-of-source-directory",children:"Visualisation of Source Directory"}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/repo-visualization.svg",alt:"File Breakdown"})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"development-tools",children:"Development Tools"}),"\n",(0,t.jsx)(n.h3,{id:"performance---lighthouse",children:"Performance - Lighthouse"}),"\n",(0,t.jsx)(n.p,{children:"The easiest method of checking performance is to use Chromium's build in auditing tool, Lighthouse. To run the test, open Developer Tools (usually F12) --\x3e Lighthouse and click on the 'Generate Report' button at the bottom."}),"\n",(0,t.jsx)(n.h3,{id:"dependencies---bundlephobia",children:"Dependencies - BundlePhobia"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://bundlephobia.com/",children:"BundlePhobia"})," is a really useful app that lets you analyze the cost of adding any particular dependency to an application"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"notes-1",children:"Notes"}),"\n",(0,t.jsx)(n.h3,{id:"known-warnings",children:"Known Warnings"}),"\n",(0,t.jsx)(n.p,{children:"When running the build command, several warnings appear. These are not errors, and do not affect the security or performance of the application. They will be addressed in a future update"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"WARN A new version of sass-loader is available. Please upgrade for best experience."})," - Currently we're using an older version of SASS loader, since the more recent releases do not seem to be compatible with the Vue CLI's webpack configuration."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"WARN asset size limit: The following asset(s) exceed the recommended size limit (244 KiB)."})," - For the PWA to support Windows 10, a splash screen asset is required, and is quite large. This throws a warning, however PWA assets are not loaded until needed, so shouldn't have any impact on application performance. A similar warning is thrown for the Raleway font, and that is looking to be addressed."]})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(c,{...e})}):c(e)}},8453(e,n,i){i.d(n,{R:()=>r,x:()=>l});var s=i(6540);const t={},o=s.createContext(t);function r(e){const n=s.useContext(o);return s.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function l(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),s.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5992],{2745(e,n,i){i.r(n),i.d(n,{assets:()=>a,contentTitle:()=>l,default:()=>h,frontMatter:()=>r,metadata:()=>s,toc:()=>d});const s=JSON.parse('{"id":"developing","title":"Developing","description":"This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture.","source":"@site/docs/developing.md","sourceDirName":".","slug":"/developing","permalink":"/docs/developing","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/developing.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Contributing","permalink":"/docs/contributing"},"next":{"title":"Development Guides","permalink":"/docs/development-guides"}}');var t=i(4848),o=i(8453);const r={},l="Developing",a={},d=[{value:"Setting up the Dev Environment",id:"setting-up-the-dev-environment",level:2},{value:"Prerequisites",id:"prerequisites",level:3},{value:"Running the Project",id:"running-the-project",level:3},{value:"Project Commands",id:"project-commands",level:3},{value:"Basics",id:"basics",level:4},{value:"Development",id:"development",level:4},{value:"Utils and Checks",id:"utils-and-checks",level:4},{value:"Alternate Start Commands",id:"alternate-start-commands",level:4},{value:"Notes",id:"notes",level:4},{value:"Environmental Variables",id:"environmental-variables",level:3},{value:"Environment Modes",id:"environment-modes",level:3},{value:"Git Strategy",id:"git-strategy",level:2},{value:"Git Flow",id:"git-flow",level:3},{value:"Git Branch Naming",id:"git-branch-naming",level:3},{value:"Commit Emojis",id:"commit-emojis",level:3},{value:"PR Guidelines",id:"pr-guidelines",level:3},{value:"Resources for Beginners",id:"resources-for-beginners",level:2},{value:"App Info",id:"app-info",level:2},{value:"Style Guide",id:"style-guide",level:3},{value:"Application Structure",id:"application-structure",level:3},{value:"Files in the Root: ./",id:"files-in-the-root-",level:4},{value:"Frontend Source: ./src/",id:"frontend-source-src",level:4},{value:"Visualisation of Source Directory",id:"visualisation-of-source-directory",level:4},{value:"Development Tools",id:"development-tools",level:2},{value:"Performance - Lighthouse",id:"performance---lighthouse",level:3},{value:"Dependencies - BundlePhobia",id:"dependencies---bundlephobia",level:3},{value:"Notes",id:"notes-1",level:2},{value:"Known Warnings",id:"known-warnings",level:3}];function c(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"developing",children:"Developing"})}),"\n",(0,t.jsxs)(n.p,{children:["This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture.\nIf you're adding new features, you may want to check out the ",(0,t.jsx)(n.a,{href:"/docs/development-guides",children:"Development Guides"})," docs, for tutorials covering basic tasks."]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#setting-up-the-dev-environment",children:"Setting up the Development Environment"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#prerequisites",children:"Prerequisites"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#running-the-project",children:"Running the App"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#project-commands",children:"Project Commands"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#environmental-variables",children:"Environmental Variables"})}),"\n"]}),"\n"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.a,{href:"#git-strategy",children:"Git Strategy"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#git-flow",children:"Flow"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#git-branch-naming",children:"Branches"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#commit-emojis",children:"Commit emojis"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#pr-guidelines",children:"PR Guidelines"})}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#resources-for-beginners",children:"Resources for Beginners"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#app-info",children:"App Info"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#style-guide",children:"Code Style Guide"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#application-structure",children:"Application Structure"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#development-tools",children:"Development Tools"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#notes",children:"Misc / Notes"})}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"setting-up-the-dev-environment",children:"Setting up the Dev Environment"}),"\n",(0,t.jsx)(n.h3,{id:"prerequisites",children:"Prerequisites"}),"\n",(0,t.jsxs)(n.p,{children:["You will need either the latest or LTS version of ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://nodejs.org/",children:"Node.js"})})," to build and serve the application and ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://git-scm.com/downloads",children:"Git"})})," to easily fetch the code, and push any changes. If you plan on running or deploying the container, you'll also need ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://docs.docker.com/get-docker/",children:"Docker"})}),". To avoid any unexpected issues, ensure you've got at least ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://www.npmjs.com/get-npm",children:"NPM"})})," V 7.5 or ",(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.a,{href:"https://classic.yarnpkg.com/en/docs/install/#windows-stable",children:"Yarn"})})," 1.22 (you may find ",(0,t.jsx)(n.a,{href:"https://github.com/nvm-sh/nvm",children:"NVM"})," helpful for switching/ managing versions)."]}),"\n",(0,t.jsx)(n.h3,{id:"running-the-project",children:"Running the Project"}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsxs)(n.li,{children:["Get Code: ",(0,t.jsx)(n.code,{children:"git clone https://github.com/Lissy93/dashy.git"})]}),"\n",(0,t.jsxs)(n.li,{children:["Navigate into the directory: ",(0,t.jsx)(n.code,{children:"cd dashy"})]}),"\n",(0,t.jsxs)(n.li,{children:["Install dependencies: ",(0,t.jsx)(n.code,{children:"yarn"})]}),"\n",(0,t.jsxs)(n.li,{children:["Start dev server: ",(0,t.jsx)(n.code,{children:"yarn dev"})]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["Dashy should now be being served on ",(0,t.jsx)(n.a,{href:"http://localhost:8080/",children:"http://localhost:8080/"}),". Hot reload is enabled, so making changes to any of the files will trigger them to be rebuilt and the page refreshed."]}),"\n",(0,t.jsx)(n.h3,{id:"project-commands",children:"Project Commands"}),"\n",(0,t.jsx)(n.h4,{id:"basics",children:"Basics"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn build"})})," - Builds the production bundle into ",(0,t.jsx)(n.code,{children:"./dist"}),". The Vite dev/runtime server serves ",(0,t.jsx)(n.code,{children:"user-data/conf.yml"})," on each request, so a rebuild is only needed when source code or assets change, not when config changes"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn start"})})," - Starts the Node server, which serves the built site from ",(0,t.jsx)(n.code,{children:"./dist"})," and the live config from ",(0,t.jsx)(n.code,{children:"user-data/"}),". Run ",(0,t.jsx)(n.code,{children:"yarn build"})," first"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"development",children:"Development"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn dev"})})," - Starts the development server with hot reloading"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn lint"})})," - Lints code to ensure it follows a consistent, neat style"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn test"})})," - Runs tests, and outputs results"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"utils-and-checks",children:"Utils and Checks"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn validate-config"})})," - If you have quite a long configuration file, you may wish to check that it's all good to go, before deploying the app. This can be done with ",(0,t.jsx)(n.code,{children:"yarn validate-config"})," or ",(0,t.jsx)(n.code,{children:"docker exec -it [container-id] yarn validate-config"}),". Your config file needs to be in ",(0,t.jsx)(n.code,{children:"/user-data/conf.yml"})," (or within your Docker container at ",(0,t.jsx)(n.code,{children:"/app/user-data/conf.yml"}),"). This will first check that your YAML is valid, and then validates it against Dashy's ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"schema"}),"."]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn health-check"})})," - Checks that the application is up and running on it's specified port, and outputs current status and response times. Useful for integrating into your monitoring service, if you need to maintain high system availability"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"alternate-start-commands",children:"Alternate Start Commands"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn build && yarn start"})})," - Builds the app, then starts the production Node server. Use this for a manual production-style run on bare metal. With Vite, ",(0,t.jsx)(n.code,{children:"conf.yml"})," is served from ",(0,t.jsx)(n.code,{children:"user-data/"})," at runtime, so config changes only require a page refresh"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.strong,{children:(0,t.jsx)(n.code,{children:"yarn pm2-start"})})," - Starts the Node server using ",(0,t.jsx)(n.a,{href:"https://pm2.keymetrics.io/",children:"PM2"}),", a process manager for Node.js applications, that helps them stay alive. PM2 has some built-in basic monitoring features, and an optional ",(0,t.jsx)(n.a,{href:"https://pm2.io/",children:"management solution"}),". If you are running the app on bare metal, it is recommended to use this start command"]}),"\n"]}),"\n",(0,t.jsx)(n.h4,{id:"notes",children:"Notes"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["If you are using NPM, replace ",(0,t.jsx)(n.code,{children:"yarn"})," with ",(0,t.jsx)(n.code,{children:"npm run"})]}),"\n",(0,t.jsxs)(n.li,{children:["If you are using Docker, precede each command with ",(0,t.jsx)(n.code,{children:"docker exec -it [container-id]"}),". Container ID can be found by running ",(0,t.jsx)(n.code,{children:"docker ps"})]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"environmental-variables",children:"Environmental Variables"}),"\n",(0,t.jsxs)(n.p,{children:["All environmental variables are optional. Currently there are not many environmental variables used, as most of the user preferences are stored under ",(0,t.jsx)(n.code,{children:"appConfig"})," in the ",(0,t.jsx)(n.code,{children:"conf.yml"})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["You can set variables either in your environment, or using the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.env",children:(0,t.jsx)(n.code,{children:".env"})})," file."]}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"NODE_ENV"})," - Current environment, can be either development, production or test"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"PORT"})," - The port to expose the running application on"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"HOST"})," - The host that Dashy is running on, domain or IP"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"BASE_URL"})," - The default base path for serving up static assets"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"VITE_APP_DOMAIN"})," - Usually the same as BASE_URL, but accessible in frontend"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"INTEGRITY"})," - Should enable SRI for build script and link resources"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"IS_DOCKER"})," - Computed automatically on build. Indicates if running in container"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"VITE_APP_VERSION"})," - Again, set automatically using package.json during build time"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"BACKUP_DIR"})," - Directory for conf.yml backups"]}),"\n",(0,t.jsxs)(n.li,{children:[(0,t.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS"})," - Set to 'true' to skip the pre-save backup step (useful on read-only filesystems or where permissions don't allow it)"]}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"environment-modes",children:"Environment Modes"}),"\n",(0,t.jsxs)(n.p,{children:["You can set the environment using the ",(0,t.jsx)(n.code,{children:"NODE_ENV"})," variable. By default, the correct environment should be selected based on the script you run to start the app. The following environments are supported: ",(0,t.jsx)(n.code,{children:"production"}),", ",(0,t.jsx)(n.code,{children:"development"})," and ",(0,t.jsx)(n.code,{children:"test"}),". For more info, see ",(0,t.jsx)(n.a,{href:"https://cli.vuejs.org/guide/mode-and-env.html#modes",children:"Vue CLI Environment Modes"}),"."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"git-strategy",children:"Git Strategy"}),"\n",(0,t.jsx)(n.h3,{id:"git-flow",children:"Git Flow"}),"\n",(0,t.jsxs)(n.p,{children:["Like most Git repos, we are following the ",(0,t.jsx)(n.a,{href:"https://guides.github.com/introduction/flow",children:"Github Flow"})," standard."]}),"\n",(0,t.jsxs)(n.ol,{children:["\n",(0,t.jsx)(n.li,{children:"Create a branch (or fork if you don'd have write access)"}),"\n",(0,t.jsx)(n.li,{children:"Code some awesome stuff \ud83e\uddd1\u200d\ud83d\udcbb"}),"\n",(0,t.jsx)(n.li,{children:"Add, commit and push your changes to your branch/ fork"}),"\n",(0,t.jsx)(n.li,{children:"Head over to GitHub and create a Pull Request"}),"\n",(0,t.jsx)(n.li,{children:"Fill in the required sections in the template, and hit submit"}),"\n",(0,t.jsx)(n.li,{children:"Follow up with any reviews on your code"}),"\n",(0,t.jsx)(n.li,{children:"Merge \ud83c\udf89"}),"\n"]}),"\n",(0,t.jsx)(n.h3,{id:"git-branch-naming",children:"Git Branch Naming"}),"\n",(0,t.jsxs)(n.p,{children:["The format of your branch name should be something similar to: ",(0,t.jsx)(n.code,{children:"[TYPE]/[TICKET]_[TITLE]"}),"\nFor example, ",(0,t.jsx)(n.code,{children:"FEATURE/420_Awesome-feature"})," or ",(0,t.jsx)(n.code,{children:"FIX/690_login-server-error"})]}),"\n",(0,t.jsx)(n.h3,{id:"commit-emojis",children:"Commit Emojis"}),"\n",(0,t.jsx)(n.p,{children:"Using a single emoji at the start of each commit message, to indicate the type task, makes the commit ledger easier to understand, plus it looks cool."}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["\ud83c\udfa8 ",(0,t.jsx)(n.code,{children:":art:"})," - Improve structure / format of the code."]}),"\n",(0,t.jsxs)(n.li,{children:["\u26a1\ufe0f ",(0,t.jsx)(n.code,{children:":zap:"})," - Improve performance."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd25 ",(0,t.jsx)(n.code,{children:":fire:"})," - Remove code or files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc1b ",(0,t.jsx)(n.code,{children:":bug:"})," - Fix a bug."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\ude91\ufe0f ",(0,t.jsx)(n.code,{children:":ambulance:"})," - Critical hotfix"]}),"\n",(0,t.jsxs)(n.li,{children:["\u2728 ",(0,t.jsx)(n.code,{children:":sparkles:"})," - Introduce new features."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udcdd ",(0,t.jsx)(n.code,{children:":memo:"})," - Add or update documentation."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\ude80 ",(0,t.jsx)(n.code,{children:":rocket:"})," - Deploy stuff."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc84 ",(0,t.jsx)(n.code,{children:":lipstick:"})," - Add or update the UI and style files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf89 ",(0,t.jsx)(n.code,{children:":tada:"})," - Begin a project."]}),"\n",(0,t.jsxs)(n.li,{children:["\u2705 ",(0,t.jsx)(n.code,{children:":white_check_mark:"})," - Add, update, or pass tests."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd12\ufe0f ",(0,t.jsx)(n.code,{children:":lock:"})," - Fix security issues."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd16 ",(0,t.jsx)(n.code,{children:":bookmark:"})," - Make a Release or Version tag."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udea8 ",(0,t.jsx)(n.code,{children:":rotating_light:"})," - Fix compiler / linter warnings."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udea7 ",(0,t.jsx)(n.code,{children:":construction:"})," - Work in progress."]}),"\n",(0,t.jsxs)(n.li,{children:["\u2b06\ufe0f ",(0,t.jsx)(n.code,{children:":arrow_up:"})," - Upgrade dependencies."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udc77 ",(0,t.jsx)(n.code,{children:":construction_worker:"})," - Add or update CI build system."]}),"\n",(0,t.jsxs)(n.li,{children:["\u267b\ufe0f ",(0,t.jsx)(n.code,{children:":recycle:"})," - Refactor code."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83e\ude79 ",(0,t.jsx)(n.code,{children:":adhesive_bandage:"})," - Simple fix for a non-critical issue."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\udd27 ",(0,t.jsx)(n.code,{children:":wrench:"})," - Add or update configuration files."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf71 ",(0,t.jsx)(n.code,{children:":bento:"})," - Add or update assets."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83d\uddc3\ufe0f ",(0,t.jsx)(n.code,{children:":card_file_box:"})," - Perform database schema related changes."]}),"\n",(0,t.jsxs)(n.li,{children:["\u270f\ufe0f ",(0,t.jsx)(n.code,{children:":pencil2:"})," - Fix typos."]}),"\n",(0,t.jsxs)(n.li,{children:["\ud83c\udf10 ",(0,t.jsx)(n.code,{children:":globe_with_meridians:"})," - Internationalization and translations."]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["For a full list of options, see ",(0,t.jsx)(n.a,{href:"https://gitmoji.dev/",children:"gitmoji.dev"})]}),"\n",(0,t.jsx)(n.h3,{id:"pr-guidelines",children:"PR Guidelines"}),"\n",(0,t.jsx)(n.p,{children:"Once you've made your changes, and pushed them to your fork or branch, you're ready to open a pull request!"}),"\n",(0,t.jsx)(n.p,{children:"For a pull request to be merged, it must:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Must be backwards compatible"}),"\n",(0,t.jsx)(n.li,{children:"The build, lint and tests (run by GH actions) must pass"}),"\n",(0,t.jsx)(n.li,{children:"There must not be any merge conflicts"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"When you submit your PR, include the required info, by filling out the PR template. Including:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"A brief description of your changes"}),"\n",(0,t.jsx)(n.li,{children:"The issue, ticket or discussion number (if applicable)"}),"\n",(0,t.jsx)(n.li,{children:"For UI relate updates include a screenshot"}),"\n",(0,t.jsx)(n.li,{children:"If any dependencies were added, explain why it was needed, state the cost associated, and confirm it does not introduce any security issues"}),"\n",(0,t.jsx)(n.li,{children:"Finally, check the checkboxes, to confirm that the standards are met, and hit submit!"}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"resources-for-beginners",children:"Resources for Beginners"}),"\n",(0,t.jsx)(n.p,{children:"New to Web Development? Glad you're here! Dashy is a pretty simple app, so it should make a good candidate for your first PR. Presuming that you already have a basic knowledge of JavaScript, the following articles should point you in the right direction for getting up to speed with the technologies used in this project:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://opensource.guide/how-to-contribute/",children:"Open Source for Beginners"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://v3.vuejs.org/guide/introduction.html",children:"Introduction to Vue.js"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.taniarascia.com/getting-started-with-vue/",children:"Vue.js Walkthrough"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://github.com/lukehoban/es6features",children:"ES6 Features"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://blog.logrocket.com/the-definitive-guide-to-scss/",children:"Definitive guide to SCSS"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://docker-curriculum.com/",children:"Complete beginners guide to Docker"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://training.play-with-docker.com/",children:"Docker Classroom - Interactive Tutorials"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.freecodecamp.org/news/learn-typescript-in-5-minutes-13eda868daeb/",children:"Quick start TypeScript guide"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.typescripttutorial.net/",children:"Complete TypeScript tutorial series"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://blog.logrocket.com/vue-typescript-tutorial-examples/",children:"Using TypeScript with Vue.js"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"http://git-cheatsheet.com/",children:"Git cheat sheet"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"https://www.freecodecamp.org/news/what-is-npm-a-node-package-manager-tutorial-for-beginners/",children:"Basics of using NPM"})}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["As well as Node, Git and Docker- you'll also need an IDE (e.g. ",(0,t.jsx)(n.a,{href:"https://code.visualstudio.com/",children:"VS Code"})," or ",(0,t.jsx)(n.a,{href:"https://www.vim.org/",children:"Vim"}),") and a terminal (Windows users may find ",(0,t.jsx)(n.a,{href:"https://docs.microsoft.com/en-us/windows/wsl/",children:"WSL"})," more convenient)."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"app-info",children:"App Info"}),"\n",(0,t.jsx)(n.h3,{id:"style-guide",children:"Style Guide"}),"\n",(0,t.jsxs)(n.p,{children:["Linting is done using ",(0,t.jsx)(n.a,{href:"https://eslint.org/",children:"ESLint"}),", and using the ",(0,t.jsx)(n.a,{href:"https://github.com/vuejs/eslint-config-standard",children:"Vue.js Styleguide"}),", which is very similar to the ",(0,t.jsx)(n.a,{href:"https://github.com/airbnb/javascript",children:"AirBnB Styleguide"}),". You can run ",(0,t.jsx)(n.code,{children:"yarn lint"})," to report and fix issues. While the dev server is running, issues will be reported to the console automatically, and any lint errors will trigger the build to fail. Note that all lint checks must pass before any PR can be merged. Linting is also run as a git pre-commit hook"]}),"\n",(0,t.jsx)(n.p,{children:"The most significant things to note are:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:"Indentation should be done with two spaces"}),"\n",(0,t.jsx)(n.li,{children:"Strings should use single quotes"}),"\n",(0,t.jsx)(n.li,{children:"All statements must end in a semi-colon"}),"\n",(0,t.jsx)(n.li,{children:"The final element in all objects must be preceded with a comma"}),"\n",(0,t.jsx)(n.li,{children:"Maximum line length is 100"}),"\n",(0,t.jsx)(n.li,{children:"There must be exactly one blank line between sections, before function names, and at the end of the file"}),"\n",(0,t.jsx)(n.li,{children:"With conditionals, put else on the same line as your if block's closing brace"}),"\n",(0,t.jsx)(n.li,{children:"All multiline blocks must use braces"}),"\n",(0,t.jsx)(n.li,{children:"Avoid console statements in the frontend"}),"\n"]}),"\n",(0,t.jsx)(n.p,{children:"Styleguides:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["Vue: ",(0,t.jsx)(n.a,{href:"https://vuejs.org/v2/style-guide/",children:"Vue styleguide"})]}),"\n",(0,t.jsxs)(n.li,{children:["JavaScript: ",(0,t.jsx)(n.a,{href:"https://github.com/airbnb/javascript",children:"github.com/airbnb/javascript"})]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h3,{id:"application-structure",children:"Application Structure"}),"\n",(0,t.jsxs)(n.h4,{id:"files-in-the-root-",children:["Files in the Root: ",(0,t.jsx)(n.code,{children:"./"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"\u256e\n\u251c\u2500\u2500 package.json # Project meta-data, dependencies and paths to scripts\n\u251c\u2500\u2500 src/ # Project front-end source code\n\u251c\u2500\u2500 server.js # A Node.js server to serve up the /dist directory\n\u251c\u2500\u2500 services/ # All server-side endpoints and utilities\n\u251c\u2500\u2500 vue.config.js # Vue.js configuration\n\u251c\u2500\u2500 Dockerfile # The blueprint for building the Docker container\n\u251c\u2500\u2500 docker-compose.yml # A Docker run command\n\u251c\u2500\u2500 .env # Location for any environmental variables\n\u251c\u2500\u2500 yarn.lock # Auto-generated list of current packages and version numbers\n\u251c\u2500\u2500 docs/ # Markdown documentation\n\u251c\u2500\u2500 README.md # Readme, basic info for getting started\n\u251c\u2500\u2500 LICENSE.md # License for use\n\u256f\n"})}),"\n",(0,t.jsxs)(n.h4,{id:"frontend-source-src",children:["Frontend Source: ",(0,t.jsx)(n.code,{children:"./src/"})]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-text",children:"./src\n\u251c\u2500\u2500 App.vue # Vue.js starting file\n\u251c\u2500\u2500 assets # Static non-compiled assets\n\u2502 \u251c\u2500\u2500 fonts # .ttf font files\n\u2502 \u251c\u2500\u2500 locales # All app text, each language in a separate JSON file\n\u2502 \u2570\u2500\u2500 interface-icons # SVG icons used in the app\n\u251c\u2500\u2500 components # All front-end Vue web components\n\u2502 \u251c\u2500\u2500 Charts # Charting components for dynamically displaying widget data\n\u2502 \u2502 \u251c\u2500\u2500 Gauge.vue # A speed-dial style chart for showing 0 - 100 values\n\u2502 \u2502 \u2570\u2500\u2500 PercentageChart.vue # A horizontal bar for showing percentage breakdowns\n\u2502 \u251c\u2500\u2500 Configuration # Components relating to the user config pop-up\n\u2502 \u2502 \u251c\u2500\u2500 AppInfoModal.vue # A modal showing core app info, like version, language, etc\n\u2502 \u2502 \u251c\u2500\u2500 AppVersion.vue # Shows current version from package.json, compares with GitHub\n\u2502 \u2502 \u251c\u2500\u2500 CloudBackupRestore.vue # Form where the user manages cloud sync options\n\u2502 \u2502 \u251c\u2500\u2500 ConfigContainer.vue # Main container, wrapping all other config components\n\u2502 \u2502 \u251c\u2500\u2500 CustomCss.vue # Form where the user can input custom CSS\n\u2502 \u2502 \u2570\u2500\u2500 JsonEditor.vue # JSON editor, where the user can modify the main config file\n\u2502 \u251c\u2500\u2500 FormElements # Basic form elements used throughout the app\n\u2502 \u2502 \u251c\u2500\u2500 Button.vue # Standard button component\n\u2502 \u2502 \u251c\u2500\u2500 Radio.vue # Standard radio button input\n\u2502 \u2502 \u251c\u2500\u2500 Select.vue # Standard dropdown input selector\n\u2502 \u2502 \u251c\u2500\u2500 Input.vue # Standard text field input component\n\u2502 \u2502 \u2570\u2500\u2500 Toggle.vue # Standard on / off toggle switch\n\u2502 \u251c\u2500\u2500 InteractiveEditor # Components for the interactive UI config editor\n\u2502 \u2502 \u251c\u2500\u2500 AddNewSectionLauncher # Button that launches the EditSection form, used for adding new section\n\u2502 \u2502 \u251c\u2500\u2500 EditAppConfig.vue # Form for editing appConfig\n\u2502 \u2502 \u251c\u2500\u2500 EditPageInfo.vue # Form for editing pageInfo\n\u2502 \u2502 \u251c\u2500\u2500 EditSection.vue # Form for adding / editing sections\n\u2502 \u2502 \u251c\u2500\u2500 EditItem.vue # Form for adding or editing items\n\u2502 \u2502 \u251c\u2500\u2500 EditModeSaveMenu.vue # The bar at the bottom of screen in edit mode, containing save buttons\n\u2502 \u2502 \u251c\u2500\u2500 EditModeTopBanner.vue # The bar at the top of screen in edit mode\n\u2502 \u2502 \u251c\u2500\u2500 ExportConfigMenu.vue # Modal for viewing / exporting edited config\n\u2502 \u2502 \u251c\u2500\u2500 MoveItemTo.vue # Form for moving / copying items to other sections\n\u2502 \u2502 \u2570\u2500\u2500 SaveCancelButtons.vue # Buttons visible in all the edit menus, to save or cancel changes\n\u2502 \u251c\u2500\u2500 LinkItems # Components for Sections and Link Items\n\u2502 \u2502 \u251c\u2500\u2500 Collapsable.vue # The collapsible functionality of sections\n\u2502 \u2502 \u251c\u2500\u2500 IframeModal.vue # Pop-up iframe modal, for viewing websites within the app\n\u2502 \u2502 \u251c\u2500\u2500 Item.vue # Main link item, which is displayed within an item group\n\u2502 \u2502 \u251c\u2500\u2500 ItemGroup.vue # Item group is a section containing icons\n\u2502 \u2502 \u251c\u2500\u2500 ItemIcon.vue # The icon used by both items and sections\n\u2502 \u2502 \u251c\u2500\u2500 ItemOpenMethodIcon.vue # A small icon, visible on hover, indicating opening method\n\u2502 \u2502 \u251c\u2500\u2500 ItemContextMenu.vue # The right-click menu, for showing Item opening methods and info\n\u2502 \u2502 \u251c\u2500\u2500 SectionContextMenu.vue # The right-click menu, for showing Section edit/ open options\n\u2502 \u2502 \u2570\u2500\u2500 StatusIndicator.vue # Traffic light dot, showing if app is online or down\n\u2502 \u251c\u2500\u2500 Minimal View # Components used for the startpage / minimal alternative view\n\u2502 \u2502 \u251c\u2500\u2500 MinimalHeading.vue # Title part of minimal view\n\u2502 \u2502 \u251c\u2500\u2500 MinimalSearch.vue # Search bar for minimal view\n\u2502 \u2502 \u2570\u2500\u2500 MinimalSection.vue # Tabbed-Item section for minimal view\n\u2502 \u251c\u2500\u2500 PageStrcture # Components relating the main structure of the page\n\u2502 \u2502 \u251c\u2500\u2500 Footer.vue # Footer, visible at the bottom of all pages\n\u2502 \u2502 \u251c\u2500\u2500 Header.vue # Header, visible at the top of pages, and includes title and nav\n\u2502 \u2502 \u251c\u2500\u2500 LoadingScreen.vue # Splash screen shown on first load\n\u2502 \u2502 \u251c\u2500\u2500 Nav.vue # Navigation bar, includes a list of links\n\u2502 \u2502 \u2570\u2500\u2500 PageTitle.vue # Page title and sub-title, visible within the Header\n\u2502 \u251c\u2500\u2500 Workspace # Components used for the multi-tasking/ Workspace view\n\u2502 \u2502 \u251c\u2500\u2500 MultiTaskingWeb.vue # When multi-tasking enabled, generates new iframe\n\u2502 \u2502 \u251c\u2500\u2500 SideBar.vue # The left sidebar for the workspace view\n\u2502 \u2502 \u251c\u2500\u2500 SideBarItem.vue # App item for the sidebar view\n\u2502 \u2502 \u251c\u2500\u2500 SideBarSection.vue # Collapsible collection of items within workspace sidebar\n\u2502 \u2502 \u251c\u2500\u2500 WebContent.vue # Workspace iframe view, displays content of current app\n\u2502 \u2502 \u2570\u2500\u2500 WidgetView.vue # Workspace container for displaying widgets in main content\n\u2502 \u251c\u2500\u2500 Widgets # Directory contains all custom widget components\n\u2502 \u2502 \u2570\u2500\u2500 .... # Too many to list, see widget docs instead\n\u2502 \u2570\u2500\u2500 Settings # Components relating to the quick-settings, in the top-right\n\u2502 \u251c\u2500\u2500 AuthButtons.vue # Logout button and other app info\n\u2502 \u251c\u2500\u2500 ConfigLauncher.vue # Icon that when clicked will launch the Configuration component\n\u2502 \u251c\u2500\u2500 CustomThemeMaker.vue # Color pickers for letting user build their own theme\n\u2502 \u251c\u2500\u2500 ItemSizeSelector.vue # Set of buttons used to set and save item size\n\u2502 \u251c\u2500\u2500 KeyboardShortcutInfo.vue# Small pop-up displaying the available keyboard shortcuts\n\u2502 \u251c\u2500\u2500 LanguageSwitcher.vue # Dropdown in a modal for changing app language\n\u2502 \u251c\u2500\u2500 LayoutSelector.vue # Set of buttons, letting the user select their desired layout\n\u2502 \u251c\u2500\u2500 SearchBar.vue # The input field in the header, used for searching the app\n\u2502 \u251c\u2500\u2500 SettingsContainer.vue # Container that wraps all the quick-settings components\n\u2502 \u2570\u2500\u2500 ThemeSelector.vue # Drop-down menu enabling the user to select and change themes\n\u251c\u2500\u2500 main.js # Main front-end entry point\n\u251c\u2500\u2500 registerServiceWorker.js # Registers and manages service workers, for PWA apps\n\u251c\u2500\u2500 router.js # Defines all available application routes\n\u251c\u2500\u2500 styles # Directory of all globally used common SCSS styles\n\u2502 \u251c\u2500\u2500 color-palette.scss # All color variable names and default values\n\u2502 \u251c\u2500\u2500 color-themes.scss # All variable values for built-in themes\n\u2502 \u251c\u2500\u2500 dimensions.scss # Dimensions and sizes as variables\n\u2502 \u251c\u2500\u2500 global-styles.scss # Basics and style resets used globally\n\u2502 \u251c\u2500\u2500 media-queries.scss # Screen sizes and media queries\n\u2502 \u251c\u2500\u2500 style-helpers.scss # SCSS functions used for modifying values\n\u2502 \u251c\u2500\u2500 typography.scss # Font and text styles used globally\n\u2502 \u2570\u2500\u2500 user-defined-themes.scss # Empty, put any custom styles or themes here\n\u251c\u2500\u2500 mixins # Reusable component bases, extended by other views / components\n\u2502 \u251c\u2500\u2500 ChartingMixin.js # Functions for rendering charts in widget components\n\u2502 \u251c\u2500\u2500 GlancesMixin.js # Functions for fetching system info from Glances for widgets\n\u2502 \u251c\u2500\u2500 HomeMixin.js # Functions for homepage, used by default, minimal and workspace views\n\u2502 \u2570\u2500\u2500 WidgetMixin.js # Functions for all widgets, like data fetching, updating and error handling\n\u251c\u2500\u2500 utils # Directory of re-used helper functions\n\u2502 \u251c\u2500\u2500 ArrowKeyNavigation.js # Functionality for arrow-key navigation\n\u2502 \u251c\u2500\u2500 Auth.js # Handles all authentication related actions\n\u2502 \u251c\u2500\u2500 CheckSectionVisibility.js # Checks which parts of the page should be visible/ hidden based on config\n\u2502 \u251c\u2500\u2500 ClickOutside.js # A directive for detecting click, used to hide dropdown, modal or context menu\n\u2502 \u251c\u2500\u2500 ConfigHelpers.js # Helper functions for managing configuration\n\u2502 \u251c\u2500\u2500 CloudBackup.js # Functionality for encrypting, processing and network calls\n\u2502 \u251c\u2500\u2500 ConfigSchema.json # The schema, used to validate the users conf.yml file\n\u2502 \u251c\u2500\u2500 ConfigAccumulator.js # Central place for managing and combining config\n\u2502 \u251c\u2500\u2500 ConfigHelpers.js # Collection of helper functions to process config using accumulator\n\u2502 \u251c\u2500\u2500 ConfigValidator.js # A helper script that validates the config file against schema\n\u2502 \u251c\u2500\u2500 CoolConsole.js # Prints info, warning and error messages to browser console, with a cool style\n\u2502 \u251c\u2500\u2500 defaults.js # Global constants and their default values\n\u2502 \u251c\u2500\u2500 emojis.json # List of emojis with unicode and shortcode, used for emoji icon feature\n\u2502 \u251c\u2500\u2500 EmojiUnicodeRegex.js # Regular expression to validate emoji unicode format, for emoji icons\n\u2502 \u251c\u2500\u2500 ErrorHandler.js # Helper function called when an error is returned\n\u2502 \u251c\u2500\u2500 InitServiceWorker.js # Initializes and manages service worker, if enabled\n\u2502 \u251c\u2500\u2500 Search.js # Helper functions for searching/ filtering items in all views\n\u2502 \u251c\u2500\u2500 JsonToYaml.js # Function that parses and converts raw JSON into valid YAML\n\u2502 \u251c\u2500\u2500 KeycloakAuth.js # Singleton class to manage Keycloak authentication\n\u2502 \u251c\u2500\u2500 OidcAuth.js # Manages OIDC authentication with external providers\n\u2502 \u251c\u2500\u2500 HeaderAuth.js # Handles header-based authentication via reverse proxy\n\u2502 \u251c\u2500\u2500 languages.js # Handles fetching, switching and validating languages\n\u2502 \u2570\u2500\u2500 ThemeHelper.js # Function that handles the fetching and setting of user themes\n\u2570\u2500\u2500 views # Directory of available pages, corresponding to available routes\n \u251c\u2500\u2500 Home.vue # The home page container\n \u251c\u2500\u2500 About.vue # About page\n \u251c\u2500\u2500 Login.vue # Authentication page\n \u251c\u2500\u2500 Minimal.vue # The minimal view\n \u251c\u2500\u2500 Workspace.vue # The workspace view with apps in sidebar\n \u251c\u2500\u2500 DownloadConfig.vue # Config export page\n \u2570\u2500\u2500 404.vue # Not found page\n"})}),"\n",(0,t.jsx)(n.h4,{id:"visualisation-of-source-directory",children:"Visualisation of Source Directory"}),"\n",(0,t.jsx)(n.p,{children:(0,t.jsx)(n.img,{src:"https://raw.githubusercontent.com/Lissy93/dashy/master/docs/assets/repo-visualization.svg",alt:"File Breakdown"})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"development-tools",children:"Development Tools"}),"\n",(0,t.jsx)(n.h3,{id:"performance---lighthouse",children:"Performance - Lighthouse"}),"\n",(0,t.jsx)(n.p,{children:"The easiest method of checking performance is to use Chromium's build in auditing tool, Lighthouse. To run the test, open Developer Tools (usually F12) --\x3e Lighthouse and click on the 'Generate Report' button at the bottom."}),"\n",(0,t.jsx)(n.h3,{id:"dependencies---bundlephobia",children:"Dependencies - BundlePhobia"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.a,{href:"https://bundlephobia.com/",children:"BundlePhobia"})," is a really useful app that lets you analyze the cost of adding any particular dependency to an application"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"notes-1",children:"Notes"}),"\n",(0,t.jsx)(n.h3,{id:"known-warnings",children:"Known Warnings"}),"\n",(0,t.jsx)(n.p,{children:"When running the build command, several warnings appear. These are not errors, and do not affect the security or performance of the application. They will be addressed in a future update"}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"WARN A new version of sass-loader is available. Please upgrade for best experience."})," - Currently we're using an older version of SASS loader, since the more recent releases do not seem to be compatible with the Vue CLI's webpack configuration."]}),"\n",(0,t.jsxs)(n.p,{children:[(0,t.jsx)(n.code,{children:"WARN asset size limit: The following asset(s) exceed the recommended size limit (244 KiB)."})," - For the PWA to support Windows 10, a splash screen asset is required, and is quite large. This throws a warning, however PWA assets are not loaded until needed, so shouldn't have any impact on application performance. A similar warning is thrown for the Raleway font, and that is looking to be addressed."]})]})}function h(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,t.jsx)(n,{...e,children:(0,t.jsx)(c,{...e})}):c(e)}},8453(e,n,i){i.d(n,{R:()=>r,x:()=>l});var s=i(6540);const t={},o=s.createContext(t);function r(e){const n=s.useContext(o);return s.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function l(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),s.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/72e14192.fb9e9c93.js b/assets/js/72e14192.2af0d362.js similarity index 99% rename from assets/js/72e14192.fb9e9c93.js rename to assets/js/72e14192.2af0d362.js index 666d0ded..7507bc57 100644 --- a/assets/js/72e14192.fb9e9c93.js +++ b/assets/js/72e14192.2af0d362.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[2814],{1953(e,s,n){n.r(s),n.d(s,{assets:()=>l,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>c});const i=JSON.parse('{"id":"quick-start","title":"Quick Start","description":"Welcome to Dashy! So glad you\'re here \ud83d\ude0a In a couple of minutes, you\'ll have your new dashboard up and running \ud83d\ude80","source":"@site/docs/quick-start.md","sourceDirName":".","slug":"/quick-start","permalink":"/docs/quick-start","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/quick-start.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","next":{"title":"Deployment","permalink":"/docs/deployment"}}');var t=n(4848),o=n(8453);const r={},a="Quick Start",l={},c=[{value:"1. Prerequisites",id:"1-prerequisites",level:2},{value:"2. Installation",id:"2-installation",level:2},{value:"3. User Data Directory",id:"3-user-data-directory",level:2},{value:"4. Configure",id:"4-configure",level:2},{value:"5. Further Customisation",id:"5-further-customisation",level:2},{value:"6. Final Note",id:"6-final-note",level:2},{value:"Alternative Deployment Method 1 - From Source",id:"alternative-deployment-method-1---from-source",level:2},{value:"Alternative Deployment Method 2 - Netlify",id:"alternative-deployment-method-2---netlify",level:2},{value:"Alternative Deployment Method 3 - Cloud Services",id:"alternative-deployment-method-3---cloud-services",level:2}];function d(e){const s={a:"a",admonition:"admonition",br:"br",code:"code",h1:"h1",h2:"h2",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(s.header,{children:(0,t.jsx)(s.h1,{id:"quick-start",children:"Quick Start"})}),"\n",(0,t.jsx)(s.p,{children:"Welcome to Dashy! So glad you're here \ud83d\ude0a In a couple of minutes, you'll have your new dashboard up and running \ud83d\ude80"}),"\n",(0,t.jsxs)(s.p,{children:[(0,t.jsx)(s.strong,{children:"TLDR;"})," Run ",(0,t.jsx)(s.code,{children:"docker run -p 8080:8080 lissy93/dashy"}),", then open ",(0,t.jsx)(s.code,{children:"http://localhost:8080"})]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"1-prerequisites",children:"1. Prerequisites"}),"\n",(0,t.jsxs)(s.p,{children:["The quickest and easiest method of running Dashy is using Docker (or another container engine). You can find installation instructions for your system in the ",(0,t.jsx)(s.a,{href:"https://docs.docker.com/get-docker/",children:"Docker Documentation"}),".\nIf you don't want to use Docker, then you can use one of Dashy's other supported installation methods instead, all of which are outlined in the ",(0,t.jsx)(s.a,{href:"/docs/deployment",children:"Deployment Docs"}),"."]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"2-installation",children:"2. Installation"}),"\n",(0,t.jsx)(s.p,{children:"To pull the latest image, and build and start the app run:"}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/dashy-data:/app/user-data \\\n --name dashy \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,t.jsxs)(s.p,{children:["Your dashboard should now be up and running at ",(0,t.jsx)(s.code,{children:"http://localhost:8080"})," (or your servers IP address/ domain, and the port that you chose) \ud83c\udf89"]}),"\n",(0,t.jsxs)(s.p,{children:["Dashy is also available via GHCR (",(0,t.jsx)(s.code,{children:"ghcr.io/lissy93/dashy"}),").",(0,t.jsx)(s.br,{}),"\nYou can either use ",(0,t.jsx)(s.code,{children:":latest"})," or pin to specific versions (like ",(0,t.jsx)(s.code,{children:"4.0.0"}),").",(0,t.jsx)(s.br,{}),"\nAll images are multi-arch (works on amd64, arm64, and arm/v7).",(0,t.jsx)(s.br,{}),"\nTo use with compose, see our sample ",(0,t.jsx)(s.a,{href:"https://github.com/lissy93/dashy/blob/master/docker-compose.yml",children:(0,t.jsx)(s.code,{children:"docker-compose.yml"})}),".",(0,t.jsx)(s.br,{}),"\nOnce up and running, check the ",(0,t.jsx)(s.a,{href:"https://dashy.to/docs/configuring",children:"configuring reference"})," and ",(0,t.jsx)(s.a,{href:"https://dashy.to/docs",children:"other docs"}),".",(0,t.jsx)(s.br,{})]}),"\n",(0,t.jsx)(s.admonition,{type:"note",children:(0,t.jsxs)(s.p,{children:["You need to mount a directory for your Dashy settings in ",(0,t.jsx)(s.code,{children:"/app/user-data"}),".\nThe only required file here is the ",(0,t.jsx)(s.code,{children:"conf.yml"}),", but this is also where you can put any other page configs and assets like images/icons, stylesheets, fonts, etc.\nEverything in this directory is served from Dashy's root (e.g. ",(0,t.jsx)(s.code,{children:"/app/user-data/logo.png"})," will be accessible at ",(0,t.jsx)(s.code,{children:"http://[dashy.local]/logo.png"}),")."]})}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"3-user-data-directory",children:"3. User Data Directory"}),"\n",(0,t.jsxs)(s.p,{children:["Your config file should be placed inside ",(0,t.jsx)(s.code,{children:"user-data"})," (in Docker, that's ",(0,t.jsx)(s.code,{children:"/app/user-data/"}),")."]}),"\n",(0,t.jsx)(s.p,{children:"This directory can also contain some optional assets you wish to use within your dashboard, like icons, fonts, styles, scripts, etc."}),"\n",(0,t.jsxs)(s.p,{children:["Any files placed here will be served up to the root of the domain, and override the contents of ",(0,t.jsx)(s.code,{children:"public/"}),".\nFor example, if you had ",(0,t.jsx)(s.code,{children:"user-data/favicon.ico"})," this would be accessible at ",(0,t.jsx)(s.code,{children:"http://my-dashy-instance.local/favicon.ico"})]}),"\n",(0,t.jsxs)(s.p,{children:["Example Files in ",(0,t.jsx)(s.code,{children:"user-data"}),":"]}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"conf.yml"})," - This is the only file that is compulsory, it's your main Dashy config"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"**.yml"})," - Include more config files, if you'd like to have multiple pages, see ",(0,t.jsx)(s.a,{href:"/docs/pages-and-sections#multi-page-support",children:"Multi-page support"})," for docs"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"favicon.ico"})," - The default favicon, shown in the browser's tab title"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"initialization.html"})," - Static HTML page displayed before the app has finished compiling, see ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/public/initialization.html",children:(0,t.jsx)(s.code,{children:"public/initialization.html"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"robots.txt"})," - Search engine crawl rules, override this if you want your dashboard to be indexable"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"manifest.json"})," - PWA configuration file, for installing Dashy on mobile devices"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"index.html"})," - The main index page which initializes the client-side app, copy it from ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/public/index.html",children:(0,t.jsx)(s.code,{children:"/public/index.html"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"**.html"})," - Write your own HTML pages, and access them at ",(0,t.jsx)(s.code,{children:"http://my-dashy-instance.local/my-page.html"})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"fonts/"})," - Custom fonts (be sure to include the ones already in ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/tree/master/public/fonts",children:(0,t.jsx)(s.code,{children:"public/fonts"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"item-icons/"})," - To use your own icons for items on your dashboard, see ",(0,t.jsx)(s.a,{href:"/docs/icons#local-icons",children:"Icons --\x3e Local Icons"})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"web-icons/"})," - Override Dashy logo"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"widget-resources/"})," - Fonts, icons and assets for custom widgets"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"4-configure",children:"4. Configure"}),"\n",(0,t.jsxs)(s.p,{children:["Now that you've got Dashy running, you are going to want to set it up with your own content.\nConfig is written in ",(0,t.jsx)(s.a,{href:"https://yaml.org/",children:"YAML Format"}),", and saved in ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml",children:(0,t.jsx)(s.code,{children:"/user-data/conf.yml"})}),".\nThe format on the config file is pretty straight forward. There are four root attributes:"]}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#pageinfo",children:(0,t.jsx)(s.code,{children:"pageInfo"})})," - Dashboard meta data, like title, description, nav bar links and footer text"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#appconfig-optional",children:(0,t.jsx)(s.code,{children:"appConfig"})})," - Dashboard settings, like themes, authentication, language and customization"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#section",children:(0,t.jsx)(s.code,{children:"sections"})})," - An array of sections, each including an array of items"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#pages-optional",children:(0,t.jsx)(s.code,{children:"pages"})})," - Have multiples pages in your dashboard"]}),"\n"]}),"\n",(0,t.jsxs)(s.p,{children:["You can view a full list of all available config options in the ",(0,t.jsx)(s.a,{href:"/docs/configuring",children:"Configuring Docs"}),"."]}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-yaml",children:"pageInfo:\n title: Home Lab\nsections: # An array of sections\n- name: Example Section\n icon: far fa-rocket\n items:\n - title: GitHub\n description: Dashy source code and docs\n icon: fab fa-github\n url: https://github.com/Lissy93/dashy\n - title: Issues\n description: View open issues, or raise a new one\n icon: fas fa-bug\n url: https://github.com/Lissy93/dashy/issues\n- name: Local Services\n items:\n - title: Firewall\n icon: favicon\n url: http://192.168.1.1/\n - title: Game Server\n icon: https://i.ibb.co/710B3Yc/space-invader-x256.png\n url: http://192.168.130.1/\n"})}),"\n",(0,t.jsx)(s.p,{children:"Notes:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:["You can use a Docker volume to pass your ",(0,t.jsx)(s.code,{children:"user-data"})," directory from the host into the container\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:["E.g. ",(0,t.jsx)(s.code,{children:"-v ./host-system/user-data:/app/user-data"})]}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(s.li,{children:"It's also possible to edit your config directly through the UI, and changes will be saved in this file"}),"\n",(0,t.jsxs)(s.li,{children:["Check your config against Dashy's schema, with ",(0,t.jsx)(s.code,{children:"docker exec -it [container-id] yarn validate-config"})]}),"\n",(0,t.jsxs)(s.li,{children:["You might find it helpful to look at some examples, a collection of which can be ",(0,t.jsx)(s.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"found here"})]}),"\n",(0,t.jsx)(s.li,{children:"It's also possible to load a remote config, e.g. from a GitHub Gist"}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"5-further-customisation",children:"5. Further Customisation"}),"\n",(0,t.jsxs)(s.p,{children:["Once you've got Dashy setup, you'll want to ensure the container is properly healthy, secured, backed up and kept up-to-date. All this is covered in the ",(0,t.jsx)(s.a,{href:"/docs/management",children:"Management Docs"}),"."]}),"\n",(0,t.jsx)(s.p,{children:"You might also want to check out the docs for specific features you'd like to use:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/authentication",children:"Authentication"})," - Setting up authentication to protect your dashboard"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/alternate-views",children:"Alternate Views"})," - Using the startpage and workspace view"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/backup-restore",children:"Backup & Restore"})," - Guide to Dashy's cloud sync feature"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/icons",children:"Icons"})," - Outline of all available icon types for sections and items"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/multi-language-support",children:"Localisation"})," - How to change language, or add your own"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/status-indicators",children:"Status Indicators"})," - Using Dashy to monitor uptime and status of your apps"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/searching",children:"Search & Shortcuts"})," - Using instant filter, web search and custom hotkeys"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/theming",children:"Theming"})," - Complete guide to applying, writing and modifying themes and styles"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"6-final-note",children:"6. Final Note"}),"\n",(0,t.jsxs)(s.p,{children:["If you need any help or support in getting Dashy running, head over to the ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/discussions",children:"Discussions"})," page. If you think you've found a bug, please do ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new/choose",children:"raise it"})," so it can be fixed. For contact options, see the ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/SUPPORT",children:"Support Page"}),"."]}),"\n",(0,t.jsxs)(s.p,{children:["If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the ",(0,t.jsx)(s.a,{href:"/docs/contributing",children:"Contributing Page"}),". Huge thanks to ",(0,t.jsx)(s.a,{href:"/docs/credits",children:"everyone"})," who has already helped out!"]}),"\n",(0,t.jsx)(s.p,{children:"Enjoy your dashboard :)"}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-1---from-source",children:"Alternative Deployment Method 1 - From Source"}),"\n",(0,t.jsxs)(s.p,{children:["You can also easily run the app on your system without Docker. For this ",(0,t.jsx)(s.a,{href:"https://git-scm.com/downloads",children:"Git"}),", ",(0,t.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"}),", and ",(0,t.jsx)(s.a,{href:"https://yarnpkg.com/",children:"Yarn"})," are required."]}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-bash",children:"git clone https://github.com/Lissy93/dashy.git && cd dashy\nyarn # Install dependencies\nyarn build # Build the app\nyarn start # Start the app\n"})}),"\n",(0,t.jsxs)(s.p,{children:["Then edit ",(0,t.jsx)(s.code,{children:"./user-data/conf.yml"})]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-2---netlify",children:"Alternative Deployment Method 2 - Netlify"}),"\n",(0,t.jsxs)(s.p,{children:["Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many ",(0,t.jsx)(s.a,{href:"/docs/deployment#deploy-to-cloud-service",children:"other cloud providers"}),"). All you need it a GitHub account."]}),"\n",(0,t.jsxs)(s.ol,{children:["\n",(0,t.jsx)(s.li,{children:"Fork Dashy's repository on GitHub"}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"https://app.netlify.com/login/",children:"Log in"})," to Netlify with GitHub"]}),"\n",(0,t.jsxs)(s.li,{children:['Click "New site from Git" and select your forked repo, then click ',(0,t.jsx)(s.strong,{children:"Deploy"}),"!"]}),"\n",(0,t.jsxs)(s.li,{children:["You can then edit the config in ",(0,t.jsx)(s.code,{children:"./user-data/conf.yml"})," in your repo, and Netlify will rebuild the app"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-3---cloud-services",children:"Alternative Deployment Method 3 - Cloud Services"}),"\n",(0,t.jsx)(s.p,{children:"Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/ZxtzrP3/netlify.png",width:"18"})," Deploy to Netlify"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://heroku.com/deploy?template=https://github.com/Lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/d2P1WZ7/heroku.png",width:"18"})," Deploy to Heroku"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/Ld2FZzb/vercel.png",width:"18"})," Deploy to Vercel"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://render.com/deploy?repo=https://github.com/lissy93/dashy/tree/deploy_render",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/xCHtzgh/render.png",width:"18"})," Deploy to Render"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://deploy.cloud.run/?git_repo=https://github.com/lissy93/dashy.git",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/J7MGymY/googlecloud.png",width:"18"})," Deploy to GCP"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/HVWVYF7/docker.png",width:"18"})," Deploy to PWD"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://easypanel.io/docs/templates/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/7NxnM2P/easypanel.png",width:"18"})," Deploy to Easypanel"]})}),"\n"]})]})}function h(e={}){const{wrapper:s}={...(0,o.R)(),...e.components};return s?(0,t.jsx)(s,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,s,n){n.d(s,{R:()=>r,x:()=>a});var i=n(6540);const t={},o=i.createContext(t);function r(e){const s=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function a(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:s},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[2814],{1953(e,s,n){n.r(s),n.d(s,{assets:()=>l,contentTitle:()=>a,default:()=>h,frontMatter:()=>r,metadata:()=>i,toc:()=>c});const i=JSON.parse('{"id":"quick-start","title":"Quick Start","description":"Welcome to Dashy! So glad you\'re here \ud83d\ude0a In a couple of minutes, you\'ll have your new dashboard up and running \ud83d\ude80","source":"@site/docs/quick-start.md","sourceDirName":".","slug":"/quick-start","permalink":"/docs/quick-start","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/quick-start.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","next":{"title":"Deployment","permalink":"/docs/deployment"}}');var t=n(4848),o=n(8453);const r={},a="Quick Start",l={},c=[{value:"1. Prerequisites",id:"1-prerequisites",level:2},{value:"2. Installation",id:"2-installation",level:2},{value:"3. User Data Directory",id:"3-user-data-directory",level:2},{value:"4. Configure",id:"4-configure",level:2},{value:"5. Further Customisation",id:"5-further-customisation",level:2},{value:"6. Final Note",id:"6-final-note",level:2},{value:"Alternative Deployment Method 1 - From Source",id:"alternative-deployment-method-1---from-source",level:2},{value:"Alternative Deployment Method 2 - Netlify",id:"alternative-deployment-method-2---netlify",level:2},{value:"Alternative Deployment Method 3 - Cloud Services",id:"alternative-deployment-method-3---cloud-services",level:2}];function d(e){const s={a:"a",admonition:"admonition",br:"br",code:"code",h1:"h1",h2:"h2",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(s.header,{children:(0,t.jsx)(s.h1,{id:"quick-start",children:"Quick Start"})}),"\n",(0,t.jsx)(s.p,{children:"Welcome to Dashy! So glad you're here \ud83d\ude0a In a couple of minutes, you'll have your new dashboard up and running \ud83d\ude80"}),"\n",(0,t.jsxs)(s.p,{children:[(0,t.jsx)(s.strong,{children:"TLDR;"})," Run ",(0,t.jsx)(s.code,{children:"docker run -p 8080:8080 lissy93/dashy"}),", then open ",(0,t.jsx)(s.code,{children:"http://localhost:8080"})]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"1-prerequisites",children:"1. Prerequisites"}),"\n",(0,t.jsxs)(s.p,{children:["The quickest and easiest method of running Dashy is using Docker (or another container engine). You can find installation instructions for your system in the ",(0,t.jsx)(s.a,{href:"https://docs.docker.com/get-docker/",children:"Docker Documentation"}),".\nIf you don't want to use Docker, then you can use one of Dashy's other supported installation methods instead, all of which are outlined in the ",(0,t.jsx)(s.a,{href:"/docs/deployment",children:"Deployment Docs"}),"."]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"2-installation",children:"2. Installation"}),"\n",(0,t.jsx)(s.p,{children:"To pull the latest image, and build and start the app run:"}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v ~/dashy-data:/app/user-data \\\n --name dashy \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,t.jsxs)(s.p,{children:["Your dashboard should now be up and running at ",(0,t.jsx)(s.code,{children:"http://localhost:8080"})," (or your servers IP address/ domain, and the port that you chose) \ud83c\udf89"]}),"\n",(0,t.jsxs)(s.p,{children:["Dashy is also available via GHCR (",(0,t.jsx)(s.code,{children:"ghcr.io/lissy93/dashy"}),").",(0,t.jsx)(s.br,{}),"\nYou can either use ",(0,t.jsx)(s.code,{children:":latest"})," or pin to specific versions (like ",(0,t.jsx)(s.code,{children:"4.0.0"}),").",(0,t.jsx)(s.br,{}),"\nAll images are multi-arch (works on amd64, arm64, and arm/v7).",(0,t.jsx)(s.br,{}),"\nTo use with compose, see our sample ",(0,t.jsx)(s.a,{href:"https://github.com/lissy93/dashy/blob/master/docker-compose.yml",children:(0,t.jsx)(s.code,{children:"docker-compose.yml"})}),".",(0,t.jsx)(s.br,{}),"\nOnce up and running, check the ",(0,t.jsx)(s.a,{href:"https://dashy.to/docs/configuring",children:"configuring reference"})," and ",(0,t.jsx)(s.a,{href:"https://dashy.to/docs",children:"other docs"}),".",(0,t.jsx)(s.br,{})]}),"\n",(0,t.jsx)(s.admonition,{type:"note",children:(0,t.jsxs)(s.p,{children:["You need to mount a directory for your Dashy settings in ",(0,t.jsx)(s.code,{children:"/app/user-data"}),".\nThe only required file here is the ",(0,t.jsx)(s.code,{children:"conf.yml"}),", but this is also where you can put any other page configs and assets like images/icons, stylesheets, fonts, etc.\nEverything in this directory is served from Dashy's root (e.g. ",(0,t.jsx)(s.code,{children:"/app/user-data/logo.png"})," will be accessible at ",(0,t.jsx)(s.code,{children:"http://[dashy.local]/logo.png"}),")."]})}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"3-user-data-directory",children:"3. User Data Directory"}),"\n",(0,t.jsxs)(s.p,{children:["Your config file should be placed inside ",(0,t.jsx)(s.code,{children:"user-data"})," (in Docker, that's ",(0,t.jsx)(s.code,{children:"/app/user-data/"}),")."]}),"\n",(0,t.jsx)(s.p,{children:"This directory can also contain some optional assets you wish to use within your dashboard, like icons, fonts, styles, scripts, etc."}),"\n",(0,t.jsxs)(s.p,{children:["Any files placed here will be served up to the root of the domain, and override the contents of ",(0,t.jsx)(s.code,{children:"public/"}),".\nFor example, if you had ",(0,t.jsx)(s.code,{children:"user-data/favicon.ico"})," this would be accessible at ",(0,t.jsx)(s.code,{children:"http://my-dashy-instance.local/favicon.ico"})]}),"\n",(0,t.jsxs)(s.p,{children:["Example Files in ",(0,t.jsx)(s.code,{children:"user-data"}),":"]}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"conf.yml"})," - This is the only file that is compulsory, it's your main Dashy config"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"**.yml"})," - Include more config files, if you'd like to have multiple pages, see ",(0,t.jsx)(s.a,{href:"/docs/pages-and-sections#multi-page-support",children:"Multi-page support"})," for docs"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"favicon.ico"})," - The default favicon, shown in the browser's tab title"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"initialization.html"})," - Static HTML page displayed before the app has finished compiling, see ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/public/initialization.html",children:(0,t.jsx)(s.code,{children:"public/initialization.html"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"robots.txt"})," - Search engine crawl rules, override this if you want your dashboard to be indexable"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"manifest.json"})," - PWA configuration file, for installing Dashy on mobile devices"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"index.html"})," - The main index page which initializes the client-side app, copy it from ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/public/index.html",children:(0,t.jsx)(s.code,{children:"/public/index.html"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"**.html"})," - Write your own HTML pages, and access them at ",(0,t.jsx)(s.code,{children:"http://my-dashy-instance.local/my-page.html"})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"fonts/"})," - Custom fonts (be sure to include the ones already in ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/tree/master/public/fonts",children:(0,t.jsx)(s.code,{children:"public/fonts"})})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"item-icons/"})," - To use your own icons for items on your dashboard, see ",(0,t.jsx)(s.a,{href:"/docs/icons#local-icons",children:"Icons --\x3e Local Icons"})]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"web-icons/"})," - Override Dashy logo"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.code,{children:"widget-resources/"})," - Fonts, icons and assets for custom widgets"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"4-configure",children:"4. Configure"}),"\n",(0,t.jsxs)(s.p,{children:["Now that you've got Dashy running, you are going to want to set it up with your own content.\nConfig is written in ",(0,t.jsx)(s.a,{href:"https://yaml.org/",children:"YAML Format"}),", and saved in ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml",children:(0,t.jsx)(s.code,{children:"/user-data/conf.yml"})}),".\nThe format on the config file is pretty straight forward. There are four root attributes:"]}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#pageinfo",children:(0,t.jsx)(s.code,{children:"pageInfo"})})," - Dashboard meta data, like title, description, nav bar links and footer text"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#appconfig-optional",children:(0,t.jsx)(s.code,{children:"appConfig"})})," - Dashboard settings, like themes, authentication, language and customization"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#section",children:(0,t.jsx)(s.code,{children:"sections"})})," - An array of sections, each including an array of items"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/configuring#pages-optional",children:(0,t.jsx)(s.code,{children:"pages"})})," - Have multiples pages in your dashboard"]}),"\n"]}),"\n",(0,t.jsxs)(s.p,{children:["You can view a full list of all available config options in the ",(0,t.jsx)(s.a,{href:"/docs/configuring",children:"Configuring Docs"}),"."]}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-yaml",children:"pageInfo:\n title: Home Lab\nsections: # An array of sections\n- name: Example Section\n icon: far fa-rocket\n items:\n - title: GitHub\n description: Dashy source code and docs\n icon: fab fa-github\n url: https://github.com/Lissy93/dashy\n - title: Issues\n description: View open issues, or raise a new one\n icon: fas fa-bug\n url: https://github.com/Lissy93/dashy/issues\n- name: Local Services\n items:\n - title: Firewall\n icon: favicon\n url: http://192.168.1.1/\n - title: Game Server\n icon: https://i.ibb.co/710B3Yc/space-invader-x256.png\n url: http://192.168.130.1/\n"})}),"\n",(0,t.jsx)(s.p,{children:"Notes:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:["You can use a Docker volume to pass your ",(0,t.jsx)(s.code,{children:"user-data"})," directory from the host into the container\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:["E.g. ",(0,t.jsx)(s.code,{children:"-v ./host-system/user-data:/app/user-data"})]}),"\n"]}),"\n"]}),"\n",(0,t.jsx)(s.li,{children:"It's also possible to edit your config directly through the UI, and changes will be saved in this file"}),"\n",(0,t.jsxs)(s.li,{children:["Check your config against Dashy's schema, with ",(0,t.jsx)(s.code,{children:"docker exec -it [container-id] yarn validate-config"})]}),"\n",(0,t.jsxs)(s.li,{children:["You might find it helpful to look at some examples, a collection of which can be ",(0,t.jsx)(s.a,{href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10",children:"found here"})]}),"\n",(0,t.jsx)(s.li,{children:"It's also possible to load a remote config, e.g. from a GitHub Gist"}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"5-further-customisation",children:"5. Further Customisation"}),"\n",(0,t.jsxs)(s.p,{children:["Once you've got Dashy setup, you'll want to ensure the container is properly healthy, secured, backed up and kept up-to-date. All this is covered in the ",(0,t.jsx)(s.a,{href:"/docs/management",children:"Management Docs"}),"."]}),"\n",(0,t.jsx)(s.p,{children:"You might also want to check out the docs for specific features you'd like to use:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/authentication",children:"Authentication"})," - Setting up authentication to protect your dashboard"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/alternate-views",children:"Alternate Views"})," - Using the startpage and workspace view"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/backup-restore",children:"Backup & Restore"})," - Guide to Dashy's cloud sync feature"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/icons",children:"Icons"})," - Outline of all available icon types for sections and items"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/multi-language-support",children:"Localisation"})," - How to change language, or add your own"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/status-indicators",children:"Status Indicators"})," - Using Dashy to monitor uptime and status of your apps"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/searching",children:"Search & Shortcuts"})," - Using instant filter, web search and custom hotkeys"]}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"/docs/theming",children:"Theming"})," - Complete guide to applying, writing and modifying themes and styles"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"6-final-note",children:"6. Final Note"}),"\n",(0,t.jsxs)(s.p,{children:["If you need any help or support in getting Dashy running, head over to the ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/discussions",children:"Discussions"})," page. If you think you've found a bug, please do ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/issues/new/choose",children:"raise it"})," so it can be fixed. For contact options, see the ",(0,t.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/SUPPORT",children:"Support Page"}),"."]}),"\n",(0,t.jsxs)(s.p,{children:["If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the ",(0,t.jsx)(s.a,{href:"/docs/contributing",children:"Contributing Page"}),". Huge thanks to ",(0,t.jsx)(s.a,{href:"/docs/credits",children:"everyone"})," who has already helped out!"]}),"\n",(0,t.jsx)(s.p,{children:"Enjoy your dashboard :)"}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-1---from-source",children:"Alternative Deployment Method 1 - From Source"}),"\n",(0,t.jsxs)(s.p,{children:["You can also easily run the app on your system without Docker. For this ",(0,t.jsx)(s.a,{href:"https://git-scm.com/downloads",children:"Git"}),", ",(0,t.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"}),", and ",(0,t.jsx)(s.a,{href:"https://yarnpkg.com/",children:"Yarn"})," are required."]}),"\n",(0,t.jsx)(s.pre,{children:(0,t.jsx)(s.code,{className:"language-bash",children:"git clone https://github.com/Lissy93/dashy.git && cd dashy\nyarn # Install dependencies\nyarn build # Build the app\nyarn start # Start the app\n"})}),"\n",(0,t.jsxs)(s.p,{children:["Then edit ",(0,t.jsx)(s.code,{children:"./user-data/conf.yml"})]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-2---netlify",children:"Alternative Deployment Method 2 - Netlify"}),"\n",(0,t.jsxs)(s.p,{children:["Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many ",(0,t.jsx)(s.a,{href:"/docs/deployment#deploy-to-cloud-service",children:"other cloud providers"}),"). All you need it a GitHub account."]}),"\n",(0,t.jsxs)(s.ol,{children:["\n",(0,t.jsx)(s.li,{children:"Fork Dashy's repository on GitHub"}),"\n",(0,t.jsxs)(s.li,{children:[(0,t.jsx)(s.a,{href:"https://app.netlify.com/login/",children:"Log in"})," to Netlify with GitHub"]}),"\n",(0,t.jsxs)(s.li,{children:['Click "New site from Git" and select your forked repo, then click ',(0,t.jsx)(s.strong,{children:"Deploy"}),"!"]}),"\n",(0,t.jsxs)(s.li,{children:["You can then edit the config in ",(0,t.jsx)(s.code,{children:"./user-data/conf.yml"})," in your repo, and Netlify will rebuild the app"]}),"\n"]}),"\n",(0,t.jsx)(s.hr,{}),"\n",(0,t.jsx)(s.h2,{id:"alternative-deployment-method-3---cloud-services",children:"Alternative Deployment Method 3 - Cloud Services"}),"\n",(0,t.jsx)(s.p,{children:"Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:"}),"\n",(0,t.jsxs)(s.ul,{children:["\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/ZxtzrP3/netlify.png",width:"18"})," Deploy to Netlify"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://heroku.com/deploy?template=https://github.com/Lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/d2P1WZ7/heroku.png",width:"18"})," Deploy to Heroku"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/Ld2FZzb/vercel.png",width:"18"})," Deploy to Vercel"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://render.com/deploy?repo=https://github.com/lissy93/dashy/tree/deploy_render",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/xCHtzgh/render.png",width:"18"})," Deploy to Render"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://deploy.cloud.run/?git_repo=https://github.com/lissy93/dashy.git",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/J7MGymY/googlecloud.png",width:"18"})," Deploy to GCP"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/HVWVYF7/docker.png",width:"18"})," Deploy to PWD"]})}),"\n",(0,t.jsx)(s.li,{children:(0,t.jsxs)(s.a,{href:"https://easypanel.io/docs/templates/dashy",children:[(0,t.jsx)(s.img,{src:"https://i.ibb.co/7NxnM2P/easypanel.png",width:"18"})," Deploy to Easypanel"]})}),"\n"]})]})}function h(e={}){const{wrapper:s}={...(0,o.R)(),...e.components};return s?(0,t.jsx)(s,{...e,children:(0,t.jsx)(d,{...e})}):d(e)}},8453(e,s,n){n.d(s,{R:()=>r,x:()=>a});var i=n(6540);const t={},o=i.createContext(t);function r(e){const s=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function a(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(t):e.components||t:r(e.components),i.createElement(o.Provider,{value:s},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/87124b4e.ffca39a4.js b/assets/js/87124b4e.e7c7056b.js similarity index 99% rename from assets/js/87124b4e.ffca39a4.js rename to assets/js/87124b4e.e7c7056b.js index 86d649bd..b7436e07 100644 --- a/assets/js/87124b4e.ffca39a4.js +++ b/assets/js/87124b4e.e7c7056b.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[8278],{8030(e,t,s){s.r(t),s.d(t,{assets:()=>c,contentTitle:()=>r,default:()=>h,frontMatter:()=>o,metadata:()=>n,toc:()=>l});const n=JSON.parse('{"id":"status-indicators","title":"Status Indicators","description":"Dashy has an optional feature that can display a small icon next to each of your running services, indicating it\'s current status. This can be useful if you are using Dashy as your homelab\'s start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message.","source":"@site/docs/status-indicators.md","sourceDirName":".","slug":"/status-indicators","permalink":"/docs/status-indicators","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/status-indicators.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Theming","permalink":"/docs/theming"},"next":{"title":"Authentication","permalink":"/docs/authentication"}}');var i=s(4848),a=s(8453);const o={},r="Status Indicators",c={},l=[{value:"Enabling Status Indicators",id:"enabling-status-indicators",level:2},{value:"Continuous Checking",id:"continuous-checking",level:2},{value:"Using a Different Endpoint",id:"using-a-different-endpoint",level:2},{value:"Setting Custom Headers",id:"setting-custom-headers",level:2},{value:"Disabling Security",id:"disabling-security",level:2},{value:"Allowing Alternative Status Codes",id:"allowing-alternative-status-codes",level:2},{value:"Troubleshooting Failing Status Checks",id:"troubleshooting-failing-status-checks",level:2},{value:"Color-Blind Accessibility",id:"color-blind-accessibility",level:2},{value:"How it Works",id:"how-it-works",level:2}];function d(e){const t={a:"a",code:"code",h1:"h1",h2:"h2",header:"header",img:"img",p:"p",pre:"pre",...(0,a.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(t.header,{children:(0,i.jsx)(t.h1,{id:"status-indicators",children:"Status Indicators"})}),"\n",(0,i.jsx)(t.p,{children:"Dashy has an optional feature that can display a small icon next to each of your running services, indicating it's current status. This can be useful if you are using Dashy as your homelab's start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message."}),"\n ",(0,i.jsx)(t.img,{width:"800",src:"/docs/assets/status-check-demo.gif"}),"\n",(0,i.jsx)(t.h2,{id:"enabling-status-indicators",children:"Enabling Status Indicators"}),"\n",(0,i.jsxs)(t.p,{children:["By default, this feature is off. If you do not want this feature, just don't add the ",(0,i.jsx)(t.code,{children:"statusCheck"})," to your conf.yml file, then no requests will be made."]}),"\n",(0,i.jsxs)(t.p,{children:["To enable status checks, you can either turn it on for all items, by setting ",(0,i.jsx)(t.code,{children:"appConfig.statusCheck: true"}),", like:"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"appConfig:\n statusCheck: true\n"})}),"\n",(0,i.jsxs)(t.p,{children:["Or you can enable/ disable it on a per-item basis, with the ",(0,i.jsx)(t.code,{children:"item[n].statusCheck"})," attribute"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"sections:\n- name: Firewall\n items:\n - title: OPNsense\n description: Firewall Central Management\n icon: networking/opnsense.png\n url: https://192.168.1.1\n statusCheck: false\n - title: MalTrail\n description: Malicious traffic detection system\n icon: networking/maltrail.png\n url: http://192.168.1.1:8338\n statusCheck: true\n - title: Ntopng\n description: Network traffic probe and network use monitor\n icon: networking/ntop.png\n url: http://192.168.1.1:3001\n statusCheck: true\n"})}),"\n",(0,i.jsx)(t.h2,{id:"continuous-checking",children:"Continuous Checking"}),"\n",(0,i.jsxs)(t.p,{children:["By default, with status indicators enabled Dashy will check an applications status on page load, and will not keep indicators updated. This is usually desirable behavior. However, if you do want the status indicators to continue to poll your running services, this can be enabled by setting the ",(0,i.jsx)(t.code,{children:"statusCheckInterval"})," attribute. Here you define an interval as an integer in seconds, and Dashy will poll your apps every x seconds. Note that if this number is very low (below 5 seconds), you may notice the app running slightly slower."]}),"\n",(0,i.jsx)(t.p,{children:"The following example, will instruct Dashy to continuously check the status of your services every 20 seconds."}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-text",children:"appConfig:\n statusCheck: true\n statusCheckInterval: 20\n"})}),"\n",(0,i.jsx)(t.h2,{id:"using-a-different-endpoint",children:"Using a Different Endpoint"}),"\n",(0,i.jsx)(t.p,{children:"By default, the status checker will use the URL of each application being checked. In some situations, you may want to use a different endpoint for status checking. Similarly, some services provide a dedicated path for uptime monitoring."}),"\n",(0,i.jsxs)(t.p,{children:["You can set the ",(0,i.jsx)(t.code,{children:"statusCheckUrl"})," property on any given item in order to do this. The status checker will then ping that endpoint, instead of the apps main ",(0,i.jsx)(t.code,{children:"url"})," property."]}),"\n",(0,i.jsx)(t.h2,{id:"setting-custom-headers",children:"Setting Custom Headers"}),"\n",(0,i.jsxs)(t.p,{children:["If your service is responding with an error, despite being up and running, it is most likely because custom headers for authentication, authorization or encoding are required. You can define these headers under the ",(0,i.jsx)(t.code,{children:"statusCheckHeaders"})," property for any service. It should be defined as an object format, with the name of header as the key, and header content as the value.\nFor example, ",(0,i.jsx)(t.code,{children:"statusCheckHeaders: { 'X-Custom-Header': 'foobar' }"})]}),"\n",(0,i.jsx)(t.h2,{id:"disabling-security",children:"Disabling Security"}),"\n",(0,i.jsxs)(t.p,{children:["By default, (if you're using HTTPS) any requests to insecure or non-HTTPS content will be blocked. This will cause the status check to fail. If you trust the endpoint (e.g. you're self-hosting it), then you can disable this security measure for an individual item. This is done by setting ",(0,i.jsx)(t.code,{children:"statusCheckAllowInsecure: true"})]}),"\n",(0,i.jsx)(t.h2,{id:"allowing-alternative-status-codes",children:"Allowing Alternative Status Codes"}),"\n",(0,i.jsxs)(t.p,{children:["If you expect your service to return a status code that is not in the 2XX range, and still want the indicator to be green, then you can specify an expected status code under ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes"})," for a given item. For example, ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes: '403,418'"})]}),"\n",(0,i.jsx)(t.h2,{id:"troubleshooting-failing-status-checks",children:"Troubleshooting Failing Status Checks"}),"\n",(0,i.jsx)(t.p,{children:"If you're using status checks, and despite a given service being online, the check is displaying an error, there are a couple of things you can look at:"}),"\n",(0,i.jsxs)(t.p,{children:["If your service requires requests to include any authorization in the headers, then use the ",(0,i.jsx)(t.code,{children:"statusCheckHeaders"})," property, as described ",(0,i.jsx)(t.a,{href:"#setting-custom-headers",children:"above"}),"."]}),"\n",(0,i.jsxs)(t.p,{children:["If you are still having issues, it may be because your target application is blocking requests from Dashy's IP. This is a ",(0,i.jsx)(t.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"CORS error"}),", and can be fixed by setting the headers on your target app, to include:"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,i.jsxs)(t.p,{children:["If the URL you are checking has an unsigned certificate, or is not using HTTPS, then you may need to disable the rejection of insecure requests. This can be done by setting ",(0,i.jsx)(t.code,{children:"statusCheckAllowInsecure"})," to true for a given item."]}),"\n",(0,i.jsxs)(t.p,{children:["If your service is online, but responds with a status code that is not in the 2xx range, then you can use ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes"})," to set an accepted status code."]}),"\n",(0,i.jsxs)(t.p,{children:["If you get an error, like ",(0,i.jsx)(t.code,{children:"Service Unavailable: Server resulted in a fatal error"}),", even when it's definitely online, this is most likely caused by missing the protocol. Don't forget to include ",(0,i.jsx)(t.code,{children:"https://"})," (or whatever protocol) before the URL, and ensure that if needed, you've specified the port."]}),"\n",(0,i.jsxs)(t.p,{children:["Running Dashy in HOST network mode, instead of BRIDGE will allow status check access to other services in HOST mode. For more info, see ",(0,i.jsx)(t.a,{href:"https://github.com/Lissy93/dashy/discussions/445",children:"#445"}),"."]}),"\n",(0,i.jsx)(t.p,{children:"If you have firewall rules configured, then ensure that they don't prevent Dashy from making requests to the other services you are trying to access."}),"\n",(0,i.jsxs)(t.p,{children:["Currently, the status check needs a page to be rendered, so if this URL in your browser does not return anything, then status checks will not work. This may be modified in the future, but in the meantime, a fix would be to make your own status service, which just checks if your app responds with whatever code you'd like, and then return a 200 plus renders an arbitrary message. Then just point ",(0,i.jsx)(t.code,{children:"statusCheckUrl"})," to your custom page."]}),"\n",(0,i.jsxs)(t.p,{children:["For further troubleshooting, use an application like ",(0,i.jsx)(t.a,{href:"https://postman.com",children:"Postman"})," to diagnose the issue. Set the parameter to ",(0,i.jsx)(t.code,{children:"GET"}),", and then make a call to: ",(0,i.jsx)(t.code,{children:"https://[url-of-dashy]/status-check/?&url=[service-url]"}),". Where the service URL must have first been encoded (e.g. with ",(0,i.jsx)(t.code,{children:"encodeURIComponent()"})," or ",(0,i.jsx)(t.a,{href:"https://www.urlencoder.io/",children:"urlencoder.io"}),")"]}),"\n",(0,i.jsx)(t.p,{children:"If you're serving Dashy though a CDN, instead of using the Node server or Docker image, then the Node endpoint that makes requests will not be available to you, and all requests will fail. A workaround for this may be implemented in the future, but in the meantime, your only option is to use the Docker or Node deployment method."}),"\n",(0,i.jsx)(t.h2,{id:"color-blind-accessibility",children:"Color-Blind Accessibility"}),"\n",(0,i.jsxs)(t.p,{children:["The default indicators rely on color alone to illustrate status (green, yellow, red and grey). For greater visual distinction and colorblind support, the accessibility mode will use shapes instead of dots to show status. This can be enabled by setting ",(0,i.jsx)(t.code,{children:"appConfig.statusCheckAccessibility: true"}),"."]}),"\n",(0,i.jsx)(t.p,{children:"By default, this will be: circle (online), square (offline), triangle (checking) and diamond (timed out)."}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"appConfig:\n statusCheck: true\n statusCheckAccessibility: true\n"})}),"\n",(0,i.jsxs)(t.p,{children:["Alternatively, you can set your own colors for status indicators, with these variables: ",(0,i.jsx)(t.code,{children:"--success"}),", ",(0,i.jsx)(t.code,{children:"--warning"}),", ",(0,i.jsx)(t.code,{children:"--danger"}),", see: ",(0,i.jsx)(t.a,{href:"/docs/theming#modifying-theme-colors",children:"modifying theme colors"}),"."]}),"\n",(0,i.jsx)(t.h2,{id:"how-it-works",children:"How it Works"}),"\n",(0,i.jsxs)(t.p,{children:["When the app is loaded, if ",(0,i.jsx)(t.code,{children:"appConfig.statusCheck: true"})," is set, or if any items have the ",(0,i.jsx)(t.code,{children:"statusCheck: true"})," enabled, then Dashy will make a request, to ",(0,i.jsx)(t.code,{children:"https://[your-host-name]/status-check?url=[address-or-servce]"})," (may al include GET params for headers and the secure flag), which in turn will ping that running service, and respond with a status code. Response time is calculated from the difference between start and end time of the request."]}),"\n",(0,i.jsx)(t.p,{children:"When the response completes, an indicator will display next to each item. The color denotes the status: Yellow while waiting for the response to return, green if request was successful, red if it failed, and grey if it was unable to make the request all together."}),"\n",(0,i.jsxs)(t.p,{children:["All requests are made straight from your server, there is no intermediary. So providing you are hosting Dashy yourself, and are checking the status of other self-hosted services, there shouldn't be any privacy concerns. Requests are made asynchronously, so this won't have any significant impact on page load speeds. However recurring requests (using ",(0,i.jsx)(t.code,{children:"statusCheckInterval"}),") may run more slowly if the interval between requests is very short."]})]})}function h(e={}){const{wrapper:t}={...(0,a.R)(),...e.components};return t?(0,i.jsx)(t,{...e,children:(0,i.jsx)(d,{...e})}):d(e)}},8453(e,t,s){s.d(t,{R:()=>o,x:()=>r});var n=s(6540);const i={},a=n.createContext(i);function o(e){const t=n.useContext(a);return n.useMemo(function(){return"function"==typeof e?e(t):{...t,...e}},[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),n.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[8278],{8030(e,t,s){s.r(t),s.d(t,{assets:()=>c,contentTitle:()=>r,default:()=>h,frontMatter:()=>o,metadata:()=>n,toc:()=>l});const n=JSON.parse('{"id":"status-indicators","title":"Status Indicators","description":"Dashy has an optional feature that can display a small icon next to each of your running services, indicating it\'s current status. This can be useful if you are using Dashy as your homelab\'s start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message.","source":"@site/docs/status-indicators.md","sourceDirName":".","slug":"/status-indicators","permalink":"/docs/status-indicators","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/status-indicators.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Theming","permalink":"/docs/theming"},"next":{"title":"Authentication","permalink":"/docs/authentication"}}');var i=s(4848),a=s(8453);const o={},r="Status Indicators",c={},l=[{value:"Enabling Status Indicators",id:"enabling-status-indicators",level:2},{value:"Continuous Checking",id:"continuous-checking",level:2},{value:"Using a Different Endpoint",id:"using-a-different-endpoint",level:2},{value:"Setting Custom Headers",id:"setting-custom-headers",level:2},{value:"Disabling Security",id:"disabling-security",level:2},{value:"Allowing Alternative Status Codes",id:"allowing-alternative-status-codes",level:2},{value:"Troubleshooting Failing Status Checks",id:"troubleshooting-failing-status-checks",level:2},{value:"Color-Blind Accessibility",id:"color-blind-accessibility",level:2},{value:"How it Works",id:"how-it-works",level:2}];function d(e){const t={a:"a",code:"code",h1:"h1",h2:"h2",header:"header",img:"img",p:"p",pre:"pre",...(0,a.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(t.header,{children:(0,i.jsx)(t.h1,{id:"status-indicators",children:"Status Indicators"})}),"\n",(0,i.jsx)(t.p,{children:"Dashy has an optional feature that can display a small icon next to each of your running services, indicating it's current status. This can be useful if you are using Dashy as your homelab's start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message."}),"\n ",(0,i.jsx)(t.img,{width:"800",src:"/docs/assets/status-check-demo.gif"}),"\n",(0,i.jsx)(t.h2,{id:"enabling-status-indicators",children:"Enabling Status Indicators"}),"\n",(0,i.jsxs)(t.p,{children:["By default, this feature is off. If you do not want this feature, just don't add the ",(0,i.jsx)(t.code,{children:"statusCheck"})," to your conf.yml file, then no requests will be made."]}),"\n",(0,i.jsxs)(t.p,{children:["To enable status checks, you can either turn it on for all items, by setting ",(0,i.jsx)(t.code,{children:"appConfig.statusCheck: true"}),", like:"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"appConfig:\n statusCheck: true\n"})}),"\n",(0,i.jsxs)(t.p,{children:["Or you can enable/ disable it on a per-item basis, with the ",(0,i.jsx)(t.code,{children:"item[n].statusCheck"})," attribute"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"sections:\n- name: Firewall\n items:\n - title: OPNsense\n description: Firewall Central Management\n icon: networking/opnsense.png\n url: https://192.168.1.1\n statusCheck: false\n - title: MalTrail\n description: Malicious traffic detection system\n icon: networking/maltrail.png\n url: http://192.168.1.1:8338\n statusCheck: true\n - title: Ntopng\n description: Network traffic probe and network use monitor\n icon: networking/ntop.png\n url: http://192.168.1.1:3001\n statusCheck: true\n"})}),"\n",(0,i.jsx)(t.h2,{id:"continuous-checking",children:"Continuous Checking"}),"\n",(0,i.jsxs)(t.p,{children:["By default, with status indicators enabled Dashy will check an applications status on page load, and will not keep indicators updated. This is usually desirable behavior. However, if you do want the status indicators to continue to poll your running services, this can be enabled by setting the ",(0,i.jsx)(t.code,{children:"statusCheckInterval"})," attribute. Here you define an interval as an integer in seconds, and Dashy will poll your apps every x seconds. Note that if this number is very low (below 5 seconds), you may notice the app running slightly slower."]}),"\n",(0,i.jsx)(t.p,{children:"The following example, will instruct Dashy to continuously check the status of your services every 20 seconds."}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-text",children:"appConfig:\n statusCheck: true\n statusCheckInterval: 20\n"})}),"\n",(0,i.jsx)(t.h2,{id:"using-a-different-endpoint",children:"Using a Different Endpoint"}),"\n",(0,i.jsx)(t.p,{children:"By default, the status checker will use the URL of each application being checked. In some situations, you may want to use a different endpoint for status checking. Similarly, some services provide a dedicated path for uptime monitoring."}),"\n",(0,i.jsxs)(t.p,{children:["You can set the ",(0,i.jsx)(t.code,{children:"statusCheckUrl"})," property on any given item in order to do this. The status checker will then ping that endpoint, instead of the apps main ",(0,i.jsx)(t.code,{children:"url"})," property."]}),"\n",(0,i.jsx)(t.h2,{id:"setting-custom-headers",children:"Setting Custom Headers"}),"\n",(0,i.jsxs)(t.p,{children:["If your service is responding with an error, despite being up and running, it is most likely because custom headers for authentication, authorization or encoding are required. You can define these headers under the ",(0,i.jsx)(t.code,{children:"statusCheckHeaders"})," property for any service. It should be defined as an object format, with the name of header as the key, and header content as the value.\nFor example, ",(0,i.jsx)(t.code,{children:"statusCheckHeaders: { 'X-Custom-Header': 'foobar' }"})]}),"\n",(0,i.jsx)(t.h2,{id:"disabling-security",children:"Disabling Security"}),"\n",(0,i.jsxs)(t.p,{children:["By default, (if you're using HTTPS) any requests to insecure or non-HTTPS content will be blocked. This will cause the status check to fail. If you trust the endpoint (e.g. you're self-hosting it), then you can disable this security measure for an individual item. This is done by setting ",(0,i.jsx)(t.code,{children:"statusCheckAllowInsecure: true"})]}),"\n",(0,i.jsx)(t.h2,{id:"allowing-alternative-status-codes",children:"Allowing Alternative Status Codes"}),"\n",(0,i.jsxs)(t.p,{children:["If you expect your service to return a status code that is not in the 2XX range, and still want the indicator to be green, then you can specify an expected status code under ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes"})," for a given item. For example, ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes: '403,418'"})]}),"\n",(0,i.jsx)(t.h2,{id:"troubleshooting-failing-status-checks",children:"Troubleshooting Failing Status Checks"}),"\n",(0,i.jsx)(t.p,{children:"If you're using status checks, and despite a given service being online, the check is displaying an error, there are a couple of things you can look at:"}),"\n",(0,i.jsxs)(t.p,{children:["If your service requires requests to include any authorization in the headers, then use the ",(0,i.jsx)(t.code,{children:"statusCheckHeaders"})," property, as described ",(0,i.jsx)(t.a,{href:"#setting-custom-headers",children:"above"}),"."]}),"\n",(0,i.jsxs)(t.p,{children:["If you are still having issues, it may be because your target application is blocking requests from Dashy's IP. This is a ",(0,i.jsx)(t.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"CORS error"}),", and can be fixed by setting the headers on your target app, to include:"]}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,i.jsxs)(t.p,{children:["If the URL you are checking has an unsigned certificate, or is not using HTTPS, then you may need to disable the rejection of insecure requests. This can be done by setting ",(0,i.jsx)(t.code,{children:"statusCheckAllowInsecure"})," to true for a given item."]}),"\n",(0,i.jsxs)(t.p,{children:["If your service is online, but responds with a status code that is not in the 2xx range, then you can use ",(0,i.jsx)(t.code,{children:"statusCheckAcceptCodes"})," to set an accepted status code."]}),"\n",(0,i.jsxs)(t.p,{children:["If you get an error, like ",(0,i.jsx)(t.code,{children:"Service Unavailable: Server resulted in a fatal error"}),", even when it's definitely online, this is most likely caused by missing the protocol. Don't forget to include ",(0,i.jsx)(t.code,{children:"https://"})," (or whatever protocol) before the URL, and ensure that if needed, you've specified the port."]}),"\n",(0,i.jsxs)(t.p,{children:["Running Dashy in HOST network mode, instead of BRIDGE will allow status check access to other services in HOST mode. For more info, see ",(0,i.jsx)(t.a,{href:"https://github.com/Lissy93/dashy/discussions/445",children:"#445"}),"."]}),"\n",(0,i.jsx)(t.p,{children:"If you have firewall rules configured, then ensure that they don't prevent Dashy from making requests to the other services you are trying to access."}),"\n",(0,i.jsxs)(t.p,{children:["Currently, the status check needs a page to be rendered, so if this URL in your browser does not return anything, then status checks will not work. This may be modified in the future, but in the meantime, a fix would be to make your own status service, which just checks if your app responds with whatever code you'd like, and then return a 200 plus renders an arbitrary message. Then just point ",(0,i.jsx)(t.code,{children:"statusCheckUrl"})," to your custom page."]}),"\n",(0,i.jsxs)(t.p,{children:["For further troubleshooting, use an application like ",(0,i.jsx)(t.a,{href:"https://postman.com",children:"Postman"})," to diagnose the issue. Set the parameter to ",(0,i.jsx)(t.code,{children:"GET"}),", and then make a call to: ",(0,i.jsx)(t.code,{children:"https://[url-of-dashy]/status-check/?&url=[service-url]"}),". Where the service URL must have first been encoded (e.g. with ",(0,i.jsx)(t.code,{children:"encodeURIComponent()"})," or ",(0,i.jsx)(t.a,{href:"https://www.urlencoder.io/",children:"urlencoder.io"}),")"]}),"\n",(0,i.jsx)(t.p,{children:"If you're serving Dashy though a CDN, instead of using the Node server or Docker image, then the Node endpoint that makes requests will not be available to you, and all requests will fail. A workaround for this may be implemented in the future, but in the meantime, your only option is to use the Docker or Node deployment method."}),"\n",(0,i.jsx)(t.h2,{id:"color-blind-accessibility",children:"Color-Blind Accessibility"}),"\n",(0,i.jsxs)(t.p,{children:["The default indicators rely on color alone to illustrate status (green, yellow, red and grey). For greater visual distinction and colorblind support, the accessibility mode will use shapes instead of dots to show status. This can be enabled by setting ",(0,i.jsx)(t.code,{children:"appConfig.statusCheckAccessibility: true"}),"."]}),"\n",(0,i.jsx)(t.p,{children:"By default, this will be: circle (online), square (offline), triangle (checking) and diamond (timed out)."}),"\n",(0,i.jsx)(t.pre,{children:(0,i.jsx)(t.code,{className:"language-yaml",children:"appConfig:\n statusCheck: true\n statusCheckAccessibility: true\n"})}),"\n",(0,i.jsxs)(t.p,{children:["Alternatively, you can set your own colors for status indicators, with these variables: ",(0,i.jsx)(t.code,{children:"--success"}),", ",(0,i.jsx)(t.code,{children:"--warning"}),", ",(0,i.jsx)(t.code,{children:"--danger"}),", see: ",(0,i.jsx)(t.a,{href:"/docs/theming#modifying-theme-colors",children:"modifying theme colors"}),"."]}),"\n",(0,i.jsx)(t.h2,{id:"how-it-works",children:"How it Works"}),"\n",(0,i.jsxs)(t.p,{children:["When the app is loaded, if ",(0,i.jsx)(t.code,{children:"appConfig.statusCheck: true"})," is set, or if any items have the ",(0,i.jsx)(t.code,{children:"statusCheck: true"})," enabled, then Dashy will make a request, to ",(0,i.jsx)(t.code,{children:"https://[your-host-name]/status-check?url=[address-or-servce]"})," (may al include GET params for headers and the secure flag), which in turn will ping that running service, and respond with a status code. Response time is calculated from the difference between start and end time of the request."]}),"\n",(0,i.jsx)(t.p,{children:"When the response completes, an indicator will display next to each item. The color denotes the status: Yellow while waiting for the response to return, green if request was successful, red if it failed, and grey if it was unable to make the request all together."}),"\n",(0,i.jsxs)(t.p,{children:["All requests are made straight from your server, there is no intermediary. So providing you are hosting Dashy yourself, and are checking the status of other self-hosted services, there shouldn't be any privacy concerns. Requests are made asynchronously, so this won't have any significant impact on page load speeds. However recurring requests (using ",(0,i.jsx)(t.code,{children:"statusCheckInterval"}),") may run more slowly if the interval between requests is very short."]})]})}function h(e={}){const{wrapper:t}={...(0,a.R)(),...e.components};return t?(0,i.jsx)(t,{...e,children:(0,i.jsx)(d,{...e})}):d(e)}},8453(e,t,s){s.d(t,{R:()=>o,x:()=>r});var n=s(6540);const i={},a=n.createContext(i);function o(e){const t=n.useContext(a);return n.useMemo(function(){return"function"==typeof e?e(t):{...t,...e}},[t,e])}function r(e){let t;return t=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),n.createElement(a.Provider,{value:t},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/9b4185c1.86e6c5f9.js b/assets/js/9b4185c1.3b8a6f84.js similarity index 99% rename from assets/js/9b4185c1.86e6c5f9.js rename to assets/js/9b4185c1.3b8a6f84.js index 93d8ebd1..9fc3111c 100644 --- a/assets/js/9b4185c1.86e6c5f9.js +++ b/assets/js/9b4185c1.3b8a6f84.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5330],{9356(e,s,n){n.r(s),n.d(s,{assets:()=>c,contentTitle:()=>d,default:()=>h,frontMatter:()=>o,metadata:()=>i,toc:()=>a});const i=JSON.parse('{"id":"privacy","title":"Privacy & Security","description":"Dashy was built with privacy in mind.","source":"@site/docs/privacy.md","sourceDirName":".","slug":"/privacy","permalink":"/docs/privacy","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/privacy.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Development Guides","permalink":"/docs/development-guides"},"next":{"title":"Credits","permalink":"/docs/credits"}}');var r=n(4848),t=n(8453);const o={},d="Privacy & Security",c={},a=[{value:"Contents",id:"contents",level:2},{value:"Browser Storage",id:"browser-storage",level:2},{value:"Cookies",id:"cookies",level:3},{value:"Session Storage",id:"session-storage",level:3},{value:"Local Storage",id:"local-storage",level:3},{value:"Deleting Stored Data",id:"deleting-stored-data",level:3},{value:"External Requests",id:"external-requests",level:2},{value:"Icons",id:"icons",level:3},{value:"Font Awesome",id:"font-awesome",level:4},{value:"Material Design Icons",id:"material-design-icons",level:4},{value:"Favicon Fetching",id:"favicon-fetching",level:4},{value:"Generative Icons",id:"generative-icons",level:4},{value:"Self-Hosted Icons",id:"self-hosted-icons",level:4},{value:"Other Icons",id:"other-icons",level:4},{value:"Web Assets",id:"web-assets",level:4},{value:"Themes",id:"themes",level:3},{value:"Status Checking",id:"status-checking",level:3},{value:"Update Checks",id:"update-checks",level:3},{value:"Cloud Backup",id:"cloud-backup",level:3},{value:"Web Search",id:"web-search",level:3},{value:"Initialization Page",id:"initialization-page",level:3},{value:"Anonymous Error Reporting",id:"anonymous-error-reporting",level:3},{value:"Widgets",id:"widgets",level:3},{value:"Dependencies",id:"dependencies",level:2},{value:"Securing your Environment",id:"securing-your-environment",level:2},{value:"Security Features",id:"security-features",level:2},{value:"Subresource Integrity",id:"subresource-integrity",level:3},{value:"SSL",id:"ssl",level:3},{value:"Authentication",id:"authentication",level:3},{value:"Configuration Lockdown",id:"configuration-lockdown",level:3},{value:"Disabling Features",id:"disabling-features",level:3},{value:"Threat Model",id:"threat-model",level:2},{value:"Intended Deployment",id:"intended-deployment",level:3},{value:"Trust Boundaries",id:"trust-boundaries",level:3},{value:"Assets",id:"assets",level:3},{value:"When Dashy is NOT the Right Choice",id:"when-dashy-is-not-the-right-choice",level:3},{value:"Known Limitations",id:"known-limitations",level:2},{value:"Update & Patch Policy",id:"update--patch-policy",level:2},{value:"Reporting a Security Issue",id:"reporting-a-security-issue",level:2}];function l(e){const s={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",kbd:"kbd",li:"li",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(s.header,{children:(0,r.jsx)(s.h1,{id:"privacy--security",children:"Privacy & Security"})}),"\n",(0,r.jsx)(s.p,{children:"Dashy was built with privacy in mind.\nSelf-hosting your own apps and services is a great way to protect yourself from the mass data collection employed by big tech companies, and Dashy was designed to make self-hosting easier."}),"\n",(0,r.jsx)(s.p,{children:"Dashy operates on the premise, that:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:"No external data requests should ever be made, unless explicitly enabled by the user"}),"\n",(0,r.jsx)(s.li,{children:"All code is 100% open source, clearly documented and intended to be easily auditable"}),"\n",(0,r.jsx)(s.li,{children:"Privacy-respecting by default. No premium features, analytics, tracking or ads"}),"\n"]}),"\n",(0,r.jsx)(s.p,{children:"This document outlines all network requests, data storage requirements, security configurations and data handling processes."}),"\n",(0,r.jsx)(s.admonition,{type:"tip",children:(0,r.jsxs)(s.p,{children:["Btw (shameless plug), if you care about your privacy, you might also like ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/awesome-privacy/",children:"awesome-privacy"}),"!",(0,r.jsx)(s.br,{})]})}),"\n",(0,r.jsx)(s.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["Privacy\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#browser-storage",children:"Browser Storage"})}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.a,{href:"#external-requests",children:"External Requests"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#icons",children:"Icons"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#themes",children:"Themes"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#status-checking",children:"Status Checking"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#update-checks",children:"Update Checks"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#cloud-backup",children:"Cloud Backup"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#web-search",children:"Web Search"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#initialization-page",children:"Initialization Page"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#anonymous-error-reporting",children:"Anonymous Error Reporting"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#widgets",children:"Widgets"})}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(s.li,{children:["Security\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#dependencies",children:"Dependencies"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#securing-your-environment",children:"Securing your Environment"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#security-features",children:"Security Features"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#threat-model",children:"Threat Model"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#known-limitations",children:"Known Limitations"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#update--patch-policy",children:"Update & Patch Policy"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#reporting-a-security-issue",children:"Reporting a Security Issue"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"browser-storage",children:"Browser Storage"}),"\n",(0,r.jsx)(s.p,{children:"In order for user preferences to be persisted some data is stored locally in your browsers storage.\nNo personal info is kept here, none of this data can be accessed by other domains, no data is ever sent to any server without your prior consent, and all data is removed when no longer needed unless you delete it sooner."}),"\n",(0,r.jsx)(s.p,{children:"The following section outlines all data that is stored in the browsers, as cookies, session storage or local storage."}),"\n",(0,r.jsx)(s.h3,{id:"cookies",children:"Cookies"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/Guides/Cookies",children:"Cookies"})," will expire after their pre-defined lifetime.\nDashy uses cookies for authentication, when enabled."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"AUTH_TOKEN"})," - A unique token, generated from a hash of users credentials, to verify they are authenticated. Only used when auth is enabled."]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"session-storage",children:"Session Storage"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage",children:"Session storage"})," is deleted when the current session ends (tab / window is closed).\nDashy uses session storage service worker status and error list."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"SW_STATUS"})," - The current status of any service workers"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"ERROR_LOG"})," - List of recent errors"]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"local-storage",children:"Local Storage"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage",children:"Local storage"})," is persisted between sessions, and only deleted when manually removed.\nDashy can use local storage to keep track of your preferences."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LANGUAGE"})," - The locale to show app text in"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"HIDE_INFO_NOTIFICATION"})," - Set to true once user dismissed welcome message, so that it's not shown again"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LAYOUT_ORIENTATION"})," - Preferred section layout, either auto, horizontal, vertical or masonry"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"COLLAPSE_STATE"})," - Remembers which sections are collapsed"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"ICON_SIZE"})," - Size of items, either small, medium or large"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"THEME"})," - Users applied theme"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"CUSTOM_COLORS"})," - Any color modifications made to a given theme"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"BACKUP_ID"})," - If a backup has been made, the ID is stored here"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"BACKUP_HASH"})," - A unique hash of the previous backups meta data"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"HIDE_SETTINGS"})," - Lets user hide or show the settings menu"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"USERNAME"})," - If user logged in, store username. Only used to show welcome message, not used for auth"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"CONF_SECTIONS"})," - Array of sections, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"PAGE_INFO"})," - Config page info, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"APP_CONFIG"})," - App config, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"MOST_USED"})," - If smart sort is used to order items by most used, store open count"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LAST_USED"})," - If smart sort is used to order items by last used, store timestamps"]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"deleting-stored-data",children:"Deleting Stored Data"}),"\n",(0,r.jsxs)(s.p,{children:["You can manually view and delete session storage, local storage and cookies at anytime. First ",(0,r.jsx)(s.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open"})," your browsers developer tools (usually ",(0,r.jsx)(s.kbd,{children:"F12"}),"), then under the Application tab select the storage category. Here you will see a list of stored data, and you can select any item and delete it."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"external-requests",children:"External Requests"}),"\n",(0,r.jsx)(s.p,{children:"By default, Dashy will not make any external requests, unless you configure it to. Some features (which are off by default) do require internet access, and this section outlines those features, the services used, and (where applicable) links to their privacy policies."}),"\n",(0,r.jsx)(s.h3,{id:"icons",children:"Icons"}),"\n",(0,r.jsx)(s.h4,{id:"font-awesome",children:"Font Awesome"}),"\n",(0,r.jsxs)(s.p,{children:["If either any of your sections, items or themes are using icons from font-awesome, then it will be automatically enabled. But you can also manually enable or disable it by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableFontAwesome"})," to ",(0,r.jsx)(s.code,{children:"true"})," / ",(0,r.jsx)(s.code,{children:"false"}),". Requests are made directly to Font-Awesome CDN, for more info, see the ",(0,r.jsx)(s.a,{href:"https://fontawesome.com/privacy",children:"Font Awesome Privacy Policy"}),"."]}),"\n",(0,r.jsx)(s.h4,{id:"material-design-icons",children:"Material Design Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If either any of your sections, items or themes are mdi icons, then it will be automatically enabled. But you can also manually enable or disable it by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableMaterialDesignIcons"})," to ",(0,r.jsx)(s.code,{children:"true"})," / ",(0,r.jsx)(s.code,{children:"false"}),". Requests are made directly to Material-Design-Icons CDN, for more info, see the ",(0,r.jsx)(s.a,{href:"https://materialdesignicons.com/",children:"Material Design Icons Website"}),"."]}),"\n",(0,r.jsx)(s.h4,{id:"favicon-fetching",children:"Favicon Fetching"}),"\n",(0,r.jsxs)(s.p,{children:["If an item's icon is set to ",(0,r.jsx)(s.code,{children:"favicon"}),", then it will be auto-fetched from the corresponding URL. Since not all websites have their icon located at ",(0,r.jsx)(s.code,{children:"/favicon.ico"}),", and if they do, it's often very low resolution (like ",(0,r.jsx)(s.code,{children:"16 x 16 px"}),"). Therefore, the default behavior is for Dashy to check if the URL is public, and if so will use an API to fetch the favicon. For self-hosted services, the favicon will be fetched from the default path, and no external requests will be made."]}),"\n",(0,r.jsxs)(s.p,{children:["The default favicon API is ",(0,r.jsx)(s.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"}),", but this can be changed by setting ",(0,r.jsx)(s.code,{children:"appConfig.faviconApi"})," to an alternate source (",(0,r.jsx)(s.code,{children:"iconhorse"}),", ",(0,r.jsx)(s.code,{children:"faviconkit"}),", ",(0,r.jsx)(s.code,{children:"besticon"}),", ",(0,r.jsx)(s.code,{children:"duckduckgo"}),", ",(0,r.jsx)(s.code,{children:"google"})," and ",(0,r.jsx)(s.code,{children:"allesedv"})," are supported). If you do not want to use any API, then you can set this property to ",(0,r.jsx)(s.code,{children:"local"}),", and the favicon will be fetched from the default path. For hosted services, this will still incur an external request."]}),"\n",(0,r.jsx)(s.h4,{id:"generative-icons",children:"Generative Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If an item has the icon set to ",(0,r.jsx)(s.code,{children:"generative"}),", then an external request it made to ",(0,r.jsx)(s.a,{href:"https://dicebear.com/",children:"Dice Bear"})," to fetch the uniquely generated icon. The URL of a given service is used as the key for generating the icon, but it is first hashed and encoded for basic privacy. For more info, please reference the ",(0,r.jsx)(s.a,{href:"https://avatars.dicebear.com/legal/privacy-policy",children:"Dicebear Privacy Policy"})]}),"\n",(0,r.jsxs)(s.p,{children:["As a fallback, if Dicebear fails, then ",(0,r.jsx)(s.a,{href:"https://evatar.io/",children:"Evatar"})," is used."]}),"\n",(0,r.jsx)(s.h4,{id:"self-hosted-icons",children:"Self-Hosted Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If an item's icon uses the ",(0,r.jsx)(s.code,{children:"sh-"})," prefix, icons are fetched from the ",(0,r.jsx)(s.a,{href:"https://selfh.st/icons/",children:"selfh.st icons"})," CDN at ",(0,r.jsx)(s.code,{children:"https://cdn.jsdelivr.net/gh/selfhst/icons"}),". This only applies when you explicitly use the ",(0,r.jsx)(s.code,{children:"sh-"})," prefix for an icon."]}),"\n",(0,r.jsx)(s.h4,{id:"other-icons",children:"Other Icons"}),"\n",(0,r.jsxs)(s.p,{children:["Section icons, item icons and app icons are able to accept a URL to a raw image, if the image is hosted online then an external request will be made. To avoid the need to make external requests for icon assets, you can either use a self-hosted CDN, or store your images within ",(0,r.jsx)(s.code,{children:"./public/item-icons"})," (which can be mounted as a volume if you're using Docker)."]}),"\n",(0,r.jsx)(s.h4,{id:"web-assets",children:"Web Assets"}),"\n",(0,r.jsx)(s.p,{children:"By default, all assets required by Dashy come bundled within the source, and so no external requests are made. If you add an additional font, which is imported from a CDN, then that will incur an external request. The same applies for other web assets, like external images, scripts or styles."}),"\n",(0,r.jsx)(s.h3,{id:"themes",children:"Themes"}),"\n",(0,r.jsx)(s.p,{children:"Certain themes may use external assets (such as fonts or images). These are only loaded if you actively select the theme. Currently, this applies to: Adventure, Vaporwave, Glass, Glass-2 and Night Sky themes, which load background images from external sources."}),"\n",(0,r.jsx)(s.h3,{id:"status-checking",children:"Status Checking"}),"\n",(0,r.jsx)(s.p,{children:"The status checking feature allows you to ping your apps/ services to check if they are currently operational."}),"\n",(0,r.jsx)(s.p,{children:"Dashy will ping your services directly, and does not rely on any third party. If you are checking the uptime status of a public/ hosted application, then please refer to that services privacy policy. For all self-hosted services, requests happen locally within your network, and are not external."}),"\n",(0,r.jsx)(s.h3,{id:"update-checks",children:"Update Checks"}),"\n",(0,r.jsxs)(s.p,{children:["When the application loads, it checks for updates. The results of which are displayed in the config menu of the UI. This was implemented because using a very outdated version of Dashy may have unfixed issues. Your version is fetched from the source (local request), but the latest version is fetched from GitHub, which is an external request. This can be disabled by setting ",(0,r.jsx)(s.code,{children:"appConfig.disableUpdateChecks: true"})]}),"\n",(0,r.jsx)(s.h3,{id:"cloud-backup",children:"Cloud Backup"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy has an optional End-to-End encrypted ",(0,r.jsx)(s.a,{href:"/docs/backup-restore",children:"cloud backup feature"}),". No data is ever transmitted unless you actively enable this feature through the UI."]}),"\n",(0,r.jsxs)(s.p,{children:["All data is encrypted before being sent to the backend. This is done in ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/CloudBackup.js",children:(0,r.jsx)(s.code,{children:"CloudBackup.js"})}),", using ",(0,r.jsx)(s.a,{href:"https://github.com/brix/crypto-js",children:"crypto.js"}),"'s AES method, using the users chosen password as the key. The data is then sent to a ",(0,r.jsx)(s.a,{href:"https://developers.cloudflare.com/workers/learning/how-workers-works",children:"Cloudflare worker"})," (a platform for running serverless functions), and stored in a ",(0,r.jsx)(s.a,{href:"https://developers.cloudflare.com/workers/learning/how-kv-works",children:"KV"})," data store."]}),"\n",(0,r.jsx)(s.p,{children:"Your selected password never leaves your device, and is hashed before being compared. It is only possible to restore a configuration if you have both the backup ID and decryption password. Because the data is encrypted on the client-side (before being sent to the cloud), it is not possible for a man-in-the-middle, government entity, website owner, or even Cloudflare to be able read any of your data."}),"\n",(0,r.jsx)(s.h3,{id:"web-search",children:"Web Search"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy has a primitive ",(0,r.jsx)(s.a,{href:"/docs/searching#web-search",children:"web search feature"}),". No external requests are made, instead you are redirected to your chosen search engine (defaults to DuckDuckGo), using your chosen opening method."]}),"\n",(0,r.jsxs)(s.p,{children:["This feature can be disabled under appConfig, with ",(0,r.jsx)(s.code,{children:"webSearch: { disableWebSearch: true }"})]}),"\n",(0,r.jsx)(s.h3,{id:"initialization-page",children:"Initialization Page"}),"\n",(0,r.jsxs)(s.p,{children:["When Dashy is first building (before the compiled app is ready), a static initialization page is displayed. This page loads a Google Font (",(0,r.jsx)(s.a,{href:"https://fonts.google.com/specimen/Fredoka+One",children:"Fredoka One"}),") from ",(0,r.jsx)(s.code,{children:"https://fonts.googleapis.com"}),". This only occurs during the build process, not during normal app usage. For more info, see ",(0,r.jsx)(s.a,{href:"https://policies.google.com/privacy",children:"Google's Privacy Policy"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"anonymous-error-reporting",children:"Anonymous Error Reporting"}),"\n",(0,r.jsxs)(s.p,{children:["Error reporting is disabled by default, and no data will ever be sent without your explicit consent. In fact, the error tracking code isn't even imported unless you have actively enabled it. ",(0,r.jsx)(s.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," is used for this, it's an open source error tracking and performance monitoring tool, used to identify any issues which occur in the production app (if you enable it)."]}),"\n",(0,r.jsxs)(s.p,{children:["The crash report includes the file or line of code that triggered the error, and a 2-layer deep stack trace. Reoccurring errors will also include the following user information: OS type (Mac, Windows, Linux, Android or iOS) and browser type (Firefox, Chrome, IE, Safari). Data scrubbing is enabled. IP address will not be stored. If any potentially identifiable data ever finds its way into a crash report, it will be automatically and permanently erased. All statistics collected are anonymized and stored securely, and are automatically deleted after 14 days. For more about privacy and security, see the ",(0,r.jsx)(s.a,{href:"https://sentry.io/security/",children:"Sentry Docs"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["Enabling anonymous error reporting helps me to discover bugs I was unaware of, and then fix them, in order to make Dashy more reliable long term. Error reporting is activated by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableErrorReporting: true"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["If you need to monitor bugs yourself, then you can ",(0,r.jsx)(s.a,{href:"https://develop.sentry.dev/self-hosted/",children:"self-host your own Sentry Server"}),", and use it by setting ",(0,r.jsx)(s.code,{children:"appConfig.sentryDsn"})," to your Sentry instances ",(0,r.jsx)(s.a,{href:"https://docs.sentry.io/product/sentry-basics/dsn-explainer/",children:"Data Source Name"}),", then just enable error reporting in Dashy."]}),"\n",(0,r.jsx)(s.h3,{id:"widgets",children:"Widgets"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports ",(0,r.jsx)(s.a,{href:"/docs/widgets",children:"Widgets"})," for displaying dynamic content. Below is a list of all widgets that make external data requests, along with the endpoint they call and a link to the Privacy Policy of that service."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Widget"}),(0,r.jsx)(s.th,{children:"Endpoint"}),(0,r.jsx)(s.th,{children:"Privacy Policy"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.a,{href:"/docs/widgets#weather",children:"Weather"})," / ",(0,r.jsx)(s.a,{href:"/docs/widgets#weather-forecast",children:"Weather Forecast"})]}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.openweathermap.org"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#rss-feed",children:"RSS Feed"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.rss2json.com/v1/api.json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://rss2json.com/privacy-policy",children:"Rss2Json Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#public-ip",children:"IP Address"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://free.freeipapi.com/api/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://freeipapi.com/",children:"FreeIPAPI"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://ipinfo.io/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipinfo.io/privacy-policy",children:"IPInfo Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.ipquery.io/"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipquery.io/",children:"IPQuery"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"http://ip-api.com/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ip-api.com/docs/legal",children:"IP-API Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.ipgeolocation.io/ipgeo"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipgeolocation.io/privacy.html",children:"IPGeoLocation Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#ip-blacklist",children:"IP Blacklist"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.blacklistchecker.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://blacklistchecker.com/privacy",children:"Blacklist Checker Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#domain-monitor",children:"Domain Monitor"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.whoapi.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://whoapi.com/privacy-policy/",children:"WhoAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.a,{href:"/docs/widgets#crypto-watch-list",children:"Crypto Watch List"})," / ",(0,r.jsx)(s.a,{href:"/docs/widgets#crypto-token-price-history",children:"Token Price History"})]}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.coingecko.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#wallet-balance",children:"Wallet Balance"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.blockcypher.com/"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.blockcypher.com/privacy.html",children:"BlockCypher Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#code-stats",children:"Code::Stats"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://codestats.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://codestats.net/tos#privacy",children:"Code::Stats Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#addyio",children:"addy.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://app.addy.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://addy.io/privacy/",children:"addy.io Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#vulnerability-feed",children:"Vulnerability Feed"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://services.nvd.nist.gov/rest/json/cves/2.0"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.nist.gov/privacy-policy",children:"NIST Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#exchange-rates",children:"Exchange Rate"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://v6.exchangerate-api.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.exchangerate-api.com/terms",children:"ExchangeRateAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#public-holidays",children:"Public Holidays"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://kayaposoft.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/jurajmajer/enrico",children:"jurajmajer/enrico"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#covid-19-status",children:"Covid-19 Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://disease.sh/v3/covid-19"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/disease-sh/api",children:"disease-sh/api"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#sports-scores",children:"Sports Scores"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://thesportsdb.com"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#news-headlines",children:"News Headlines"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.currentsapi.services"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://currentsapi.services/privacy",children:"CurrentsAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#mullvad-status",children:"Mullvad Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://am.i.mullvad.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://mullvad.net/en/help/privacy-policy/",children:"Mullvad Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#tfl-status",children:"TFL Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.tfl.gov.uk"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://tfl.gov.uk/corporate/privacy-and-cookies/",children:"TFL Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#stock-price-history",children:"Stock Price History"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.alphavantage.co"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.alphavantage.co/privacy/",children:"AlphaVantage Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#eth-gas-prices",children:"ETH Gas Prices"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://ethgas.watch"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/wslyvh/ethgaswatch",children:"wslyvh/ethgaswatch"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#joke",children:"Joke"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://v2.jokeapi.dev"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://sv443.net/privacypolicy/en",children:"SV443's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#chuck-norris-jokes",children:"Chuck Norris Jokes"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.chucknorris.io"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#xkcd-comic",children:"XKCD Comic"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://xkcd.vercel.app"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://xkcd.com",children:"XKCD"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#flight-data",children:"Flight Data"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://aerodatabox.p.rapidapi.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.aerodatabox.com/#h.p_cxtiyzwf_wqd",children:"AeroDataBox Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#astronomy-picture-of-the-day",children:"Astronomy Picture of the Day"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://apod.as93.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.nasa.gov/about/highlights/HP_Privacy.html",children:"NASA's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#github-trending",children:"GitHub Trending"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://trend.doforce.xyz"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#github-profile-stats",children:"GitHub Profile Stats"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://github-readme-stats.vercel.app"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#cron-monitoring-health-checks",children:"Cron Monitoring (Health Checks)"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://healthchecks.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://healthchecks.io/privacy/",children:"Health-Checks Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#hacker-news-trending",children:"Hacker News Trending"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://hacker-news.firebaseio.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.ycombinator.com/legal#privacy",children:"Y Combinator Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#minecraft-server-status",children:"Minecraft Server Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.mcsrvstat.us"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#mvg",children:"MVG"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.mvg.de/api/fib/v2/"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#rescue-time",children:"RescueTime"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.rescuetime.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.rescuetime.com/privacy",children:"RescueTime Privacy Policy"})})]})]})]}),"\n",(0,r.jsx)(s.p,{children:"Note: There are also many widgets that connect to self-hosted services (such as Pi-hole, AdGuard, Glances, Nextcloud, Proxmox, Uptime Kuma, etc.). These only make requests to your own configured server addresses and do not contact any third-party services."}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"dependencies",children:"Dependencies"}),"\n",(0,r.jsxs)(s.p,{children:["As with most web projects, Dashy relies on several ",(0,r.jsx)(s.a,{href:"/docs/credits#dependencies",children:"dependencies"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["Dependencies can introduce security vulnerabilities, but since all these packages are open source any issues are usually very quickly spotted. Dashy is using Snyk for dependency security monitoring, and you can see ",(0,r.jsx)(s.a,{href:"https://snyk.io/test/github/lissy93/dashy",children:"the latest report here"}),". If any issue is detected by Snyk, a note about it will appear at the top of the Readme, and will usually be fixed within 48 hours."]}),"\n",(0,r.jsxs)(s.p,{children:["Note that packages listed under ",(0,r.jsx)(s.code,{children:"devDependencies"})," section are only used for building the project, and are not included in the production environment."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"securing-your-environment",children:"Securing your Environment"}),"\n",(0,r.jsx)(s.p,{children:"There is very little complexity involved with Dashy, and therefore the attack surface is reasonably small, but it is still important to follow best practices for all your self-hosted apps:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Use SSL/HTTPS"})," for securing traffic in transit, see ",(0,r.jsx)(s.a,{href:"/docs/management#ssl-certificates",children:"Management Docs: SSL Certificates"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Configure authentication"})," to prevent unauthorized access, see ",(0,r.jsx)(s.a,{href:"/docs/authentication",children:"Authentication Docs"}),". For internet-facing instances, use ",(0,r.jsx)(s.a,{href:"/docs/authentication#keycloak",children:"Keycloak"}),", ",(0,r.jsx)(s.a,{href:"/docs/authentication#oidc",children:"OIDC"}),", or an ",(0,r.jsx)(s.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternative server-side method"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Place behind a reverse proxy"})," if exposing to the internet, see ",(0,r.jsx)(s.a,{href:"/docs/management#network-exposure",children:"Management Docs: Network Exposure"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Harden your containers"})," if running in Docker, see ",(0,r.jsx)(s.a,{href:"/docs/management#container-security",children:"Management Docs: Container Security"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Keep Dashy and your system up-to-date"})," to ensure known vulnerabilities are patched"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Configure firewall rules"})," to restrict access to only necessary ports and networks"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Use a VPN"})," for private access without exposing Dashy to the public internet"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsxs)(s.strong,{children:["Follow ",(0,r.jsx)(s.a,{href:"https://docs.docker.com/engine/security/",children:"Docker security best practices"})]})," including running as non-root, limiting capabilities, and using read-only volumes"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"security-features",children:"Security Features"}),"\n",(0,r.jsx)(s.h3,{id:"subresource-integrity",children:"Subresource Integrity"}),"\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/Security/Subresource_Integrity",children:"Subresource Integrity"})," or SRI is a security feature that enables browsers to verify that resources they fetch are delivered without unexpected manipulation. It works by allowing you to provide a cryptographic hash that a fetched resource must match. This prevents the app from loading any resources that have been manipulated, by verifying the files hashes. It safeguards against the risk of an attacker injecting arbitrary malicious content into any files served up via a CDN."]}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports SRI, and it is recommended to enable this if you are hosting your dashboard via a public CDN. To enable SRI, set the ",(0,r.jsx)(s.code,{children:"INTEGRITY"})," environmental variable to ",(0,r.jsx)(s.code,{children:"true"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"ssl",children:"SSL"}),"\n",(0,r.jsxs)(s.p,{children:["Native SSL support is enabled, for setup instructions, see the ",(0,r.jsx)(s.a,{href:"/docs/management#ssl-certificates",children:"Management Docs"})]}),"\n",(0,r.jsx)(s.h3,{id:"authentication",children:"Authentication"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports built-in auth, server-based SSO using Keycloak or any OIDC provider, and header-based authentication for reverse proxy setups. Full details of which, along with alternate authentication methods can be found in the ",(0,r.jsx)(s.a,{href:"/docs/authentication",children:"Authentication Docs"}),". If your dashboard is exposed to the internet and/ or contains any sensitive info it is strongly recommended to configure access control with Keycloak, OIDC, or another server-side method."]}),"\n",(0,r.jsx)(s.h3,{id:"configuration-lockdown",children:"Configuration Lockdown"}),"\n",(0,r.jsx)(s.p,{children:"Dashy provides several options to restrict what users can modify:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.preventWriteToDisk"})," - Prevents config changes from being saved to the server"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.preventLocalSave"})," - Prevents config changes from being saved to browser storage"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.disableConfiguration"})," - Hides the config UI from all users"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.disableConfigurationForNonAdmin"})," - Hides the config UI for non-admin users"]}),"\n"]}),"\n",(0,r.jsxs)(s.p,{children:["These can be combined with the ",(0,r.jsx)(s.code,{children:"admin"})," and ",(0,r.jsx)(s.code,{children:"normal"})," user roles to give fine-grained control. Admin users can save config changes, while normal users have read-only access. For more details, see the ",(0,r.jsx)(s.a,{href:"/docs/authentication#permissions",children:"Authentication Docs: Permissions"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"disabling-features",children:"Disabling Features"}),"\n",(0,r.jsx)(s.p,{children:"You may wish to disable features that you don't want to use, if they involve storing data in the browser or making network requests."}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["To disable smart-sort (uses local storage), set ",(0,r.jsx)(s.code,{children:"appConfig.disableSmartSort: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To disable update checks (makes external request to GH), set ",(0,r.jsx)(s.code,{children:"appConfig.disableUpdateChecks: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To disable web search (redirect to external / internal content), set ",(0,r.jsx)(s.code,{children:"appConfig.webSearch.disableWebSearch: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep status checks disabled (external/ internal requests), set ",(0,r.jsx)(s.code,{children:"appConfig.statusCheck: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep font-awesome icons disabled (external requests), set ",(0,r.jsx)(s.code,{children:"appConfig.enableFontAwesome: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep error reporting disabled (external requests and data collection), set ",(0,r.jsx)(s.code,{children:"appConfig.enableErrorReporting: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep the service worker disabled (stores cache of app in browser data), set ",(0,r.jsx)(s.code,{children:"appConfig.enableServiceWorker: false"})]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"threat-model",children:"Threat Model"}),"\n",(0,r.jsx)(s.p,{children:"Dashy is a statically-hosted dashboard application, designed to be self-hosted on a private network. This threat model outlines the intended deployment context, trust boundaries, known risks and accepted trade-offs, to help users assess whether Dashy is appropriate for their environment."}),"\n",(0,r.jsx)(s.h3,{id:"intended-deployment",children:"Intended Deployment"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy is designed to run on a ",(0,r.jsx)(s.strong,{children:"private local network"})," (e.g. a home lab), accessed by a ",(0,r.jsx)(s.strong,{children:"small number of trusted users"}),". It is a convenience tool for organizing links to self-hosted services - it is not designed to protect sensitive resources or act as an access control layer."]}),"\n",(0,r.jsxs)(s.p,{children:["If exposed to the internet, Dashy ",(0,r.jsx)(s.strong,{children:"must"})," be placed behind a reverse proxy with server-side authentication (e.g. Authelia, Authentik, Cloudflare Access). The built-in client-side auth is a convenience feature for private networks, not a security boundary."]}),"\n",(0,r.jsx)(s.h3,{id:"trust-boundaries",children:"Trust Boundaries"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Boundary"}),(0,r.jsx)(s.th,{children:"Trusted Side"}),(0,r.jsx)(s.th,{children:"Untrusted Side"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Local network"}),(0,r.jsx)(s.td,{children:"LAN users, self-hosted services"}),(0,r.jsx)(s.td,{children:"The public internet"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:["Config file (",(0,r.jsx)(s.code,{children:"conf.yml"}),")"]}),(0,r.jsx)(s.td,{children:"Server admin who writes the config"}),(0,r.jsx)(s.td,{children:"End users who view the dashboard"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Browser storage"}),(0,r.jsx)(s.td,{children:"The current browser session"}),(0,r.jsx)(s.td,{children:"Other domains, other users of the same device"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"CORS proxy / status checks"}),(0,r.jsx)(s.td,{children:"Configured target URLs (set by admin)"}),(0,r.jsx)(s.td,{children:"Arbitrary URLs (if auth is not enabled)"})]})]})]}),"\n",(0,r.jsx)(s.h3,{id:"assets",children:"Assets"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Asset"}),(0,r.jsx)(s.th,{children:"Description"}),(0,r.jsx)(s.th,{children:"Storage"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Dashboard configuration"}),(0,r.jsx)(s.td,{children:"Service URLs, section layout, app settings"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server, optionally cached in browser localStorage"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"User credentials"}),(0,r.jsx)(s.td,{children:"SHA-256 password hashes, Keycloak/OIDC client IDs"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"API keys"}),(0,r.jsx)(s.td,{children:"Keys for widget services (weather, stocks, etc.)"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server or environment variables"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Auth tokens"}),(0,r.jsx)(s.td,{children:"Session token derived from credentials"}),(0,r.jsxs)(s.td,{children:["Browser cookie (",(0,r.jsx)(s.code,{children:"dashyAuthToken"}),")"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"User preferences"}),(0,r.jsx)(s.td,{children:"Theme, layout, language, collapsed sections"}),(0,r.jsx)(s.td,{children:"Browser localStorage"})]})]})]}),"\n",(0,r.jsx)(s.h3,{id:"when-dashy-is-not-the-right-choice",children:"When Dashy is NOT the Right Choice"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["You need a ",(0,r.jsx)(s.strong,{children:"multi-tenant"})," dashboard with per-user audit trails"]}),"\n",(0,r.jsxs)(s.li,{children:["You are deploying on the ",(0,r.jsx)(s.strong,{children:"public internet without a reverse proxy"})]}),"\n",(0,r.jsxs)(s.li,{children:["Your dashboard contains ",(0,r.jsx)(s.strong,{children:"secrets or credentials"})," that must be protected from all users who can reach the server"]}),"\n",(0,r.jsxs)(s.li,{children:["You require ",(0,r.jsx)(s.strong,{children:"FIPS-compliant"})," or ",(0,r.jsx)(s.strong,{children:"SOC 2"})," certified software"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"known-limitations",children:"Known Limitations"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Report"}),(0,r.jsx)(s.th,{children:"Response"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Client-side auth can be bypassed via browser dev tools"'}),(0,r.jsxs)(s.td,{children:["Correct (only if neither ",(0,r.jsx)(s.code,{children:"ENABLE_HTTP_AUTH"})," is set, nor any other auth mode). Client-side auth is a convenience for private networks, not a security boundary. Use server-side auth for untrusted environments."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"CORS proxy can make requests to internal services"'}),(0,r.jsx)(s.td,{children:"Correct. The CORS proxy is designed to reach services on your network. It is protected by auth middleware when auth is enabled. Do not expose Dashy without auth on an untrusted network."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Status checks can be used for SSRF"'}),(0,r.jsxs)(s.td,{children:["Status check target URLs are set by the server admin in ",(0,r.jsx)(s.code,{children:"conf.yml"}),", not by end users. If auth is enabled, the endpoint is protected."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Password hashes are stored in plaintext in conf.yml"'}),(0,r.jsx)(s.td,{children:"They are SHA-256 hashes, not plaintext passwords. The config file should be readable only by the server admin, and protected by HTTP auth when served."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"localStorage/cookies are not encrypted"'}),(0,r.jsx)(s.td,{children:"Browser storage is scoped to the origin and inaccessible to other domains. On a shared device, use your browser's profile isolation."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No CSRF protection"'}),(0,r.jsx)(s.td,{children:"Dashy's state-changing operations (config save) are protected by auth middleware. CSRF is a low risk on a private network dashboard."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Docker container runs as root"'}),(0,r.jsxs)(s.td,{children:["The container is sandboxed. For hardened deployments, override with ",(0,r.jsx)(s.code,{children:"--user"})," flag or configure in your Docker Compose file as described in the ",(0,r.jsx)(s.a,{href:"/docs/management#container-security",children:"container security docs"}),"."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Auth cookie is not HttpOnly/Secure"'}),(0,r.jsxs)(s.td,{children:["The token is needed by client-side JavaScript for auth state. On a private network over plain HTTP, the ",(0,r.jsx)(s.code,{children:"Secure"})," flag would break auth. Use HTTPS + a reverse proxy to add these flags if needed."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Iframe/embed widget can load arbitrary URLs"'}),(0,r.jsxs)(s.td,{children:["The widget config is written by the server admin, not end users. If you don't trust your config authors, disable the config editor with ",(0,r.jsx)(s.code,{children:"disableConfiguration"}),"."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"RSS widget renders HTML content"'}),(0,r.jsx)(s.td,{children:"RSS content is sanitized with DOMPurify before rendering. Script tags, event handlers and other dangerous elements are stripped."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No Content-Security-Policy headers"'}),(0,r.jsx)(s.td,{children:"CSP should be configured at the reverse proxy layer, since the correct policy depends on which widgets and icon CDNs you use. Dashy can't set a universal CSP that works for all configurations."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Config backups are not encrypted at rest"'}),(0,r.jsxs)(s.td,{children:["Backups are stored server-side alongside the original config. If an attacker has filesystem access, they already have ",(0,r.jsx)(s.code,{children:"conf.yml"}),". Encryption at rest is the responsibility of the host OS/volume."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No rate limiting on endpoints"'}),(0,r.jsx)(s.td,{children:"Rate limiting should be applied at the reverse proxy layer, where it can be tuned per-deployment. Dashy is not designed to be directly exposed to untrusted traffic."})]})]})]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"update--patch-policy",children:"Update & Patch Policy"}),"\n",(0,r.jsxs)(s.p,{children:["We follow Semantic Versioning for all releases. Security fixes are shipped as patch releases as quickly as possible and are published via immutable Git tags and Docker image tags. Users are encouraged to pin to a specific version in production and monitor releases on GitHub for security updates. The ",(0,r.jsx)(s.code,{children:":latest"})," Docker tag is provided for convenience but should not be relied on in production environments."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"reporting-a-security-issue",children:"Reporting a Security Issue"}),"\n",(0,r.jsxs)(s.p,{children:["Please see our ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/?tab=security-ov-file",children:"Security.md"})," doc for how to report issues.\nWe have an actively monitored security mailbox supporting PGP, as well as a GitHub Advisories vulnerability reporting program."]})]})}function h(e={}){const{wrapper:s}={...(0,t.R)(),...e.components};return s?(0,r.jsx)(s,{...e,children:(0,r.jsx)(l,{...e})}):l(e)}},8453(e,s,n){n.d(s,{R:()=>o,x:()=>d});var i=n(6540);const r={},t=i.createContext(r);function o(e){const s=i.useContext(t);return i.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function d(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:o(e.components),i.createElement(t.Provider,{value:s},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5330],{9356(e,s,n){n.r(s),n.d(s,{assets:()=>c,contentTitle:()=>d,default:()=>h,frontMatter:()=>o,metadata:()=>i,toc:()=>a});const i=JSON.parse('{"id":"privacy","title":"Privacy & Security","description":"Dashy was built with privacy in mind.","source":"@site/docs/privacy.md","sourceDirName":".","slug":"/privacy","permalink":"/docs/privacy","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/privacy.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Development Guides","permalink":"/docs/development-guides"},"next":{"title":"Credits","permalink":"/docs/credits"}}');var r=n(4848),t=n(8453);const o={},d="Privacy & Security",c={},a=[{value:"Contents",id:"contents",level:2},{value:"Browser Storage",id:"browser-storage",level:2},{value:"Cookies",id:"cookies",level:3},{value:"Session Storage",id:"session-storage",level:3},{value:"Local Storage",id:"local-storage",level:3},{value:"Deleting Stored Data",id:"deleting-stored-data",level:3},{value:"External Requests",id:"external-requests",level:2},{value:"Icons",id:"icons",level:3},{value:"Font Awesome",id:"font-awesome",level:4},{value:"Material Design Icons",id:"material-design-icons",level:4},{value:"Favicon Fetching",id:"favicon-fetching",level:4},{value:"Generative Icons",id:"generative-icons",level:4},{value:"Self-Hosted Icons",id:"self-hosted-icons",level:4},{value:"Other Icons",id:"other-icons",level:4},{value:"Web Assets",id:"web-assets",level:4},{value:"Themes",id:"themes",level:3},{value:"Status Checking",id:"status-checking",level:3},{value:"Update Checks",id:"update-checks",level:3},{value:"Cloud Backup",id:"cloud-backup",level:3},{value:"Web Search",id:"web-search",level:3},{value:"Initialization Page",id:"initialization-page",level:3},{value:"Anonymous Error Reporting",id:"anonymous-error-reporting",level:3},{value:"Widgets",id:"widgets",level:3},{value:"Dependencies",id:"dependencies",level:2},{value:"Securing your Environment",id:"securing-your-environment",level:2},{value:"Security Features",id:"security-features",level:2},{value:"Subresource Integrity",id:"subresource-integrity",level:3},{value:"SSL",id:"ssl",level:3},{value:"Authentication",id:"authentication",level:3},{value:"Configuration Lockdown",id:"configuration-lockdown",level:3},{value:"Disabling Features",id:"disabling-features",level:3},{value:"Threat Model",id:"threat-model",level:2},{value:"Intended Deployment",id:"intended-deployment",level:3},{value:"Trust Boundaries",id:"trust-boundaries",level:3},{value:"Assets",id:"assets",level:3},{value:"When Dashy is NOT the Right Choice",id:"when-dashy-is-not-the-right-choice",level:3},{value:"Known Limitations",id:"known-limitations",level:2},{value:"Update & Patch Policy",id:"update--patch-policy",level:2},{value:"Reporting a Security Issue",id:"reporting-a-security-issue",level:2}];function l(e){const s={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",kbd:"kbd",li:"li",p:"p",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(s.header,{children:(0,r.jsx)(s.h1,{id:"privacy--security",children:"Privacy & Security"})}),"\n",(0,r.jsx)(s.p,{children:"Dashy was built with privacy in mind.\nSelf-hosting your own apps and services is a great way to protect yourself from the mass data collection employed by big tech companies, and Dashy was designed to make self-hosting easier."}),"\n",(0,r.jsx)(s.p,{children:"Dashy operates on the premise, that:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:"No external data requests should ever be made, unless explicitly enabled by the user"}),"\n",(0,r.jsx)(s.li,{children:"All code is 100% open source, clearly documented and intended to be easily auditable"}),"\n",(0,r.jsx)(s.li,{children:"Privacy-respecting by default. No premium features, analytics, tracking or ads"}),"\n"]}),"\n",(0,r.jsx)(s.p,{children:"This document outlines all network requests, data storage requirements, security configurations and data handling processes."}),"\n",(0,r.jsx)(s.admonition,{type:"tip",children:(0,r.jsxs)(s.p,{children:["Btw (shameless plug), if you care about your privacy, you might also like ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/awesome-privacy/",children:"awesome-privacy"}),"!",(0,r.jsx)(s.br,{})]})}),"\n",(0,r.jsx)(s.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["Privacy\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#browser-storage",children:"Browser Storage"})}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.a,{href:"#external-requests",children:"External Requests"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#icons",children:"Icons"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#themes",children:"Themes"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#status-checking",children:"Status Checking"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#update-checks",children:"Update Checks"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#cloud-backup",children:"Cloud Backup"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#web-search",children:"Web Search"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#initialization-page",children:"Initialization Page"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#anonymous-error-reporting",children:"Anonymous Error Reporting"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#widgets",children:"Widgets"})}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(s.li,{children:["Security\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#dependencies",children:"Dependencies"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#securing-your-environment",children:"Securing your Environment"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#security-features",children:"Security Features"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#threat-model",children:"Threat Model"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#known-limitations",children:"Known Limitations"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#update--patch-policy",children:"Update & Patch Policy"})}),"\n",(0,r.jsx)(s.li,{children:(0,r.jsx)(s.a,{href:"#reporting-a-security-issue",children:"Reporting a Security Issue"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"browser-storage",children:"Browser Storage"}),"\n",(0,r.jsx)(s.p,{children:"In order for user preferences to be persisted some data is stored locally in your browsers storage.\nNo personal info is kept here, none of this data can be accessed by other domains, no data is ever sent to any server without your prior consent, and all data is removed when no longer needed unless you delete it sooner."}),"\n",(0,r.jsx)(s.p,{children:"The following section outlines all data that is stored in the browsers, as cookies, session storage or local storage."}),"\n",(0,r.jsx)(s.h3,{id:"cookies",children:"Cookies"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/Guides/Cookies",children:"Cookies"})," will expire after their pre-defined lifetime.\nDashy uses cookies for authentication, when enabled."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"AUTH_TOKEN"})," - A unique token, generated from a hash of users credentials, to verify they are authenticated. Only used when auth is enabled."]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"session-storage",children:"Session Storage"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/API/Window/sessionStorage",children:"Session storage"})," is deleted when the current session ends (tab / window is closed).\nDashy uses session storage service worker status and error list."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"SW_STATUS"})," - The current status of any service workers"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"ERROR_LOG"})," - List of recent errors"]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"local-storage",children:"Local Storage"}),"\n",(0,r.jsxs)(s.blockquote,{children:["\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage",children:"Local storage"})," is persisted between sessions, and only deleted when manually removed.\nDashy can use local storage to keep track of your preferences."]}),"\n"]}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LANGUAGE"})," - The locale to show app text in"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"HIDE_INFO_NOTIFICATION"})," - Set to true once user dismissed welcome message, so that it's not shown again"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LAYOUT_ORIENTATION"})," - Preferred section layout, either auto, horizontal, vertical or masonry"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"COLLAPSE_STATE"})," - Remembers which sections are collapsed"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"ICON_SIZE"})," - Size of items, either small, medium or large"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"THEME"})," - Users applied theme"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"CUSTOM_COLORS"})," - Any color modifications made to a given theme"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"BACKUP_ID"})," - If a backup has been made, the ID is stored here"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"BACKUP_HASH"})," - A unique hash of the previous backups meta data"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"HIDE_SETTINGS"})," - Lets user hide or show the settings menu"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"USERNAME"})," - If user logged in, store username. Only used to show welcome message, not used for auth"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"CONF_SECTIONS"})," - Array of sections, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"PAGE_INFO"})," - Config page info, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"APP_CONFIG"})," - App config, only used when user applies changes locally"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"MOST_USED"})," - If smart sort is used to order items by most used, store open count"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"LAST_USED"})," - If smart sort is used to order items by last used, store timestamps"]}),"\n"]}),"\n",(0,r.jsx)(s.h3,{id:"deleting-stored-data",children:"Deleting Stored Data"}),"\n",(0,r.jsxs)(s.p,{children:["You can manually view and delete session storage, local storage and cookies at anytime. First ",(0,r.jsx)(s.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open"})," your browsers developer tools (usually ",(0,r.jsx)(s.kbd,{children:"F12"}),"), then under the Application tab select the storage category. Here you will see a list of stored data, and you can select any item and delete it."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"external-requests",children:"External Requests"}),"\n",(0,r.jsx)(s.p,{children:"By default, Dashy will not make any external requests, unless you configure it to. Some features (which are off by default) do require internet access, and this section outlines those features, the services used, and (where applicable) links to their privacy policies."}),"\n",(0,r.jsx)(s.h3,{id:"icons",children:"Icons"}),"\n",(0,r.jsx)(s.h4,{id:"font-awesome",children:"Font Awesome"}),"\n",(0,r.jsxs)(s.p,{children:["If either any of your sections, items or themes are using icons from font-awesome, then it will be automatically enabled. But you can also manually enable or disable it by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableFontAwesome"})," to ",(0,r.jsx)(s.code,{children:"true"})," / ",(0,r.jsx)(s.code,{children:"false"}),". Requests are made directly to Font-Awesome CDN, for more info, see the ",(0,r.jsx)(s.a,{href:"https://fontawesome.com/privacy",children:"Font Awesome Privacy Policy"}),"."]}),"\n",(0,r.jsx)(s.h4,{id:"material-design-icons",children:"Material Design Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If either any of your sections, items or themes are mdi icons, then it will be automatically enabled. But you can also manually enable or disable it by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableMaterialDesignIcons"})," to ",(0,r.jsx)(s.code,{children:"true"})," / ",(0,r.jsx)(s.code,{children:"false"}),". Requests are made directly to Material-Design-Icons CDN, for more info, see the ",(0,r.jsx)(s.a,{href:"https://materialdesignicons.com/",children:"Material Design Icons Website"}),"."]}),"\n",(0,r.jsx)(s.h4,{id:"favicon-fetching",children:"Favicon Fetching"}),"\n",(0,r.jsxs)(s.p,{children:["If an item's icon is set to ",(0,r.jsx)(s.code,{children:"favicon"}),", then it will be auto-fetched from the corresponding URL. Since not all websites have their icon located at ",(0,r.jsx)(s.code,{children:"/favicon.ico"}),", and if they do, it's often very low resolution (like ",(0,r.jsx)(s.code,{children:"16 x 16 px"}),"). Therefore, the default behavior is for Dashy to check if the URL is public, and if so will use an API to fetch the favicon. For self-hosted services, the favicon will be fetched from the default path, and no external requests will be made."]}),"\n",(0,r.jsxs)(s.p,{children:["The default favicon API is ",(0,r.jsx)(s.a,{href:"https://favicon.allesedv.com/",children:"allesedv.com"}),", but this can be changed by setting ",(0,r.jsx)(s.code,{children:"appConfig.faviconApi"})," to an alternate source (",(0,r.jsx)(s.code,{children:"iconhorse"}),", ",(0,r.jsx)(s.code,{children:"faviconkit"}),", ",(0,r.jsx)(s.code,{children:"besticon"}),", ",(0,r.jsx)(s.code,{children:"duckduckgo"}),", ",(0,r.jsx)(s.code,{children:"google"})," and ",(0,r.jsx)(s.code,{children:"allesedv"})," are supported). If you do not want to use any API, then you can set this property to ",(0,r.jsx)(s.code,{children:"local"}),", and the favicon will be fetched from the default path. For hosted services, this will still incur an external request."]}),"\n",(0,r.jsx)(s.h4,{id:"generative-icons",children:"Generative Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If an item has the icon set to ",(0,r.jsx)(s.code,{children:"generative"}),", then an external request it made to ",(0,r.jsx)(s.a,{href:"https://dicebear.com/",children:"Dice Bear"})," to fetch the uniquely generated icon. The URL of a given service is used as the key for generating the icon, but it is first hashed and encoded for basic privacy. For more info, please reference the ",(0,r.jsx)(s.a,{href:"https://avatars.dicebear.com/legal/privacy-policy",children:"Dicebear Privacy Policy"})]}),"\n",(0,r.jsxs)(s.p,{children:["As a fallback, if Dicebear fails, then ",(0,r.jsx)(s.a,{href:"https://evatar.io/",children:"Evatar"})," is used."]}),"\n",(0,r.jsx)(s.h4,{id:"self-hosted-icons",children:"Self-Hosted Icons"}),"\n",(0,r.jsxs)(s.p,{children:["If an item's icon uses the ",(0,r.jsx)(s.code,{children:"sh-"})," prefix, icons are fetched from the ",(0,r.jsx)(s.a,{href:"https://selfh.st/icons/",children:"selfh.st icons"})," CDN at ",(0,r.jsx)(s.code,{children:"https://cdn.jsdelivr.net/gh/selfhst/icons"}),". This only applies when you explicitly use the ",(0,r.jsx)(s.code,{children:"sh-"})," prefix for an icon."]}),"\n",(0,r.jsx)(s.h4,{id:"other-icons",children:"Other Icons"}),"\n",(0,r.jsxs)(s.p,{children:["Section icons, item icons and app icons are able to accept a URL to a raw image, if the image is hosted online then an external request will be made. To avoid the need to make external requests for icon assets, you can either use a self-hosted CDN, or store your images within ",(0,r.jsx)(s.code,{children:"./public/item-icons"})," (which can be mounted as a volume if you're using Docker)."]}),"\n",(0,r.jsx)(s.h4,{id:"web-assets",children:"Web Assets"}),"\n",(0,r.jsx)(s.p,{children:"By default, all assets required by Dashy come bundled within the source, and so no external requests are made. If you add an additional font, which is imported from a CDN, then that will incur an external request. The same applies for other web assets, like external images, scripts or styles."}),"\n",(0,r.jsx)(s.h3,{id:"themes",children:"Themes"}),"\n",(0,r.jsx)(s.p,{children:"Certain themes may use external assets (such as fonts or images). These are only loaded if you actively select the theme. Currently, this applies to: Adventure, Vaporwave, Glass, Glass-2 and Night Sky themes, which load background images from external sources."}),"\n",(0,r.jsx)(s.h3,{id:"status-checking",children:"Status Checking"}),"\n",(0,r.jsx)(s.p,{children:"The status checking feature allows you to ping your apps/ services to check if they are currently operational."}),"\n",(0,r.jsx)(s.p,{children:"Dashy will ping your services directly, and does not rely on any third party. If you are checking the uptime status of a public/ hosted application, then please refer to that services privacy policy. For all self-hosted services, requests happen locally within your network, and are not external."}),"\n",(0,r.jsx)(s.h3,{id:"update-checks",children:"Update Checks"}),"\n",(0,r.jsxs)(s.p,{children:["When the application loads, it checks for updates. The results of which are displayed in the config menu of the UI. This was implemented because using a very outdated version of Dashy may have unfixed issues. Your version is fetched from the source (local request), but the latest version is fetched from GitHub, which is an external request. This can be disabled by setting ",(0,r.jsx)(s.code,{children:"appConfig.disableUpdateChecks: true"})]}),"\n",(0,r.jsx)(s.h3,{id:"cloud-backup",children:"Cloud Backup"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy has an optional End-to-End encrypted ",(0,r.jsx)(s.a,{href:"/docs/backup-restore",children:"cloud backup feature"}),". No data is ever transmitted unless you actively enable this feature through the UI."]}),"\n",(0,r.jsxs)(s.p,{children:["All data is encrypted before being sent to the backend. This is done in ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/CloudBackup.js",children:(0,r.jsx)(s.code,{children:"CloudBackup.js"})}),", using ",(0,r.jsx)(s.a,{href:"https://github.com/brix/crypto-js",children:"crypto.js"}),"'s AES method, using the users chosen password as the key. The data is then sent to a ",(0,r.jsx)(s.a,{href:"https://developers.cloudflare.com/workers/learning/how-workers-works",children:"Cloudflare worker"})," (a platform for running serverless functions), and stored in a ",(0,r.jsx)(s.a,{href:"https://developers.cloudflare.com/workers/learning/how-kv-works",children:"KV"})," data store."]}),"\n",(0,r.jsx)(s.p,{children:"Your selected password never leaves your device, and is hashed before being compared. It is only possible to restore a configuration if you have both the backup ID and decryption password. Because the data is encrypted on the client-side (before being sent to the cloud), it is not possible for a man-in-the-middle, government entity, website owner, or even Cloudflare to be able read any of your data."}),"\n",(0,r.jsx)(s.h3,{id:"web-search",children:"Web Search"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy has a primitive ",(0,r.jsx)(s.a,{href:"/docs/searching#web-search",children:"web search feature"}),". No external requests are made, instead you are redirected to your chosen search engine (defaults to DuckDuckGo), using your chosen opening method."]}),"\n",(0,r.jsxs)(s.p,{children:["This feature can be disabled under appConfig, with ",(0,r.jsx)(s.code,{children:"webSearch: { disableWebSearch: true }"})]}),"\n",(0,r.jsx)(s.h3,{id:"initialization-page",children:"Initialization Page"}),"\n",(0,r.jsxs)(s.p,{children:["When Dashy is first building (before the compiled app is ready), a static initialization page is displayed. This page loads a Google Font (",(0,r.jsx)(s.a,{href:"https://fonts.google.com/specimen/Fredoka+One",children:"Fredoka One"}),") from ",(0,r.jsx)(s.code,{children:"https://fonts.googleapis.com"}),". This only occurs during the build process, not during normal app usage. For more info, see ",(0,r.jsx)(s.a,{href:"https://policies.google.com/privacy",children:"Google's Privacy Policy"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"anonymous-error-reporting",children:"Anonymous Error Reporting"}),"\n",(0,r.jsxs)(s.p,{children:["Error reporting is disabled by default, and no data will ever be sent without your explicit consent. In fact, the error tracking code isn't even imported unless you have actively enabled it. ",(0,r.jsx)(s.a,{href:"https://github.com/getsentry/sentry",children:"Sentry"})," is used for this, it's an open source error tracking and performance monitoring tool, used to identify any issues which occur in the production app (if you enable it)."]}),"\n",(0,r.jsxs)(s.p,{children:["The crash report includes the file or line of code that triggered the error, and a 2-layer deep stack trace. Reoccurring errors will also include the following user information: OS type (Mac, Windows, Linux, Android or iOS) and browser type (Firefox, Chrome, IE, Safari). Data scrubbing is enabled. IP address will not be stored. If any potentially identifiable data ever finds its way into a crash report, it will be automatically and permanently erased. All statistics collected are anonymized and stored securely, and are automatically deleted after 14 days. For more about privacy and security, see the ",(0,r.jsx)(s.a,{href:"https://sentry.io/security/",children:"Sentry Docs"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["Enabling anonymous error reporting helps me to discover bugs I was unaware of, and then fix them, in order to make Dashy more reliable long term. Error reporting is activated by setting ",(0,r.jsx)(s.code,{children:"appConfig.enableErrorReporting: true"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["If you need to monitor bugs yourself, then you can ",(0,r.jsx)(s.a,{href:"https://develop.sentry.dev/self-hosted/",children:"self-host your own Sentry Server"}),", and use it by setting ",(0,r.jsx)(s.code,{children:"appConfig.sentryDsn"})," to your Sentry instances ",(0,r.jsx)(s.a,{href:"https://docs.sentry.io/product/sentry-basics/dsn-explainer/",children:"Data Source Name"}),", then just enable error reporting in Dashy."]}),"\n",(0,r.jsx)(s.h3,{id:"widgets",children:"Widgets"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports ",(0,r.jsx)(s.a,{href:"/docs/widgets",children:"Widgets"})," for displaying dynamic content. Below is a list of all widgets that make external data requests, along with the endpoint they call and a link to the Privacy Policy of that service."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Widget"}),(0,r.jsx)(s.th,{children:"Endpoint"}),(0,r.jsx)(s.th,{children:"Privacy Policy"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.a,{href:"/docs/widgets#weather",children:"Weather"})," / ",(0,r.jsx)(s.a,{href:"/docs/widgets#weather-forecast",children:"Weather Forecast"})]}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.openweathermap.org"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#rss-feed",children:"RSS Feed"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.rss2json.com/v1/api.json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://rss2json.com/privacy-policy",children:"Rss2Json Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#public-ip",children:"IP Address"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://free.freeipapi.com/api/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://freeipapi.com/",children:"FreeIPAPI"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://ipinfo.io/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipinfo.io/privacy-policy",children:"IPInfo Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.ipquery.io/"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipquery.io/",children:"IPQuery"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"http://ip-api.com/json"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ip-api.com/docs/legal",children:"IP-API Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.ipgeolocation.io/ipgeo"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://ipgeolocation.io/privacy.html",children:"IPGeoLocation Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#ip-blacklist",children:"IP Blacklist"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.blacklistchecker.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://blacklistchecker.com/privacy",children:"Blacklist Checker Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#domain-monitor",children:"Domain Monitor"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.whoapi.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://whoapi.com/privacy-policy/",children:"WhoAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.a,{href:"/docs/widgets#crypto-watch-list",children:"Crypto Watch List"})," / ",(0,r.jsx)(s.a,{href:"/docs/widgets#crypto-token-price-history",children:"Token Price History"})]}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.coingecko.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#wallet-balance",children:"Wallet Balance"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.blockcypher.com/"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.blockcypher.com/privacy.html",children:"BlockCypher Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#code-stats",children:"Code::Stats"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://codestats.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://codestats.net/tos#privacy",children:"Code::Stats Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#addyio",children:"addy.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://app.addy.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://addy.io/privacy/",children:"addy.io Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#vulnerability-feed",children:"Vulnerability Feed"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://services.nvd.nist.gov/rest/json/cves/2.0"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.nist.gov/privacy-policy",children:"NIST Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#exchange-rates",children:"Exchange Rate"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://v6.exchangerate-api.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.exchangerate-api.com/terms",children:"ExchangeRateAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#public-holidays",children:"Public Holidays"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://kayaposoft.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/jurajmajer/enrico",children:"jurajmajer/enrico"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#covid-19-status",children:"Covid-19 Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://disease.sh/v3/covid-19"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/disease-sh/api",children:"disease-sh/api"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#sports-scores",children:"Sports Scores"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://thesportsdb.com"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#news-headlines",children:"News Headlines"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.currentsapi.services"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://currentsapi.services/privacy",children:"CurrentsAPI Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#mullvad-status",children:"Mullvad Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://am.i.mullvad.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://mullvad.net/en/help/privacy-policy/",children:"Mullvad Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#tfl-status",children:"TFL Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.tfl.gov.uk"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://tfl.gov.uk/corporate/privacy-and-cookies/",children:"TFL Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#stock-price-history",children:"Stock Price History"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.alphavantage.co"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.alphavantage.co/privacy/",children:"AlphaVantage Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#eth-gas-prices",children:"ETH Gas Prices"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://ethgas.watch"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://github.com/wslyvh/ethgaswatch",children:"wslyvh/ethgaswatch"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#joke",children:"Joke"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://v2.jokeapi.dev"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://sv443.net/privacypolicy/en",children:"SV443's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#chuck-norris-jokes",children:"Chuck Norris Jokes"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.chucknorris.io"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#xkcd-comic",children:"XKCD Comic"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://xkcd.vercel.app"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://xkcd.com",children:"XKCD"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#flight-data",children:"Flight Data"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://aerodatabox.p.rapidapi.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.aerodatabox.com/#h.p_cxtiyzwf_wqd",children:"AeroDataBox Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#astronomy-picture-of-the-day",children:"Astronomy Picture of the Day"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://apod.as93.net"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.nasa.gov/about/highlights/HP_Privacy.html",children:"NASA's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#github-trending",children:"GitHub Trending"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://trend.doforce.xyz"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#github-profile-stats",children:"GitHub Profile Stats"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://github-readme-stats.vercel.app"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#cron-monitoring-health-checks",children:"Cron Monitoring (Health Checks)"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://healthchecks.io"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://healthchecks.io/privacy/",children:"Health-Checks Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#hacker-news-trending",children:"Hacker News Trending"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://hacker-news.firebaseio.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.ycombinator.com/legal#privacy",children:"Y Combinator Privacy Policy"})})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#minecraft-server-status",children:"Minecraft Server Status"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://api.mcsrvstat.us"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#mvg",children:"MVG"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.mvg.de/api/fib/v2/"})}),(0,r.jsx)(s.td,{children:"No Policy Available"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"/docs/widgets#rescue-time",children:"RescueTime"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.code,{children:"https://www.rescuetime.com"})}),(0,r.jsx)(s.td,{children:(0,r.jsx)(s.a,{href:"https://www.rescuetime.com/privacy",children:"RescueTime Privacy Policy"})})]})]})]}),"\n",(0,r.jsx)(s.p,{children:"Note: There are also many widgets that connect to self-hosted services (such as Pi-hole, AdGuard, Glances, Nextcloud, Proxmox, Uptime Kuma, etc.). These only make requests to your own configured server addresses and do not contact any third-party services."}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"dependencies",children:"Dependencies"}),"\n",(0,r.jsxs)(s.p,{children:["As with most web projects, Dashy relies on several ",(0,r.jsx)(s.a,{href:"/docs/credits#dependencies",children:"dependencies"}),"."]}),"\n",(0,r.jsxs)(s.p,{children:["Dependencies can introduce security vulnerabilities, but since all these packages are open source any issues are usually very quickly spotted. Dashy is using Snyk for dependency security monitoring, and you can see ",(0,r.jsx)(s.a,{href:"https://snyk.io/test/github/lissy93/dashy",children:"the latest report here"}),". If any issue is detected by Snyk, a note about it will appear at the top of the Readme, and will usually be fixed within 48 hours."]}),"\n",(0,r.jsxs)(s.p,{children:["Note that packages listed under ",(0,r.jsx)(s.code,{children:"devDependencies"})," section are only used for building the project, and are not included in the production environment."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"securing-your-environment",children:"Securing your Environment"}),"\n",(0,r.jsx)(s.p,{children:"There is very little complexity involved with Dashy, and therefore the attack surface is reasonably small, but it is still important to follow best practices for all your self-hosted apps:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Use SSL/HTTPS"})," for securing traffic in transit, see ",(0,r.jsx)(s.a,{href:"/docs/management#ssl-certificates",children:"Management Docs: SSL Certificates"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Configure authentication"})," to prevent unauthorized access, see ",(0,r.jsx)(s.a,{href:"/docs/authentication",children:"Authentication Docs"}),". For internet-facing instances, use ",(0,r.jsx)(s.a,{href:"/docs/authentication#keycloak",children:"Keycloak"}),", ",(0,r.jsx)(s.a,{href:"/docs/authentication#oidc",children:"OIDC"}),", or an ",(0,r.jsx)(s.a,{href:"/docs/authentication#alternative-authentication-methods",children:"alternative server-side method"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Place behind a reverse proxy"})," if exposing to the internet, see ",(0,r.jsx)(s.a,{href:"/docs/management#network-exposure",children:"Management Docs: Network Exposure"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Harden your containers"})," if running in Docker, see ",(0,r.jsx)(s.a,{href:"/docs/management#container-security",children:"Management Docs: Container Security"})]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Keep Dashy and your system up-to-date"})," to ensure known vulnerabilities are patched"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Configure firewall rules"})," to restrict access to only necessary ports and networks"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.strong,{children:"Use a VPN"})," for private access without exposing Dashy to the public internet"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsxs)(s.strong,{children:["Follow ",(0,r.jsx)(s.a,{href:"https://docs.docker.com/engine/security/",children:"Docker security best practices"})]})," including running as non-root, limiting capabilities, and using read-only volumes"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"security-features",children:"Security Features"}),"\n",(0,r.jsx)(s.h3,{id:"subresource-integrity",children:"Subresource Integrity"}),"\n",(0,r.jsxs)(s.p,{children:[(0,r.jsx)(s.a,{href:"https://developer.mozilla.org/en-US/docs/Web/Security/Subresource_Integrity",children:"Subresource Integrity"})," or SRI is a security feature that enables browsers to verify that resources they fetch are delivered without unexpected manipulation. It works by allowing you to provide a cryptographic hash that a fetched resource must match. This prevents the app from loading any resources that have been manipulated, by verifying the files hashes. It safeguards against the risk of an attacker injecting arbitrary malicious content into any files served up via a CDN."]}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports SRI, and it is recommended to enable this if you are hosting your dashboard via a public CDN. To enable SRI, set the ",(0,r.jsx)(s.code,{children:"INTEGRITY"})," environmental variable to ",(0,r.jsx)(s.code,{children:"true"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"ssl",children:"SSL"}),"\n",(0,r.jsxs)(s.p,{children:["Native SSL support is enabled, for setup instructions, see the ",(0,r.jsx)(s.a,{href:"/docs/management#ssl-certificates",children:"Management Docs"})]}),"\n",(0,r.jsx)(s.h3,{id:"authentication",children:"Authentication"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy supports built-in auth, server-based SSO using Keycloak or any OIDC provider, and header-based authentication for reverse proxy setups. Full details of which, along with alternate authentication methods can be found in the ",(0,r.jsx)(s.a,{href:"/docs/authentication",children:"Authentication Docs"}),". If your dashboard is exposed to the internet and/ or contains any sensitive info it is strongly recommended to configure access control with Keycloak, OIDC, or another server-side method."]}),"\n",(0,r.jsx)(s.h3,{id:"configuration-lockdown",children:"Configuration Lockdown"}),"\n",(0,r.jsx)(s.p,{children:"Dashy provides several options to restrict what users can modify:"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.preventWriteToDisk"})," - Prevents config changes from being saved to the server"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.preventLocalSave"})," - Prevents config changes from being saved to browser storage"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.disableConfiguration"})," - Hides the config UI from all users"]}),"\n",(0,r.jsxs)(s.li,{children:[(0,r.jsx)(s.code,{children:"appConfig.disableConfigurationForNonAdmin"})," - Hides the config UI for non-admin users"]}),"\n"]}),"\n",(0,r.jsxs)(s.p,{children:["These can be combined with the ",(0,r.jsx)(s.code,{children:"admin"})," and ",(0,r.jsx)(s.code,{children:"normal"})," user roles to give fine-grained control. Admin users can save config changes, while normal users have read-only access. For more details, see the ",(0,r.jsx)(s.a,{href:"/docs/authentication#permissions",children:"Authentication Docs: Permissions"}),"."]}),"\n",(0,r.jsx)(s.h3,{id:"disabling-features",children:"Disabling Features"}),"\n",(0,r.jsx)(s.p,{children:"You may wish to disable features that you don't want to use, if they involve storing data in the browser or making network requests."}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["To disable smart-sort (uses local storage), set ",(0,r.jsx)(s.code,{children:"appConfig.disableSmartSort: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To disable update checks (makes external request to GH), set ",(0,r.jsx)(s.code,{children:"appConfig.disableUpdateChecks: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To disable web search (redirect to external / internal content), set ",(0,r.jsx)(s.code,{children:"appConfig.webSearch.disableWebSearch: true"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep status checks disabled (external/ internal requests), set ",(0,r.jsx)(s.code,{children:"appConfig.statusCheck: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep font-awesome icons disabled (external requests), set ",(0,r.jsx)(s.code,{children:"appConfig.enableFontAwesome: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep error reporting disabled (external requests and data collection), set ",(0,r.jsx)(s.code,{children:"appConfig.enableErrorReporting: false"})]}),"\n",(0,r.jsxs)(s.li,{children:["To keep the service worker disabled (stores cache of app in browser data), set ",(0,r.jsx)(s.code,{children:"appConfig.enableServiceWorker: false"})]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"threat-model",children:"Threat Model"}),"\n",(0,r.jsx)(s.p,{children:"Dashy is a statically-hosted dashboard application, designed to be self-hosted on a private network. This threat model outlines the intended deployment context, trust boundaries, known risks and accepted trade-offs, to help users assess whether Dashy is appropriate for their environment."}),"\n",(0,r.jsx)(s.h3,{id:"intended-deployment",children:"Intended Deployment"}),"\n",(0,r.jsxs)(s.p,{children:["Dashy is designed to run on a ",(0,r.jsx)(s.strong,{children:"private local network"})," (e.g. a home lab), accessed by a ",(0,r.jsx)(s.strong,{children:"small number of trusted users"}),". It is a convenience tool for organizing links to self-hosted services - it is not designed to protect sensitive resources or act as an access control layer."]}),"\n",(0,r.jsxs)(s.p,{children:["If exposed to the internet, Dashy ",(0,r.jsx)(s.strong,{children:"must"})," be placed behind a reverse proxy with server-side authentication (e.g. Authelia, Authentik, Cloudflare Access). The built-in client-side auth is a convenience feature for private networks, not a security boundary."]}),"\n",(0,r.jsx)(s.h3,{id:"trust-boundaries",children:"Trust Boundaries"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Boundary"}),(0,r.jsx)(s.th,{children:"Trusted Side"}),(0,r.jsx)(s.th,{children:"Untrusted Side"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Local network"}),(0,r.jsx)(s.td,{children:"LAN users, self-hosted services"}),(0,r.jsx)(s.td,{children:"The public internet"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsxs)(s.td,{children:["Config file (",(0,r.jsx)(s.code,{children:"conf.yml"}),")"]}),(0,r.jsx)(s.td,{children:"Server admin who writes the config"}),(0,r.jsx)(s.td,{children:"End users who view the dashboard"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Browser storage"}),(0,r.jsx)(s.td,{children:"The current browser session"}),(0,r.jsx)(s.td,{children:"Other domains, other users of the same device"})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"CORS proxy / status checks"}),(0,r.jsx)(s.td,{children:"Configured target URLs (set by admin)"}),(0,r.jsx)(s.td,{children:"Arbitrary URLs (if auth is not enabled)"})]})]})]}),"\n",(0,r.jsx)(s.h3,{id:"assets",children:"Assets"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Asset"}),(0,r.jsx)(s.th,{children:"Description"}),(0,r.jsx)(s.th,{children:"Storage"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Dashboard configuration"}),(0,r.jsx)(s.td,{children:"Service URLs, section layout, app settings"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server, optionally cached in browser localStorage"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"User credentials"}),(0,r.jsx)(s.td,{children:"SHA-256 password hashes, Keycloak/OIDC client IDs"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"API keys"}),(0,r.jsx)(s.td,{children:"Keys for widget services (weather, stocks, etc.)"}),(0,r.jsxs)(s.td,{children:[(0,r.jsx)(s.code,{children:"conf.yml"})," on server or environment variables"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"Auth tokens"}),(0,r.jsx)(s.td,{children:"Session token derived from credentials"}),(0,r.jsxs)(s.td,{children:["Browser cookie (",(0,r.jsx)(s.code,{children:"dashyAuthToken"}),")"]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:"User preferences"}),(0,r.jsx)(s.td,{children:"Theme, layout, language, collapsed sections"}),(0,r.jsx)(s.td,{children:"Browser localStorage"})]})]})]}),"\n",(0,r.jsx)(s.h3,{id:"when-dashy-is-not-the-right-choice",children:"When Dashy is NOT the Right Choice"}),"\n",(0,r.jsxs)(s.ul,{children:["\n",(0,r.jsxs)(s.li,{children:["You need a ",(0,r.jsx)(s.strong,{children:"multi-tenant"})," dashboard with per-user audit trails"]}),"\n",(0,r.jsxs)(s.li,{children:["You are deploying on the ",(0,r.jsx)(s.strong,{children:"public internet without a reverse proxy"})]}),"\n",(0,r.jsxs)(s.li,{children:["Your dashboard contains ",(0,r.jsx)(s.strong,{children:"secrets or credentials"})," that must be protected from all users who can reach the server"]}),"\n",(0,r.jsxs)(s.li,{children:["You require ",(0,r.jsx)(s.strong,{children:"FIPS-compliant"})," or ",(0,r.jsx)(s.strong,{children:"SOC 2"})," certified software"]}),"\n"]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"known-limitations",children:"Known Limitations"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(s.table,{children:[(0,r.jsx)(s.thead,{children:(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.th,{children:"Report"}),(0,r.jsx)(s.th,{children:"Response"})]})}),(0,r.jsxs)(s.tbody,{children:[(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Client-side auth can be bypassed via browser dev tools"'}),(0,r.jsxs)(s.td,{children:["Correct (only if neither ",(0,r.jsx)(s.code,{children:"ENABLE_HTTP_AUTH"})," is set, nor any other auth mode). Client-side auth is a convenience for private networks, not a security boundary. Use server-side auth for untrusted environments."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"CORS proxy can make requests to internal services"'}),(0,r.jsx)(s.td,{children:"Correct. The CORS proxy is designed to reach services on your network. It is protected by auth middleware when auth is enabled. Do not expose Dashy without auth on an untrusted network."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Status checks can be used for SSRF"'}),(0,r.jsxs)(s.td,{children:["Status check target URLs are set by the server admin in ",(0,r.jsx)(s.code,{children:"conf.yml"}),", not by end users. If auth is enabled, the endpoint is protected."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Password hashes are stored in plaintext in conf.yml"'}),(0,r.jsx)(s.td,{children:"They are SHA-256 hashes, not plaintext passwords. The config file should be readable only by the server admin, and protected by HTTP auth when served."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"localStorage/cookies are not encrypted"'}),(0,r.jsx)(s.td,{children:"Browser storage is scoped to the origin and inaccessible to other domains. On a shared device, use your browser's profile isolation."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No CSRF protection"'}),(0,r.jsx)(s.td,{children:"Dashy's state-changing operations (config save) are protected by auth middleware. CSRF is a low risk on a private network dashboard."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Docker container runs as root"'}),(0,r.jsxs)(s.td,{children:["The container is sandboxed. For hardened deployments, override with ",(0,r.jsx)(s.code,{children:"--user"})," flag or configure in your Docker Compose file as described in the ",(0,r.jsx)(s.a,{href:"/docs/management#container-security",children:"container security docs"}),"."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Auth cookie is not HttpOnly/Secure"'}),(0,r.jsxs)(s.td,{children:["The token is needed by client-side JavaScript for auth state. On a private network over plain HTTP, the ",(0,r.jsx)(s.code,{children:"Secure"})," flag would break auth. Use HTTPS + a reverse proxy to add these flags if needed."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Iframe/embed widget can load arbitrary URLs"'}),(0,r.jsxs)(s.td,{children:["The widget config is written by the server admin, not end users. If you don't trust your config authors, disable the config editor with ",(0,r.jsx)(s.code,{children:"disableConfiguration"}),"."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"RSS widget renders HTML content"'}),(0,r.jsx)(s.td,{children:"RSS content is sanitized with DOMPurify before rendering. Script tags, event handlers and other dangerous elements are stripped."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No Content-Security-Policy headers"'}),(0,r.jsx)(s.td,{children:"CSP should be configured at the reverse proxy layer, since the correct policy depends on which widgets and icon CDNs you use. Dashy can't set a universal CSP that works for all configurations."})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"Config backups are not encrypted at rest"'}),(0,r.jsxs)(s.td,{children:["Backups are stored server-side alongside the original config. If an attacker has filesystem access, they already have ",(0,r.jsx)(s.code,{children:"conf.yml"}),". Encryption at rest is the responsibility of the host OS/volume."]})]}),(0,r.jsxs)(s.tr,{children:[(0,r.jsx)(s.td,{children:'"No rate limiting on endpoints"'}),(0,r.jsx)(s.td,{children:"Rate limiting should be applied at the reverse proxy layer, where it can be tuned per-deployment. Dashy is not designed to be directly exposed to untrusted traffic."})]})]})]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"update--patch-policy",children:"Update & Patch Policy"}),"\n",(0,r.jsxs)(s.p,{children:["We follow Semantic Versioning for all releases. Security fixes are shipped as patch releases as quickly as possible and are published via immutable Git tags and Docker image tags. Users are encouraged to pin to a specific version in production and monitor releases on GitHub for security updates. The ",(0,r.jsx)(s.code,{children:":latest"})," Docker tag is provided for convenience but should not be relied on in production environments."]}),"\n",(0,r.jsx)(s.hr,{}),"\n",(0,r.jsx)(s.h2,{id:"reporting-a-security-issue",children:"Reporting a Security Issue"}),"\n",(0,r.jsxs)(s.p,{children:["Please see our ",(0,r.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/?tab=security-ov-file",children:"Security.md"})," doc for how to report issues.\nWe have an actively monitored security mailbox supporting PGP, as well as a GitHub Advisories vulnerability reporting program."]})]})}function h(e={}){const{wrapper:s}={...(0,t.R)(),...e.components};return s?(0,r.jsx)(s,{...e,children:(0,r.jsx)(l,{...e})}):l(e)}},8453(e,s,n){n.d(s,{R:()=>o,x:()=>d});var i=n(6540);const r={},t=i.createContext(r);function o(e){const s=i.useContext(t);return i.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function d(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:o(e.components),i.createElement(t.Provider,{value:s},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/9be9f402.8cfa9280.js b/assets/js/9be9f402.e16dc57d.js similarity index 99% rename from assets/js/9be9f402.8cfa9280.js rename to assets/js/9be9f402.e16dc57d.js index 3393bbb3..93b2b531 100644 --- a/assets/js/9be9f402.8cfa9280.js +++ b/assets/js/9be9f402.e16dc57d.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5552],{8803(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>t,default:()=>x,frontMatter:()=>l,metadata:()=>i,toc:()=>h});const i=JSON.parse('{"id":"widgets","title":"Widgets","description":"Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API.","source":"@site/docs/widgets.md","sourceDirName":".","slug":"/widgets","permalink":"/docs/widgets","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/widgets.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Icons","permalink":"/docs/icons"},"next":{"title":"Theming","permalink":"/docs/theming"}}');var r=s(4848),d=s(8453);const l={},t="Widgets",c={},h=[{value:"Contents",id:"contents",level:2},{value:"General Widgets",id:"general-widgets",level:2},{value:"Clock",id:"clock",level:3},{value:"Options",id:"options",level:4},{value:"Example",id:"example",level:4},{value:"Info",id:"info",level:4},{value:"Weather",id:"weather",level:3},{value:"Options",id:"options-1",level:4},{value:"Example",id:"example-1",level:4},{value:"Info",id:"info-1",level:4},{value:"Weather Forecast",id:"weather-forecast",level:3},{value:"Options",id:"options-2",level:4},{value:"Example",id:"example-2",level:4},{value:"Info",id:"info-2",level:4},{value:"RSS Feed",id:"rss-feed",level:3},{value:"Options",id:"options-3",level:4},{value:"Example",id:"example-3",level:4},{value:"Info",id:"info-3",level:4},{value:"Image",id:"image",level:3},{value:"Options",id:"options-4",level:4},{value:"Example",id:"example-4",level:4},{value:"Info",id:"info-4",level:4},{value:"Public IP",id:"public-ip",level:3},{value:"Options",id:"options-5",level:4},{value:"Example",id:"example-5",level:4},{value:"Supported providers",id:"supported-providers",level:4},{value:"Info",id:"info-5",level:4},{value:"IP Blacklist",id:"ip-blacklist",level:3},{value:"Options",id:"options-6",level:4},{value:"Example",id:"example-6",level:4},{value:"Info",id:"info-6",level:4},{value:"Domain Monitor",id:"domain-monitor",level:3},{value:"Options",id:"options-7",level:4},{value:"Example",id:"example-7",level:4},{value:"Info",id:"info-7",level:4},{value:"Crypto Watch List",id:"crypto-watch-list",level:3},{value:"Options",id:"options-8",level:4},{value:"Example",id:"example-8",level:4},{value:"Info",id:"info-8",level:4},{value:"Crypto Token Price History",id:"crypto-token-price-history",level:3},{value:"Options",id:"options-9",level:4},{value:"Example",id:"example-9",level:4},{value:"Info",id:"info-9",level:4},{value:"Wallet Balance",id:"wallet-balance",level:3},{value:"Options",id:"options-10",level:4},{value:"Example",id:"example-10",level:4},{value:"Info",id:"info-10",level:4},{value:"Code Stats",id:"code-stats",level:3},{value:"Options",id:"options-11",level:4},{value:"Example",id:"example-11",level:4},{value:"Info",id:"info-11",level:4},{value:"Mullvad Status",id:"mullvad-status",level:3},{value:"Options",id:"options-12",level:4},{value:"Example",id:"example-12",level:4},{value:"Info",id:"info-12",level:4},{value:"addy.io",id:"addyio",level:3},{value:"Options",id:"options-13",level:4},{value:"Example",id:"example-13",level:4},{value:"Info",id:"info-13",level:4},{value:"Vulnerability Feed",id:"vulnerability-feed",level:3},{value:"Options",id:"options-14",level:4},{value:"Example",id:"example-14",level:4},{value:"Info",id:"info-14",level:4},{value:"Exchange Rates",id:"exchange-rates",level:3},{value:"Options",id:"options-15",level:4},{value:"Example",id:"example-15",level:4},{value:"Info",id:"info-15",level:4},{value:"Public Holidays",id:"public-holidays",level:3},{value:"Options",id:"options-16",level:4},{value:"Example",id:"example-16",level:4},{value:"Info",id:"info-16",level:4},{value:"Covid-19 Status",id:"covid-19-status",level:3},{value:"Options",id:"options-17",level:4},{value:"Example",id:"example-17",level:4},{value:"Info",id:"info-17",level:4},{value:"Sports Scores",id:"sports-scores",level:3},{value:"Options",id:"options-18",level:4},{value:"Example",id:"example-18",level:4},{value:"Info",id:"info-18",level:4},{value:"News Headlines",id:"news-headlines",level:3},{value:"Options",id:"options-19",level:4},{value:"Example",id:"example-19",level:4},{value:"Info",id:"info-19",level:4},{value:"TFL Status",id:"tfl-status",level:3},{value:"Options",id:"options-20",level:4},{value:"Example",id:"example-20",level:4},{value:"Info",id:"info-20",level:4},{value:"Stock Price History",id:"stock-price-history",level:3},{value:"Options",id:"options-21",level:4},{value:"Example",id:"example-21",level:4},{value:"Info",id:"info-21",level:4},{value:"ETH Gas Prices",id:"eth-gas-prices",level:3},{value:"Options",id:"options-22",level:4},{value:"Example",id:"example-22",level:4},{value:"Info",id:"info-22",level:4},{value:"Joke",id:"joke",level:3},{value:"Options",id:"options-23",level:4},{value:"Example",id:"example-23",level:4},{value:"Info",id:"info-23",level:4},{value:"Chuck Norris quotes",id:"chuck-norris-quotes",level:3},{value:"Options",id:"options-24",level:4},{value:"Example",id:"example-24",level:4},{value:"Info",id:"info-24",level:4},{value:"XKCD Comics",id:"xkcd-comics",level:3},{value:"Options",id:"options-25",level:4},{value:"Example",id:"example-25",level:4},{value:"Info",id:"info-25",level:4},{value:"Flight Data",id:"flight-data",level:3},{value:"Options",id:"options-26",level:4},{value:"Example",id:"example-26",level:4},{value:"Info",id:"info-26",level:4},{value:"Astronomy Picture of the Day",id:"astronomy-picture-of-the-day",level:3},{value:"Options",id:"options-27",level:4},{value:"Example",id:"example-27",level:4},{value:"Info",id:"info-27",level:4},{value:"GitHub Trending",id:"github-trending",level:3},{value:"Options",id:"options-28",level:4},{value:"Example",id:"example-28",level:4},{value:"Info",id:"info-28",level:4},{value:"GitHub Profile Stats",id:"github-profile-stats",level:3},{value:"Options",id:"options-29",level:4},{value:"Example",id:"example-29",level:4},{value:"Info",id:"info-29",level:4},{value:"HealthChecks Status",id:"healthchecks-status",level:3},{value:"Options",id:"options-30",level:4},{value:"Info",id:"info-30",level:4},{value:"Hackernews Trending",id:"hackernews-trending",level:3},{value:"Options",id:"options-31",level:4},{value:"Example",id:"example-30",level:5},{value:"MVG Departure",id:"mvg-departure",level:3},{value:"Options",id:"options-32",level:4},{value:"Info",id:"info-31",level:4},{value:"MVG Connection",id:"mvg-connection",level:3},{value:"Options",id:"options-33",level:4},{value:"Info",id:"info-32",level:4},{value:"Custom List",id:"custom-list",level:3},{value:"Options",id:"options-34",level:4},{value:"Json Schema",id:"json-schema",level:4},{value:"Notes",id:"notes",level:4},{value:"Example",id:"example-31",level:4},{value:"Info",id:"info-33",level:4},{value:"Custom search",id:"custom-search",level:3},{value:"Options",id:"options-35",level:4},{value:"Notes",id:"notes-1",level:4},{value:"Example",id:"example-32",level:4},{value:"Info",id:"info-34",level:4},{value:"RescueTime Overview",id:"rescuetime-overview",level:3},{value:"Options",id:"options-36",level:4},{value:"Example",id:"example-33",level:4},{value:"Info",id:"info-35",level:4},{value:"Minecraft Server",id:"minecraft-server",level:3},{value:"Options",id:"options-37",level:4},{value:"Example",id:"example-34",level:4},{value:"Info",id:"info-36",level:4},{value:"Self-Hosted Services Widgets",id:"self-hosted-services-widgets",level:2},{value:"System Info",id:"system-info",level:3},{value:"Options",id:"options-38",level:4},{value:"Example",id:"example-35",level:4},{value:"Info",id:"info-37",level:4},{value:"Cron Monitoring (Health Checks)",id:"cron-monitoring-health-checks",level:3},{value:"Options",id:"options-39",level:4},{value:"Example",id:"example-36",level:4},{value:"Info",id:"info-38",level:4},{value:"CPU History (NetData)",id:"cpu-history-netdata",level:3},{value:"Options",id:"options-40",level:4},{value:"Example",id:"example-37",level:4},{value:"Info",id:"info-39",level:4},{value:"Memory History (NetData)",id:"memory-history-netdata",level:3},{value:"Options",id:"options-41",level:4},{value:"Example",id:"example-38",level:4},{value:"Info",id:"info-40",level:4},{value:"Load History (NetData)",id:"load-history-netdata",level:3},{value:"Options",id:"options-42",level:4},{value:"Example",id:"example-39",level:4},{value:"Info",id:"info-41",level:4},{value:"Pi-Hole Stats",id:"pi-hole-stats",level:3},{value:"Options",id:"options-43",level:4},{value:"Example",id:"example-40",level:4},{value:"Info",id:"info-42",level:4},{value:"Pi-Hole Stats v6",id:"pi-hole-stats-v6",level:3},{value:"Options",id:"options-44",level:4},{value:"Example",id:"example-41",level:4},{value:"Info",id:"info-43",level:4},{value:"Pi-Hole Queries",id:"pi-hole-queries",level:3},{value:"Options",id:"options-45",level:4},{value:"Example",id:"example-42",level:4},{value:"Info",id:"info-44",level:4},{value:"Pi-Hole Queries v6",id:"pi-hole-queries-v6",level:3},{value:"Options",id:"options-46",level:4},{value:"Example",id:"example-43",level:4},{value:"Info",id:"info-45",level:4},{value:"Pi-Hole Recent Traffic",id:"pi-hole-recent-traffic",level:3},{value:"Options",id:"options-47",level:4},{value:"Example",id:"example-44",level:4},{value:"Info",id:"info-46",level:4},{value:"Pi-Hole Recent Traffic v6",id:"pi-hole-recent-traffic-v6",level:3},{value:"Options",id:"options-48",level:4},{value:"Example",id:"example-45",level:4},{value:"Info",id:"info-47",level:4},{value:"Stat Ping Statuses",id:"stat-ping-statuses",level:3},{value:"Options",id:"options-49",level:4},{value:"Example",id:"example-46",level:4},{value:"Info",id:"info-48",level:4},{value:"Synology Download Station",id:"synology-download-station",level:3},{value:"Options",id:"options-50",level:4},{value:"Example",id:"example-47",level:4},{value:"Info",id:"info-49",level:4},{value:"AdGuard Home Block Stats",id:"adguard-home-block-stats",level:3},{value:"Options",id:"options-51",level:4},{value:"Example",id:"example-48",level:4},{value:"Info",id:"info-50",level:4},{value:"AdGuard Home Filters",id:"adguard-home-filters",level:3},{value:"Options",id:"options-52",level:4},{value:"Example",id:"example-49",level:4},{value:"Info",id:"info-51",level:4},{value:"AdGuard Home DNS Info",id:"adguard-home-dns-info",level:3},{value:"Options",id:"options-53",level:4},{value:"Example",id:"example-50",level:4},{value:"Info",id:"info-52",level:4},{value:"AdGuard Home Top Domains",id:"adguard-home-top-domains",level:3},{value:"Options",id:"options-54",level:4},{value:"Example",id:"example-51",level:4},{value:"Info",id:"info-53",level:4},{value:"Nextcloud User",id:"nextcloud-user",level:3},{value:"Options",id:"options-55",level:4},{value:"Example",id:"example-52",level:4},{value:"Info",id:"info-54",level:4},{value:"Nextcloud User Statuses",id:"nextcloud-user-statuses",level:3},{value:"Options",id:"options-56",level:4},{value:"Example",id:"example-53",level:4},{value:"Info",id:"info-55",level:4},{value:"Nextcloud Notifications",id:"nextcloud-notifications",level:3},{value:"Options",id:"options-57",level:4},{value:"Example",id:"example-54",level:4},{value:"Info",id:"info-56",level:4},{value:"Nextcloud System",id:"nextcloud-system",level:3},{value:"Options",id:"options-58",level:4},{value:"Example",id:"example-55",level:4},{value:"Info",id:"info-57",level:4},{value:"Nextcloud Stats",id:"nextcloud-stats",level:3},{value:"Options",id:"options-59",level:4},{value:"Example",id:"example-56",level:4},{value:"Info",id:"info-58",level:4},{value:"Nextcloud PHP OPcache Stats",id:"nextcloud-php-opcache-stats",level:3},{value:"Options",id:"options-60",level:4},{value:"Example",id:"example-57",level:4},{value:"Info",id:"info-59",level:4},{value:"ntfy stream",id:"ntfy-stream",level:3},{value:"Options",id:"options-61",level:4},{value:"Example",id:"example-58",level:4},{value:"Info",id:"info-60",level:4},{value:"Proxmox lists",id:"proxmox-lists",level:3},{value:"Options",id:"options-62",level:4},{value:"Example",id:"example-59",level:4},{value:"Info",id:"info-61",level:4},{value:"Troubleshooting",id:"troubleshooting",level:4},{value:"Sabnzbd",id:"sabnzbd",level:3},{value:"Options",id:"options-63",level:4},{value:"Example",id:"example-60",level:4},{value:"Info",id:"info-62",level:4},{value:"Gluetun VPN Info",id:"gluetun-vpn-info",level:3},{value:"Options",id:"options-64",level:4},{value:"Example",id:"example-61",level:4},{value:"Info",id:"info-63",level:4},{value:"Drone CI Builds",id:"drone-ci-builds",level:3},{value:"Options",id:"options-65",level:4},{value:"Example",id:"example-62",level:4},{value:"Info",id:"info-64",level:4},{value:"Filebrowser",id:"filebrowser",level:3},{value:"Options",id:"options-66",level:4},{value:"Example",id:"example-63",level:4},{value:"Widget Sections",id:"widget-sections",level:4},{value:"Info",id:"info-65",level:4},{value:"Linkding",id:"linkding",level:3},{value:"Options",id:"options-67",level:4},{value:"Example",id:"example-64",level:4},{value:"Info",id:"info-66",level:4},{value:"Uptime Kuma",id:"uptime-kuma",level:3},{value:"Options",id:"options-68",level:4},{value:"Example",id:"example-65",level:4},{value:"Info",id:"info-67",level:4},{value:"Uptime Kuma Status Page",id:"uptime-kuma-status-page",level:3},{value:"Options",id:"options-69",level:4},{value:"Example",id:"example-66",level:4},{value:"Info",id:"info-68",level:4},{value:"Tactical RMM",id:"tactical-rmm",level:3},{value:"Options",id:"options-70",level:4},{value:"Example",id:"example-67",level:4},{value:"Info",id:"info-69",level:4},{value:"System Resource Monitoring",id:"system-resource-monitoring",level:2},{value:"Glances",id:"glances",level:3},{value:"Options",id:"options-71",level:4},{value:"Info",id:"info-70",level:4},{value:"Screenshot",id:"screenshot",level:4},{value:"Current CPU Usage",id:"current-cpu-usage",level:3},{value:"Example",id:"example-68",level:4},{value:"Current CPU Usage Speedometer",id:"current-cpu-usage-speedometer",level:3},{value:"Example",id:"example-69",level:4},{value:"CPU Usage Per Core",id:"cpu-usage-per-core",level:3},{value:"Example",id:"example-70",level:4},{value:"CPU Usage History",id:"cpu-usage-history",level:3},{value:"Options",id:"options-72",level:4},{value:"Example",id:"example-71",level:4},{value:"Current Memory Usage",id:"current-memory-usage",level:3},{value:"Example",id:"example-72",level:4},{value:"Current Memory Usage Speedometer",id:"current-memory-usage-speedometer",level:3},{value:"Example",id:"example-73",level:4},{value:"Memory Usage History",id:"memory-usage-history",level:3},{value:"Options",id:"options-73",level:4},{value:"Example",id:"example-74",level:4},{value:"Disk Space",id:"disk-space",level:3},{value:"Example",id:"example-75",level:4},{value:"Disk IO",id:"disk-io",level:3},{value:"Example",id:"example-76",level:4},{value:"System Load",id:"system-load",level:3},{value:"Example",id:"example-77",level:4},{value:"Uptime",id:"uptime",level:3},{value:"Example",id:"example-78",level:4},{value:"System Load History",id:"system-load-history",level:3},{value:"Example",id:"example-79",level:4},{value:"Network Interfaces",id:"network-interfaces",level:3},{value:"Example",id:"example-80",level:4},{value:"Network Traffic",id:"network-traffic",level:3},{value:"Example",id:"example-81",level:4},{value:"Resource Usage Alerts",id:"resource-usage-alerts",level:3},{value:"Example",id:"example-82",level:4},{value:"IP Address",id:"ip-address",level:3},{value:"Example",id:"example-83",level:4},{value:"CPU Temp",id:"cpu-temp",level:3},{value:"Options",id:"options-74",level:4},{value:"Example",id:"example-84",level:4},{value:"Compact Metrics",id:"compact-metrics",level:3},{value:"Options",id:"options-75",level:4},{value:"Example",id:"example-85",level:4},{value:"Dynamic Widgets",id:"dynamic-widgets",level:2},{value:"Iframe Widget",id:"iframe-widget",level:3},{value:"Options",id:"options-76",level:4},{value:"Example",id:"example-86",level:4},{value:"HTML Embedded Widget",id:"html-embedded-widget",level:3},{value:"Options",id:"options-77",level:4},{value:"Example",id:"example-87",level:4},{value:"API Response",id:"api-response",level:3},{value:"Prometheus Data",id:"prometheus-data",level:3},{value:"Data Feed",id:"data-feed",level:3},{value:"Example",id:"example-88",level:4},{value:"Usage & Customizations",id:"usage--customizations",level:2},{value:"Widget Usage Guide",id:"widget-usage-guide",level:3},{value:"Continuous Updates",id:"continuous-updates",level:3},{value:"Proxying Requests",id:"proxying-requests",level:3},{value:"Handling Secrets",id:"handling-secrets",level:3},{value:"Setting Timeout",id:"setting-timeout",level:3},{value:"Adding Labels",id:"adding-labels",level:3},{value:"Ignoring Errors",id:"ignoring-errors",level:3},{value:"Widget Styling",id:"widget-styling",level:3},{value:"Customizing Charts",id:"customizing-charts",level:3},{value:"Language Translations",id:"language-translations",level:3},{value:"Widget UI Options",id:"widget-ui-options",level:3},{value:"Build your own Widget",id:"build-your-own-widget",level:3},{value:"Requesting a Widget",id:"requesting-a-widget",level:3},{value:"Troubleshooting Widget Errors",id:"troubleshooting-widget-errors",level:3},{value:"CORS Errors",id:"cors-errors",level:4},{value:"Option 1 - Ensure Correct Protocol",id:"option-1---ensure-correct-protocol",level:4},{value:"Option 2 - Set Headers",id:"option-2---set-headers",level:4},{value:"Option 3 - Proxying Request",id:"option-3---proxying-request",level:4},{value:"Option 4 - Use a plugin",id:"option-4---use-a-plugin",level:4},{value:"Raising an Issue",id:"raising-an-issue",level:3}];function o(e){const n={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",h5:"h5",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,d.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"widgets",children:"Widgets"})}),"\n",(0,r.jsx)(n.p,{children:"Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API."}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#general-widgets",children:"General Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#clock",children:"Clock"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather",children:"Weather"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather-forecast",children:"Weather Forecast"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#rss-feed",children:"RSS Feed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#image",children:"Image"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-ip",children:"Public IP Address"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ip-blacklist",children:"IP Blacklist Checker"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#domain-monitor",children:"Domain Monitor"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#crypto-watch-list",children:"Crypto Watch List"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#crypto-token-price-history",children:"Crypto Price History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#wallet-balance",children:"Crypto Wallet Balance"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#code-stats",children:"Code Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mullvad-status",children:"Mullvad Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#addyio",children:"Email Aliases (addy.io)"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#vulnerability-feed",children:"Vulnerability Feed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#exchange-rates",children:"Exchange Rates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-holidays",children:"Public Holidays"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#covid-19-status",children:"Covid-19 Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sports-scores",children:"Sports Scores"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#news-headlines",children:"News Headlines"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tfl-status",children:"TFL Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#stock-price-history",children:"Stock Price History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#eth-gas-prices",children:"ETH Gas Prices"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#joke",children:"Joke of the Day"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#chuck-norris-quotes",children:"Chuck Norris quotes"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#xkcd-comics",children:"XKCD Comics"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#flight-data",children:"Flight Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#astronomy-picture-of-the-day",children:"NASA APOD"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#github-trending",children:"GitHub Trending"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#github-profile-stats",children:"GitHub Profile Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthchecks-status",children:"Healthchecks Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#hackernews-trending",children:"Hackernews Trending"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mvg-departure",children:"Mvg Departure"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mvg-connection",children:"Mvg Connection"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#custom-search",children:"Custom search"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#rescuetime-overview",children:"Rescuetime overview"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#minecraft-server",children:"Minecraft Server"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#self-hosted-services-widgets",children:"Self-Hosted Services Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-info",children:"System Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cron-monitoring-health-checks",children:"Cron Monitoring"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-history-netdata",children:"CPU History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#memory-history-netdata",children:"Memory History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#load-history-netdata",children:"System Load History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-stats",children:"Pi-Hole Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-stats-v6",children:"Pi-Hole Stats v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-queries",children:"Pi-Hole Queries"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-queries-v6",children:"Pi-Hole Queries v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-recent-traffic",children:"Pi-Hole Recent Traffic"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-recent-traffic-v6",children:"Pi-Hole Recent Traffic v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#stat-ping-statuses",children:"Stat Ping Statuses"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#synology-download-station",children:"Synology Download Station"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-block-stats",children:"AdGuard Home Block Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-filters",children:"AdGuard Home Filters"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-dns-info",children:"AdGuard Home DNS Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-top-domains",children:"AdGuard Home Top Domains"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-user",children:"Nextcloud User"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-user-statuses",children:"Nextcloud User Statuses"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-notifications",children:"Nextcloud Notifications"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-system",children:"Nextcloud System"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-stats",children:"Nextcloud Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-php-opcache-stats",children:"Nextcloud PHP OPcache"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ntfy-stream",children:"Ntfy stream"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#proxmox-lists",children:"Proxmox lists"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sabnzbd",children:"Sabnzbd"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#gluetun-vpn-info",children:"Gluetun VPN Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#drone-ci-builds",children:"Drone CI Build"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#filebrowser",children:"Filebrowser"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#linkding",children:"Linkding"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#uptime-kuma",children:"Uptime Kuma"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#uptime-kuma-status-page",children:"Uptime Kuma Status Page"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tactical-rmm",children:"Tactical RMM"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#system-resource-monitoring",children:"System Resource Monitoring"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#current-cpu-usage",children:"CPU Usage Current"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-usage-per-core",children:"CPU Usage Per Core"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-usage-history",children:"CPU Usage History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#current-memory-usage",children:"Memory Usage Current"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#memory-usage-history",children:"Memory Usage History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disk-space",children:"Disk Space"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disk-io",children:"Disk IO"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-load",children:"System Load"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-load-history",children:"System Load History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-interfaces",children:"Network Interfaces"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-traffic",children:"Network Traffic"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#resource-usage-alerts",children:"Resource Usage Alerts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ip-address",children:"Public & Private IP"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-temp",children:"CPU Temperature"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#compact-metrics",children:"Compact Metrics"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#dynamic-widgets",children:"Dynamic Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#iframe-widget",children:"Iframe Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"HTML Embed Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#api-response",children:"API Response"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#prometheus-data",children:"Prometheus Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#data-feed",children:"Data Feed"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#usage--customizations",children:"Usage & Customizations"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-usage-guide",children:"Widget Usage Guide"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#continuous-updates",children:"Continuous Updates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#proxying-requests",children:"Proxying Requests"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#setting-timeout",children:"Setting Timeout"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adding-labels",children:"Adding Labels"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ignoring-errors",children:"Ignoring Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-styling",children:"Custom CSS Styling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#customizing-charts",children:"Customizing Charts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#language-translations",children:"Language Translations"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-ui-options",children:"Widget UI Options"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#build-your-own-widget",children:"Building a Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#requesting-a-widget",children:"Requesting a Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#troubleshooting-widget-errors",children:"Troubleshooting"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.h2,{id:"general-widgets",children:"General Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"clock",children:"Clock"}),"\n",(0,r.jsx)(n.p,{children:"A simple, live-updating time and date widget with time-zone support. All fields are optional."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/clock"})}),"\n",(0,r.jsx)(n.h4,{id:"options",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"timeZone"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The time zone to display date and time in.",(0,r.jsx)(n.br,{})," Specified as Region/City, for example: ",(0,r.jsx)(n.code,{children:"Australia/Melbourne"}),". See the ",(0,r.jsx)(n.a,{href:"https://timezonedb.com/time-zones",children:"Time Zone DB"})," for a full list of supported TZs. Defaults to the browser / device's local time"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"format"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["A country code for displaying the date and time in local format.",(0,r.jsx)(n.br,{}),"Specified as ",(0,r.jsx)(n.code,{children:"[ISO-3166]-[ISO-639]"}),", for example: ",(0,r.jsx)(n.code,{children:"en-AU"}),". See ",(0,r.jsx)(n.a,{href:"https://www.fincher.org/Utilities/CountryLanguageList.shtml",children:"here"})," for a full list of locales. Defaults to the browser / device's region"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"customCityName"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"By default the city from the time-zone is shown, but setting this value will override that text"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDate"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the date and city will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideSeconds"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", seconds will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"use12Hour"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", 12 hour time will be displayed. Defaults to the settings suggested by the current ",(0,r.jsx)(n.code,{children:"format"})," and ",(0,r.jsx)(n.code,{children:"timeZone"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: clock\n options:\n timeZone: Europe/London\n format: en-GB\n hideDate: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No external data requests."})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"weather",children:"Weather"}),"\n",(0,r.jsx)(n.p,{children:"A simple, live-updating local weather component, showing temperature, conditions and more info."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/weather"})}),"\n",(0,r.jsx)(n.h4,{id:"options-1",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your OpenWeatherMap API key. You can get one for free at ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"city"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A city name to use for fetching weather. This can also be a state code or country code, following the ISO-3166 format"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cityId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An OpenWeatherMap numeric city ID, used to disambiguate cities that share a name. You can find the ID in the URL of the city's page on ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," (e.g. ",(0,r.jsx)(n.code,{children:"2643743"})," for London, GB). If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," option"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The units to use for displaying data, can be either ",(0,r.jsx)(n.code,{children:"metric"})," or ",(0,r.jsx)(n.code,{children:"imperial"}),". Defaults to ",(0,r.jsx)(n.code,{children:"metric"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the additional details (wind, humidity, pressure, etc) will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lat"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To show weather for a specific location, you can provide the latitude and longitude coordinates. If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lon"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To show weather for a specific location, you can provide the latitude and longitude coordinates. If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-1",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n city: London\n units: metric\n hideDetails: true\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-1",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"weather-forecast",children:"Weather Forecast"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the weather (temperature and conditions) for the next few days for a given location. Note that this requires either the free ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"OpenWeatherMap Student Plan"}),", or the Premium Plan."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/weather-forecast"})}),"\n",(0,r.jsx)(n.h4,{id:"options-2",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your OpenWeatherMap API key. You can get one at ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," or for free via the ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"OWM Student Plan"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"city"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A city name to use for fetching weather. This can also be a state code or country code, following the ISO-3166 format"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cityId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An OpenWeatherMap numeric city ID, used to disambiguate cities that share a name. You can find the ID in the URL of the city's page on ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," (e.g. ",(0,r.jsx)(n.code,{children:"2643743"})," for London, GB). If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," option"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lat"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Latitude for a specific location. If provided alongside ",(0,r.jsx)(n.code,{children:"lon"}),", this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lon"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Longitude for a specific location. If provided alongside ",(0,r.jsx)(n.code,{children:"lat"}),", this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of days to display of forecast info to display. Defaults to ",(0,r.jsx)(n.code,{children:"4"}),", max ",(0,r.jsx)(n.code,{children:"16"})," days"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The units to use for displaying data, can be either ",(0,r.jsx)(n.code,{children:"metric"})," or ",(0,r.jsx)(n.code,{children:"imperial"}),". Defaults to ",(0,r.jsx)(n.code,{children:"metric"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the additional details (wind, humidity, pressure, etc) will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-2",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather-forecast\n options:\n city: California\n numDays: 6\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n units: imperial\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-2",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udd34 Premium (free for personal use only)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"rss-feed",children:"RSS Feed"}),"\n",(0,r.jsx)(n.p,{children:"Display news and updates from any RSS-enabled service."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/rss"})}),"\n",(0,r.jsx)(n.h4,{id:"options-3",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"rssUrl"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL location of your RSS feed"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An API key for ",(0,r.jsx)(n.a,{href:"https://rss2json.com/",children:"rss2json"}),". It's free, and will allow you to make 10,000 requests per day, you can sign up ",(0,r.jsx)(n.a,{href:"https://rss2json.com/sign-up",children:"here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The number of posts to return. If you haven't specified an API key, this will be limited to 10"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"orderBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["How results should be sorted. Can be either ",(0,r.jsx)(n.code,{children:"pubDate"}),", ",(0,r.jsx)(n.code,{children:"author"})," or ",(0,r.jsx)(n.code,{children:"title"}),". Defaults to ",(0,r.jsx)(n.code,{children:"pubDate"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"orderDirection"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Order direction of feed items to return. Can be either ",(0,r.jsx)(n.code,{children:"asc"})," or ",(0,r.jsx)(n.code,{children:"desc"}),". Defaults to ",(0,r.jsx)(n.code,{children:"desc"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"parseLocally"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If true parse the rss feed locally instead of using the rss2json API."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-3",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: rss-feed\n options:\n rssUrl: https://www.schneier.com/blog/atom.xml\n apiKey: xxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-3",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan (up to 10,000 requests / day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://rss2json.com/privacy-policy",children:"Rss2Json Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"image",children:"Image"}),"\n",(0,r.jsx)(n.p,{children:"Displays an image."}),"\n",(0,r.jsxs)(n.p,{children:["This may be useful if you have a service (such as Grafana - ",(0,r.jsx)(n.a,{href:"https://mattionline.de/grafana-api-export-graph-as-png/",children:"see example"}),"), which periodically exports charts or other data as an image."]}),"\n",(0,r.jsxs)(n.p,{children:["You can also store images within Dashy's public directory (using a Docker volume), and reference them directly. E.g. ",(0,r.jsx)(n.code,{children:"-v ./path/to/my-homelab-logo.png:/app/public/logo.png"}),", then in the widget ",(0,r.jsx)(n.code,{children:"imagePath: /logo.png"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Similarly, any web service that serves up widgets as image can be used. E.g. you could show current star chart for a GitHub repo, with: ",(0,r.jsx)(n.code,{children:"imagePath: https://starchart.cc/Lissy93/dashy.svg"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you'd like to embed a live screenshot, of all or just part of a website, then this can be done using ",(0,r.jsx)(n.a,{href:"https://apiflash.com/",children:"API Flash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Or what about showing a photo of the day? Try ",(0,r.jsx)(n.code,{children:"https://source.unsplash.com/random/400x300"})," or ",(0,r.jsx)(n.code,{children:"https://picsum.photos/400/300"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"300",src:"https://pixelflare.cc/alicia/dashy/image"})}),"\n",(0,r.jsx)(n.h4,{id:"options-4",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imagePath"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The path (local or remote) of the image to display"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imageWidth"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify a fixed width for rendered image. Accepts either integer value in ",(0,r.jsx)(n.code,{children:"px"}),", or any string value with units (e.g. ",(0,r.jsx)(n.code,{children:"420"}),", ",(0,r.jsx)(n.code,{children:"100px"}),", ",(0,r.jsx)(n.code,{children:"6.9rem"}),") (defaults to ",(0,r.jsx)(n.code,{children:"auto"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imageHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify a fixed height for rendered image. Accepts either integer value in ",(0,r.jsx)(n.code,{children:"px"}),", or any string value with units (e.g. ",(0,r.jsx)(n.code,{children:"420"}),", ",(0,r.jsx)(n.code,{children:"100px"}),", ",(0,r.jsx)(n.code,{children:"6.9rem"}),") (defaults to ",(0,r.jsx)(n.code,{children:"auto"}),")"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-4",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: image\n options:\n imagePath: https://i.ibb.co/yhbt6CY/dashy.png\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-4",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:"Unless image fetched from remote source, no external data request is made."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"public-ip",children:"Public IP"}),"\n",(0,r.jsx)(n.p,{children:"Displays your public IP address, ISP and approximate location."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/public-ip"})}),"\n",(0,r.jsx)(n.h4,{id:"options-5",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"provider"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["One of ",(0,r.jsx)(n.code,{children:"ipinfo"})," ",(0,r.jsx)(n.em,{children:"(default)"}),", ",(0,r.jsx)(n.code,{children:"freeipapi"}),", ",(0,r.jsx)(n.code,{children:"ipquery"}),", ",(0,r.jsx)(n.code,{children:"ip-api"}),", or ",(0,r.jsx)(n.code,{children:"ipgeolocation"}),". See the table below"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Required for ",(0,r.jsx)(n.code,{children:"ipgeolocation"}),". Optional for ",(0,r.jsx)(n.code,{children:"ipinfo"})," (a ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/signup",children:"free token"})," raises the rate limit from ~1k/day to ~50k/month)"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLocation"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"})," to hide the flag, ISP name and city/region \u2014 only the IP address is shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-5",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"Default (no options needed)"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Using ",(0,r.jsx)(n.code,{children:"ip-api"})," via the proxy (gets server IP, because of ",(0,r.jsx)(n.code,{children:"useProxy"}),")"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n useProxy: true\n options:\n provider: ip-api\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Using ",(0,r.jsx)(n.code,{children:"ipgeolocation"})," with a key:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n options:\n provider: ipgeolocation\n apiKey: xxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["Setting ",(0,r.jsx)(n.code,{children:"useProxy: true"})," makes the lookup happen from Dashy's server instead of your browser, so the upstream API returns the public IP of the machine running Dashy, whereas without ",(0,r.jsx)(n.code,{children:"useProxy"})," it will show your public IP."]})}),"\n",(0,r.jsx)(n.h4,{id:"supported-providers",children:"Supported providers"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Provider"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Key"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Proxy"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Notes"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"ipinfo"})," ",(0,r.jsx)(n.em,{children:"(default)"})]}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed / \ud83d\udfe0 Optional"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Service from ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/",children:"ipinfo.io"}),". Keyless gives ~1k/day; a ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/signup",children:"free token"})," raises this to ~50k/month. Sometimes blocked by adblockers."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"freeipapi"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Keyless, 60 req/min via ",(0,r.jsx)(n.a,{href:"https://freeipapi.com/",children:"freeipapi.com"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ipquery"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Modern keyless API from ",(0,r.jsx)(n.a,{href:"https://ipquery.io/",children:"ipquery.io"}),". Includes VPN/Tor/datacenter risk flags in the raw response."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ip-api"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["\ud83d\udd34 ",(0,r.jsx)(n.strong,{children:"Required"})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.a,{href:"https://ip-api.com/",children:"ip-api.com"})," is HTTP-only on the free tier, so the proxy is needed to avoid mixed-content errors."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ipgeolocation"})}),(0,r.jsxs)(n.td,{children:["\ud83d\udd34 ",(0,r.jsx)(n.strong,{children:"Required"})]}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Get a free key from ",(0,r.jsx)(n.a,{href:"https://ipgeolocation.io/signup.html",children:"ipgeolocation.io"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"info-5",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled (only ",(0,r.jsx)(n.code,{children:"provider: ip-api"})," needs the proxy)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional (only ",(0,r.jsx)(n.code,{children:"ipgeolocation"})," requires a key, and ",(0,r.jsx)(n.code,{children:"ipinfo"})," is optional)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": See the policy for whichever provider you choose: ",(0,r.jsx)(n.a,{href:"https://freeipapi.com/",children:"freeipapi.com"}),", ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/privacy-policy",children:"ipinfo.io"}),", ",(0,r.jsx)(n.a,{href:"https://ipquery.io/",children:"ipquery.io"}),", ",(0,r.jsx)(n.a,{href:"https://ip-api.com/docs/legal",children:"ip-api.com"}),", ",(0,r.jsx)(n.a,{href:"https://ipgeolocation.io/privacy.html",children:"IPGeoLocation"}),"."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ip-blacklist",children:"IP Blacklist"}),"\n",(0,r.jsxs)(n.p,{children:["Notice certain web pages aren't loading? This widget quickly shows which blacklists your IP address (or host, or email) appears on, using data from ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/",children:"blacklistchecker.com"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/ip-blacklist"})}),"\n",(0,r.jsx)(n.h4,{id:"options-6",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"ipAddress"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The IP to check. This can also be a domain/ host name or even an email address. If left blank, Dashy will use your current public IP address."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["You can get your free API key from ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/keys",children:"blacklistchecker.com"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-6",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: blacklist-check\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n ipAddress: 1.1.1.1\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-6",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/privacy",children:"BlacklistChecker Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"domain-monitor",children:"Domain Monitor"}),"\n",(0,r.jsxs)(n.p,{children:["Keep an eye on the expiry dates of your domain names, using public whois records fetched from ",(0,r.jsx)(n.a,{href:"https://whoapi.com/",children:"whoapi.com"}),". Click the domain name to view additional info, like registrar, name servers and date last updated."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/domain-monitor"})}),"\n",(0,r.jsx)(n.h4,{id:"options-7",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"domain"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The domain to check"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["You can get your free API key from ",(0,r.jsx)(n.a,{href:"https://my.whoapi.com/user/signup",children:"my.whoapi.com"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showFullInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If set to true, the toggle-full-info panel will be open by default"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-7",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: domain-monitor\n options:\n domain: example.com\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n\n - type: domain-monitor\n options:\n domain: example2.com\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-7",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan (10,000 requests)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://whoapi.com/privacy-policy/",children:"WhoAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"crypto-watch-list",children:"Crypto Watch List"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of price changes of your favorite crypto assets. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/",children:"CoinGecko"}),". All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/crypto-prices"})}),"\n",(0,r.jsx)(n.h4,{id:"options-8",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"assets"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An array of cryptocurrencies, coins and tokens. See ",(0,r.jsx)(n.a,{href:"https://api.coingecko.com/api/v3/asset_platforms",children:"list of supported assets"}),". If none are specified, then the top coins by ",(0,r.jsx)(n.code,{children:"sortBy"})," (defaults to market cap) will be returned"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"currency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The fiat currency to display price in, expressed as an ISO-4217 alpha code (see ",(0,r.jsx)(n.a,{href:"https://www.iban.com/currency-codes",children:"list of currencies"}),"). Defaults to ",(0,r.jsx)(n.code,{children:"USD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The method of sorting results. Can be ",(0,r.jsx)(n.code,{children:"marketCap"}),", ",(0,r.jsx)(n.code,{children:"volume"})," or ",(0,r.jsx)(n.code,{children:"alphabetical"}),". Defaults to ",(0,r.jsx)(n.code,{children:"marketCap"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Number of results to return, useful when no assets are specified. Defaults to either ",(0,r.jsx)(n.code,{children:"all"})," or ",(0,r.jsx)(n.code,{children:"100"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-8",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: crypto-watch-list\n options:\n limit: 10\n"})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: crypto-watch-list\n options:\n currency: GBP\n sortBy: marketCap\n assets:\n - bitcoin\n - ethereum\n - monero\n - cosmos\n - polkadot\n - dogecoin\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-8",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"crypto-token-price-history",children:"Crypto Token Price History"}),"\n",(0,r.jsxs)(n.p,{children:["Shows recent price history for a given crypto asset, using price data fetched from ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/",children:"CoinGecko"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/crypto-price-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-9",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"asset"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Name of a crypto asset, coin or token to fetch price data for, see ",(0,r.jsx)(n.a,{href:"https://api.coingecko.com/api/v3/asset_platforms",children:"list of supported assets"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"currency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The fiat currency to display results in, expressed as an ISO-4217 alpha code (see ",(0,r.jsx)(n.a,{href:"https://www.iban.com/currency-codes",children:"list of currencies"}),"). Defaults to ",(0,r.jsx)(n.code,{children:"USD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of days of price history to render. Defaults to ",(0,r.jsx)(n.code,{children:"7"}),", min: ",(0,r.jsx)(n.code,{children:"1"}),", max: ",(0,r.jsx)(n.code,{children:"30"})," days"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value. Defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"})," which inherits dashboard primary color"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-9",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: crypto-price-chart\n options:\n asset: bitcoin\n currency: GBP\n numDays: 7\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-9",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"wallet-balance",children:"Wallet Balance"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of your crypto balances and see recent transactions. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.blockcypher.com/dev/",children:"BlockCypher"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/crypto-wallet"})}),"\n",(0,r.jsx)(n.h4,{id:"options-10",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"coin"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Symbol of coin or asset, e.g. ",(0,r.jsx)(n.code,{children:"btc"}),", ",(0,r.jsx)(n.code,{children:"eth"})," or ",(0,r.jsx)(n.code,{children:"doge"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"address"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Address to monitor. This is your wallet's ",(0,r.jsx)(n.strong,{children:"public"})," / receiving address"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"network"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To use a different network, other than mainnet. Defaults to ",(0,r.jsx)(n.code,{children:"main"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of transactions to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"}),", set to large number to show all"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-10",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: wallet-balance\n options:\n coin: btc\n address: 3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-10",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.blockcypher.com/privacy.html",children:"BlockCypher Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"code-stats",children:"Code Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Display your coding summary. ",(0,r.jsx)(n.a,{href:"https://codestats.net/",children:"Code::Stats"})," is a free and open source app that aggregates statistics about your programming activity. Dashy supports both the public instance, as well as self-hosted versions."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/code-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-11",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Your CodeStats username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If your self-hosting CodeStats, then supply the host name. By default it will use the public hosted instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monthsToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the number of months to render in the historical data chart. Defaults to ",(0,r.jsx)(n.code,{children:"6"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMeta"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the meta section (username, level, all-time and recent XP)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideHistory"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the historical calendar heat map"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLanguages"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the programming languages pie chart"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMachines"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the machines percentage chart"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-11",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: code-stats\n options:\n username: alicia\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-11",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://codestats.net/tos#privacy",children:"Code::Stats Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mullvad-status",children:"Mullvad Status"}),"\n",(0,r.jsxs)(n.p,{children:["Shows your Mullvad VPN connection status, as well as server info. Fetched from ",(0,r.jsx)(n.a,{href:"https://mullvad.net/en/check/",children:"am.i.mullvad.net"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/mullvad"})}),"\n",(0,r.jsx)(n.h4,{id:"options-12",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No Options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-12",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mullvad-status\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-12",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://mullvad.net/en/help/privacy-policy/",children:"Mullvad Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"addyio",children:"addy.io"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://addy.io/",children:"addy.io"})," is a free and open source mail forwarding service. Use it to protect your real email address, by using a different alias for each of your online accounts, and have all emails land in your normal inbox(es). Supports custom domains, email replies, PGP-encryption, multiple recipients and more"]}),"\n",(0,r.jsx)(n.p,{children:"This widget display email addresses / aliases from addy.io. Click an email address to copy to clipboard, or use the toggle switch to enable/ disable it. Shows usage stats (bandwidth, used aliases etc), as well as total messages received, blocked and sent. Works with both self-hosted and managed instances of addy.io."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/addy"})}),"\n",(0,r.jsx)(n.h4,{id:"options-13",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your addy.io API Key / Personal Access Token. You can generate this under ",(0,r.jsx)(n.a,{href:"https://app.addy.io/settings/api",children:"API Settings"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If your self-hosting addy.io, then supply the host name. By default it will use the public hosted instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you're using an API version that is not version ",(0,r.jsx)(n.code,{children:"v1"}),", then specify it here"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of emails shown per page. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the sort order for email addresses. Defaults to ",(0,r.jsx)(n.code,{children:"updated_at"}),". Can be either: ",(0,r.jsx)(n.code,{children:"local_part"}),", ",(0,r.jsx)(n.code,{children:"domain"}),", ",(0,r.jsx)(n.code,{children:"email"}),", ",(0,r.jsx)(n.code,{children:"emails_forwarded"}),", ",(0,r.jsx)(n.code,{children:"emails_blocked"}),", ",(0,r.jsx)(n.code,{children:"emails_replied"}),", ",(0,r.jsx)(n.code,{children:"emails_sent"}),", ",(0,r.jsx)(n.code,{children:"created_at"}),", ",(0,r.jsx)(n.code,{children:"updated_at"})," or ",(0,r.jsx)(n.code,{children:"deleted_at"}),". Precede with a ",(0,r.jsx)(n.code,{children:"-"})," character to reverse order."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"searchTerm"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A search term to filter results by, will search the email, description and domain"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"disableControls"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Prevent any changes being made to account through the widget. User will not be able to enable or disable aliases through UI when this option is set"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMeta"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show account meta info (forward/ block count, quota usage etc)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideAliases"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show email address / alias list. Will only show account meta info"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-13",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:' - type: addy\n options:\n apiKey: "xxxxxxxxxxxxxxxxxxxxxxxx\\\n xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\\\n xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"\n limit: 5\n sortBy: created_at\n disableControls: true\n'})}),"\n",(0,r.jsx)(n.h4,{id:"info-13",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free for Self-Hosted / Free Plan available on managed instance or $1/month for premium"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://addy.io/privacy/",children:"addy.io Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"vulnerability-feed",children:"Vulnerability Feed"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of recent security advisories and vulnerabilities, with optional filtering by score, exploits, vendor and product. Using data from ",(0,r.jsx)(n.a,{href:"https://nvd.nist.gov/developers/vulnerabilities",children:"NIST Vulnerability API"}),". All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cve"})}),"\n",(0,r.jsx)(n.h4,{id:"options-14",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cveTag"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVE records that include the provided cveTag. Options are ",(0,r.jsx)(n.strong,{children:"disputed"}),", ",(0,r.jsx)(n.strong,{children:"unsupported-when-assigned"})," or ",(0,r.jsx)(n.strong,{children:"exclusively-hosted-service"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of results to fetch. Can be between ",(0,r.jsx)(n.code,{children:"5"})," and ",(0,r.jsx)(n.code,{children:"30"}),", defaults to ",(0,r.jsx)(n.code,{children:"5"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV2Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv2 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV3Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv3 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV4Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv4 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"keywordSearch"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"This parameter returns only the CVEs where a word or phrase is found in the current description"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Without a key you're limited to 5 requests every 30 seconds. You can get a free API key from ",(0,r.jsx)(n.a,{href:"https://nvd.nist.gov/developers/request-an-api-key",children:"here"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-14",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: cve-vulnerabilities\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: cve-vulnerabilities\n options:\n cveTag: disputed\n cvssV2Severity: CRITICAL\n limit: 30\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-14",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required (free apiKey recommended for multiple widgets)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.cvedetails.com/privacy.php",children:"CVE Details Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"exchange-rates",children:"Exchange Rates"}),"\n",(0,r.jsx)(n.p,{children:"Display current FX rates in your native currency. Hover over a row to view more info, or click to show rates in that currency."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/currencies"})}),"\n",(0,r.jsx)(n.h4,{id:"options-15",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"inputCurrency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The base currency to show results in. Specified as a 3-letter ISO-4217 code, see ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/docs/supported-currencies",children:"here"})," for the full list of supported currencies, and their symbols"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"outputCurrencies"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["List or currencies to show results for. Specified as a 3-letter ISO-4217 code, see ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/docs/supported-currencies",children:"here"})," for the full list of supported currencies, and their symbols"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/",children:"exchangerate-api.com"}),", usually a 24-digit alpha-numeric string. You can sign up for a free account ",(0,r.jsx)(n.a,{href:"https://app.exchangerate-api.com/sign-up",children:"here"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-15",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: exchange-rates\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxx\n inputCurrency: GBP\n outputCurrencies:\n - USD\n - JPY\n - HKD\n - KPW\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-15",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 100,000 requests/ month)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/terms",children:"ExchangeRateAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"public-holidays",children:"Public Holidays"}),"\n",(0,r.jsxs)(n.p,{children:["Counting down to the next day off work? This widget displays upcoming public holidays for your country. Data is fetched from ",(0,r.jsx)(n.a,{href:"http://kayaposoft.com/enrico/",children:"Enrico"})]}),"\n",(0,r.jsxs)(n.p,{children:["Note, config for this widget is case-sensitive (see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1268",children:"#1268"}),")"]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/holidays"})}),"\n",(0,r.jsx)(n.h4,{id:"options-16",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"country"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The country to fetch holiday data for, specified as a country code, e.g. ",(0,r.jsx)(n.code,{children:"GB"})," or ",(0,r.jsx)(n.code,{children:"US"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"state"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["restrict a country to a specific state defined by ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/ISO_3166-2",children:"ISO_3166-2"}),", e.g. ",(0,r.jsx)(n.code,{children:"LND"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"holidayType"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The type of holidays to fetch. Can be: ",(0,r.jsx)(n.code,{children:"all"}),", ",(0,r.jsx)(n.code,{children:"public_holiday"}),", ",(0,r.jsx)(n.code,{children:"observance"}),", ",(0,r.jsx)(n.code,{children:"school_holiday"}),", ",(0,r.jsx)(n.code,{children:"other_day"})," or ",(0,r.jsx)(n.code,{children:"extra_working_day"}),". Defaults to ",(0,r.jsx)(n.code,{children:"public_holiday"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monthsToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of months in advance to show. Min: ",(0,r.jsx)(n.code,{children:"1"}),", max: ",(0,r.jsx)(n.code,{children:"24"}),". Defaults to ",(0,r.jsx)(n.code,{children:"12"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The language in which the events should be. Usually local languages and english are available. Default to first available in the country. e.g. ",(0,r.jsx)(n.code,{children:"en"})," or ",(0,r.jsx)(n.code,{children:"fr"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-16",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-holidays\n options:\n country: GB\n state: LND\n holidayType: all\n monthsToShow: 12\n lang: en\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-16",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/jurajmajer/enrico",children:"jurajmajer/enrico"}),") or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"covid-19-status",children:"Covid-19 Status"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of the current COVID-19 status. Optionally also show cases by country, and a time-series chart. Uses live data from various sources, computed by ",(0,r.jsx)(n.a,{href:"https://disease.sh/",children:"disease.sh"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/covid"})}),"\n",(0,r.jsx)(n.h4,{id:"options-17",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showChart"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Also display a time-series chart showing number of recent cases"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showCountries"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Also display a list of cases per country"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Specify number of days worth of history to render on the chart"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"countries"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string[]"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An array of countries to display, specified by their ",(0,r.jsx)(n.a,{href:"https://www.iso.org/obp/ui",children:"ISO-3 codes"}),". Leave blank to show all, sorted by most cases. ",(0,r.jsx)(n.code,{children:"showCountries"})," must be set to ",(0,r.jsx)(n.code,{children:"true"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If showing all countries, set a limit for number of results to return. Defaults to ",(0,r.jsx)(n.code,{children:"10"}),", no maximum"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-17",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: covid-stats\n"})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: covid-stats\n options:\n showChart: true\n showCountries: true\n countries:\n - GBR\n - USA\n - IND\n - RUS\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-17",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/disease-sh/api",children:"disease-sh/api"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Conditions"}),": ",(0,r.jsx)(n.a,{href:"https://github.com/disease-sh/api/blob/master/TERMS",children:"Terms of Use"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"sports-scores",children:"Sports Scores"}),"\n",(0,r.jsxs)(n.p,{children:["Show recent scores and upcoming matches from your favorite sports team. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/",children:"TheSportsDB.com"}),". From the UI, you can click any other team to view their scores and upcoming games, or click a league name to see all teams."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/sports"})}),"\n",(0,r.jsx)(n.h4,{id:"options-18",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"teamId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The ID of a team to fetch scores from. You can search for your team on the ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/teams_main.php",children:"Teams Page"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"leagueId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Alternatively, provide a league ID to fetch all games from. You can find the ID on the ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/Sport/Leagues",children:"Leagues Page"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"pastOrFuture"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"past"})," to show scores for recent games, or ",(0,r.jsx)(n.code,{children:"future"})," to show upcoming games. Defaults to ",(0,r.jsx)(n.code,{children:"past"}),". You can change this within the UI"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Optionally specify your API key, which you can sign up for at ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/",children:"TheSportsDB.com"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To limit output to a certain number of matches, defaults to ",(0,r.jsx)(n.code,{children:"15"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideImage"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"})," to not render the team / match banner image, defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-18",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: sports-scores\n options:\n teamId: 133636\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-18",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 30 requests / minute, limited endpoints)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"news-headlines",children:"News Headlines"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the latest news, click to read full article. Date is fetched from various news sources using ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/en",children:"Currents API"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/news"})}),"\n",(0,r.jsx)(n.h4,{id:"options-19",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your API key for CurrentsAPI. This is free, and you can ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/en/register",children:"get one here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"country"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Fetch news only from a certain country or region. Specified as a country code, e.g. ",(0,r.jsx)(n.code,{children:"GB"})," or ",(0,r.jsx)(n.code,{children:"US"}),". See ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/regions",children:"here"})," for a list of supported regions"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"category"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Only return news from within a given category, e.g. ",(0,r.jsx)(n.code,{children:"sports"}),", ",(0,r.jsx)(n.code,{children:"programming"}),", ",(0,r.jsx)(n.code,{children:"world"}),", ",(0,r.jsx)(n.code,{children:"science"}),". The ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/categories",children:"following categories"})," are supported"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the language for returned articles as a 2-digit ISO code (limited article support). The ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/languages",children:"following languages"})," are supported, defaults to ",(0,r.jsx)(n.code,{children:"en"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results. Can be between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"200"}),", defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"keywords"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Only return articles that contain an exact match within their title or description"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideImages"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", then article image thumbnails will not be displayed"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-19",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: news-headlines\n options:\n apiKey: xxxxxxx\n category: world\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-19",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 600 requests / day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/privacy",children:"CurrentsAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"tfl-status",children:"TFL Status"}),"\n",(0,r.jsx)(n.p,{children:"Shows real-time tube status of the London Underground. All fields are optional."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/tfl"})}),"\n",(0,r.jsx)(n.h4,{id:"options-20",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showAll"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default, details for lines with a Good Service are not visible, but you can click More Details to see all. Setting this option to ",(0,r.jsx)(n.code,{children:"true"})," will show all lines on initial page load"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortAlphabetically"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default lines are sorted by current status, set this option to ",(0,r.jsx)(n.code,{children:"true"})," to instead sort them alphabetically"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"linesToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"By default all lines are shown. If you're only interested in the status of a few lines, then pass in an array of lines to show, specified by name"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-20",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: tfl-status\n"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: tfl-status\n options:\n showAll: true\n sortAlphabetically: true\n linesToShow:\n - District\n - Jubilee\n - Central\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-20",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://tfl.gov.uk/corporate/privacy-and-cookies/",children:"TFL Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"stock-price-history",children:"Stock Price History"}),"\n",(0,r.jsx)(n.p,{children:"Shows recent price history for a given publicly-traded stock or share"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/stocks"})}),"\n",(0,r.jsx)(n.h4,{id:"options-21",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/",children:"Alpha Vantage"}),", you can get a free API key ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/support/#api-key",children:"here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"stock"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The stock symbol for the asset to fetch data for"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"priceTime"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The time to fetch price for. Can be ",(0,r.jsx)(n.code,{children:"high"}),", ",(0,r.jsx)(n.code,{children:"low"}),", ",(0,r.jsx)(n.code,{children:"open"})," or ",(0,r.jsx)(n.code,{children:"close"}),". Defaults to ",(0,r.jsx)(n.code,{children:"high"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value. Defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"})," which inherits dashboard primary color"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-21",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stock-price-chart\n options:\n stock: NET\n apiKey: PGUWSWD6CZTXMT8N\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-21",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 500 requests/day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/privacy/",children:"AlphaVantage Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"eth-gas-prices",children:"ETH Gas Prices"}),"\n",(0,r.jsxs)(n.p,{children:["Renders the current Gas cost of transactions on the Ethereum network (in both GWEI and USD), along with recent historical prices. Useful for spotting a good time to transact. Uses data from ",(0,r.jsx)(n.a,{href:"https://ethgas.watch/",children:"ethgas.watch"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/eth-gas"})}),"\n",(0,r.jsx)(n.h4,{id:"options-22",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-22",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: eth-gas-prices\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-22",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/wslyvh/ethgaswatch",children:"wslyvh/ethgaswatch"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"joke",children:"Joke"}),"\n",(0,r.jsxs)(n.p,{children:["Renders a programming or generic joke. Data is fetched from the ",(0,r.jsx)(n.a,{href:"https://github.com/Sv443/JokeAPI",children:"JokesAPI"})," by @Sv443. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/joke"})}),"\n",(0,r.jsx)(n.h4,{id:"options-23",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"category"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set the category of jokes to return. Use a string to specify a single category, or an array to pass in multiple options. Available options are: ",(0,r.jsx)(n.code,{children:"all"}),", ",(0,r.jsx)(n.code,{children:"programming"}),", ",(0,r.jsx)(n.code,{children:"pun"}),", ",(0,r.jsx)(n.code,{children:"dark"}),", ",(0,r.jsx)(n.code,{children:"spooky"}),", ",(0,r.jsx)(n.code,{children:"christmas"})," and ",(0,r.jsx)(n.code,{children:"misc"}),". An up-to-date list of supported categories can be found ",(0,r.jsx)(n.a,{href:"https://v2.jokeapi.dev/categories",children:"here"}),". Defaults to ",(0,r.jsx)(n.code,{children:"all"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"safeMode"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"}),", to prevent the fetching of any NSFW jokes. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"language"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the language for returned jokes. The following languages are supported: ",(0,r.jsx)(n.code,{children:"en"}),", ",(0,r.jsx)(n.code,{children:"cs"}),", ",(0,r.jsx)(n.code,{children:"de"}),", ",(0,r.jsx)(n.code,{children:"es"}),", ",(0,r.jsx)(n.code,{children:"fr"})," and ",(0,r.jsx)(n.code,{children:"pt"}),", and an up-to-date list of supported languages can be found ",(0,r.jsx)(n.a,{href:"https://v2.jokeapi.dev/languages",children:"here"}),". By default, your system language will be used, if it's supported, otherwise English"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-23",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: joke\n options:\n safeMode: true\n language: en\n category: Programming\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-23",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/Sv443/JokeAPI",children:"Sv443/JokeAPI"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://sv443.net/privacypolicy/en",children:"SV443's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"chuck-norris-quotes",children:"Chuck Norris quotes"}),"\n",(0,r.jsxs)(n.p,{children:["Renders a Chuck Norris quote. Data is fetched from the ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/",children:"ChuckNorrisAPI"})," by @matchilling. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://tbd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-24",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"categories"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set the category of jokes to return. Use a string to specify a single category, or an array to pass in multiple options. Available options are: ",(0,r.jsx)(n.code,{children:"animal"}),",",(0,r.jsx)(n.code,{children:"career"}),",",(0,r.jsx)(n.code,{children:"celebrity"}),",",(0,r.jsx)(n.code,{children:"dev"}),",",(0,r.jsx)(n.code,{children:"explicit"}),",",(0,r.jsx)(n.code,{children:"fashion"}),",",(0,r.jsx)(n.code,{children:"food"}),",",(0,r.jsx)(n.code,{children:"history"}),",",(0,r.jsx)(n.code,{children:"money"}),",",(0,r.jsx)(n.code,{children:"movie"}),",",(0,r.jsx)(n.code,{children:"music"}),",",(0,r.jsx)(n.code,{children:"political"}),",",(0,r.jsx)(n.code,{children:"religion"}),",",(0,r.jsx)(n.code,{children:"science"}),",",(0,r.jsx)(n.code,{children:"sport"})," and ",(0,r.jsx)(n.code,{children:"travel"}),". An up-to-date list of supported categories can be found ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/jokes/categories",children:"here"}),". Defaults to not explicitely set and therefore any of the categories can come up."]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-24",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: chucknorris\n options:\n categories: history,sport\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-24",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/privacy",children:"matchilling's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"xkcd-comics",children:"XKCD Comics"}),"\n",(0,r.jsxs)(n.p,{children:["Have a laugh with the daily comic from ",(0,r.jsx)(n.a,{href:"https://xkcd.com/",children:"XKCD"}),". A classic webcomic website covering everything from Linux, math, romance, science and language. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/xkcd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-25",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"comic"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string / number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Choose which comic to display. Set to either ",(0,r.jsx)(n.code,{children:"random"}),", ",(0,r.jsx)(n.code,{children:"latest"})," or the series number of a specific comic, like ",(0,r.jsx)(n.code,{children:"627"}),". Defaults to ",(0,r.jsx)(n.code,{children:"latest"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-25",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: xkcd-comic\n options:\n comic: latest\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-25",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"flight-data",children:"Flight Data"}),"\n",(0,r.jsxs)(n.p,{children:["Displays airport departure and arrival flights, using data from ",(0,r.jsx)(n.a,{href:"https://www.aerodatabox.com/",children:"AeroDataBox"}),". Useful if you live near an airport and often wonder where the flight overhead is going to. Hover over a row for more flight data."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/flights"})}),"\n",(0,r.jsx)(n.h4,{id:"options-26",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"airport"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The airport to show flight data from. Should be specified as a 4-character ICAO-code, a full list of which can be found ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/ICAO_airport_code",children:"here"})," (example: ",(0,r.jsx)(n.code,{children:"KBJC"})," or ",(0,r.jsx)(n.code,{children:"EGKK"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A valid ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/",children:"RapidAPI"})," Key, with ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/aerodatabox/api/aerodatabox/",children:"AeroDataBox"})," enabled (check in your ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/developer/billing/subscriptions-and-usage",children:"Subscription Dashboard"}),"). This API is free to sign up for and use"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"For busy airports, you may wish to limit the number of results visible"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"direction"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default, both departure and arrival flights will be fetched, if you would like to only show flights in one direction, set this to wither ",(0,r.jsx)(n.code,{children:"departure"})," or ",(0,r.jsx)(n.code,{children:"arrival"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-26",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: flight-data\n options:\n airport: EGLC\n apiKey: XXXXX\n limit: 12\n direction: all\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-26",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 150 requests / month)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.aerodatabox.com/#h.p_cxtiyzwf_wqd",children:"AeroDataBox"})," and ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/privacy/",children:"RapidAPI Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"astronomy-picture-of-the-day",children:"Astronomy Picture of the Day"}),"\n",(0,r.jsxs)(n.p,{children:["Show the NASA Astronomy Picture of the Day. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://apod.nasa.gov/apod/",children:"APOD"})," using ",(0,r.jsx)(n.a,{href:"https://github.com/lissy93/go-apod",children:"@Lissy93/go-apod"})," / hosted at ",(0,r.jsx)(n.a,{href:"https://apod.as93.net/",children:"apod.as93.net"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://storage.googleapis.com/as93-screenshots/dashy/apod.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-27",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-27",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: apod\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-27",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/lissy93/go-apod",children:"@Lissy93/go-apod"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.nasa.gov/about/highlights/HP_Privacy.html",children:"NASA's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"github-trending",children:"GitHub Trending"}),"\n",(0,r.jsxs)(n.p,{children:["Displays currently trending projects on GitHub. Optionally specify a language and time-frame. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/gh-trending-no-cors",children:"Lissy93/gh-trending-no-cors"})," using the GitHub API. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/github-trending"})}),"\n",(0,r.jsx)(n.h4,{id:"options-28",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["A programming language to fetch trending repos from that category. E.g. ",(0,r.jsx)(n.code,{children:"javascript"})," or ",(0,r.jsx)(n.code,{children:"go"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"since"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The timeframe to use when calculating trends. Can be either ",(0,r.jsx)(n.code,{children:"daily"}),", ",(0,r.jsx)(n.code,{children:"weekly"})," or ",(0,r.jsx)(n.code,{children:"monthly"}),". Defaults to ",(0,r.jsx)(n.code,{children:"daily"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Optionally limit the number of results. Max ",(0,r.jsx)(n.code,{children:"25"}),", default is ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-28",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: github-trending-repos\n options:\n limit: 8\n since: weekly\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-28",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/gh-trending-no-cors",children:"Lissy93/gh-trending-no-cors"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"github-profile-stats",children:"GitHub Profile Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Display stats from your GitHub profile, using embedded cards from ",(0,r.jsx)(n.a,{href:"https://github.com/anuraghazra/github-readme-stats",children:"anuraghazra/github-readme-stats"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/github-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-29",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The GitHub username to fetch info for. E.g. ",(0,r.jsx)(n.code,{children:"lissy93"}),". (Not required if ",(0,r.jsx)(n.code,{children:"hideProfileCard"})," and ",(0,r.jsx)(n.code,{children:"hideLanguagesCard"})," are both set to ",(0,r.jsx)(n.code,{children:"true"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideProfileCard"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the users profile card will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLanguagesCard"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the users top languages card will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"repos"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you'd like to also display stats for some GitHub repositories, then add an array or repo names here. Specified as ",(0,r.jsx)(n.code,{children:"[username]/[repo-name]"}),", e.g. ",(0,r.jsx)(n.code,{children:"lissy93/dashy"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-29",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: github-profile-stats\n options:\n username: Lissy93\n hideLanguagesCard: true\n repos:\n - lissy93/dashy\n - lissy93/personal-security-checklist\n - lissy93/twitter-sentiment-visualisation\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-29",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/anuraghazra/github-readme-stats",children:"anuraghazra/github-readme-stats"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"healthchecks-status",children:"HealthChecks Status"}),"\n",(0,r.jsx)(n.p,{children:"Display status of one or more HealthChecks project(s). Works with healthchecks.io and your selfhosted instance."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/healthchecks"})}),"\n",(0,r.jsx)(n.h4,{id:"options-30",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["The base url of your instance, default is ",(0,r.jsx)(n.code,{children:"https://healthchecks.io"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," or ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"One or more API keys for your healthcheck projects. (Read-only works fine)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: HealthChecks\n options:\n host: https://healthcheck.your-domain.de\n apiKey: \n - abcdefg...\n - zxywvu...\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-30",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Paid / Self-hosted"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"healthchecks/healthchecks"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://healthchecks.io/privacy/",children:"Healthchecks.io Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"hackernews-trending",children:"Hackernews Trending"}),"\n",(0,r.jsx)(n.p,{children:"Display new and trending Posts from Hackernews"}),"\n",(0,r.jsx)(n.h4,{id:"options-31",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"stories"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["HN Stories to display defaults to ",(0,r.jsx)(n.code,{children:"topstories"}),". Options are: ",(0,r.jsx)(n.code,{children:"beststories"}),", ",(0,r.jsx)(n.code,{children:"topstories"})," or ",(0,r.jsx)(n.code,{children:"newstories"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"int"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The size of the list of Posts to show."})]})]})]}),"\n",(0,r.jsx)(n.h5,{id:"example-30",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: hackernews-trending\n options:\n stories: newstories\n limit: 10\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mvg-departure",children:"MVG Departure"}),"\n",(0,r.jsx)(n.p,{children:"Display departure time of a MVG (M\xfcnchner Verkehrs Gesellschaft) station."}),"\n",(0,r.jsxs)(n.p,{children:["From ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/impressum.html",children:"https://www.mvg.de/impressum.html"}),":"]}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:"[...] Die Verarbeitung unserer Inhalte oder Daten durch Dritte erfordert unsere ausdr\xfcckliche Zustimmung. F\xfcr private, nicht-kommerzielle Zwecke, wird eine gem\xe4\xdfigte Nutzung ohne unsere ausdr\xfcckliche Zustimmung geduldet. Jegliche Form von Data-Mining stellt keine gem\xe4\xdfigte Nutzung dar.[...]"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"In other words: Private, noncomercial, moderate use of the API is tolerated. They don\u2019t consider data mining as moderate use. (This is not a legal advice)"}),"\n",(0,r.jsx)(n.h4,{id:"options-32",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"location"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The name of the location (exact) or the location id, startin with ",(0,r.jsx)(n.code,{children:"de:09162:"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"integer"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Limit number of entries, defaults to 10."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A custom title to be displayed."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"header"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"bool"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Shall the title be shown?"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.line"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for given line(s)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.product"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific product (TRAM, UBAHN, SBAHN, BUS)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific destination(s)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mvg\n options:\n location: Marienplatz\n limit: 5\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-31",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Private use only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mvg.de",children:"MVG"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mvg-connection",children:"MVG Connection"}),"\n",(0,r.jsx)(n.p,{children:"Display the next connection for two addresses/coordinates, stations or POI within Munich using MVG MVG (M\xfcnchner Verkehrs Gesellschaft)."}),"\n",(0,r.jsxs)(n.p,{children:["From ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/impressum.html",children:"https://www.mvg.de/impressum.html"}),":"]}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:"[...] Die Verarbeitung unserer Inhalte oder Daten durch Dritte erfordert unsere ausdr\xfcckliche Zustimmung. F\xfcr private, nicht-kommerzielle Zwecke, wird eine gem\xe4\xdfigte Nutzung ohne unsere ausdr\xfcckliche Zustimmung geduldet. Jegliche Form von Data-Mining stellt keine gem\xe4\xdfigte Nutzung dar.[...]"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"In other words: Private, noncomercial, moderate use of the API is tolerated. They don\u2019t consider data mining as moderate use. (This is not a legal advice)"}),"\n",(0,r.jsx)(n.h4,{id:"options-33",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"origin"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Origin of the connection."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Destination of the connection."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A custom title to be displayed."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"header"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"bool"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Shall the title be shown?"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.line"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for given line(s)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.product"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific product (TRAM, UBAHN, SBAHN, BUS)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific destination(s)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mvg-connection\n options:\n from: Marienplatz\n from: Dachauer Stra\xdfe 123\n header: true\n filters:\n product: [UBAHN]\n line: [U1,U2,U4,U5]\n\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-32",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Private use only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mvg.de",children:"MVG"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"custom-list",children:"Custom List"}),"\n",(0,r.jsx)(n.p,{children:"Renders custom schema-compliant JOSN in a list."}),"\n",(0,r.jsx)(n.h4,{id:"options-34",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"text"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A string containing the url of a json file."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"text"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A title for the widget. Can be helpful if stacking multiple lists in the same section."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"daysForNew"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsx)(n.td,{children:"Used to highlight new items."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"json-schema",children:"Json Schema"}),"\n",(0,r.jsx)(n.p,{children:"The input file should comply with the following schema:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-json",children:'{\n "$schema": "http://json-schema.org/draft-04/schema#",\n "type": "array",\n "items": [\n {\n "type": "object",\n "properties": {\n "link": {\n "type": "object",\n "properties": {\n "text": {\n "type": "string"\n },\n "url": {\n "type": "string"\n },\n "title": {\n "type": "string"\n }\n },\n "required": [\n "text",\n "url",\n "title"\n ]\n },\n "value": {\n "type": "object",\n "properties": {\n "text": {\n "type": "string"\n },\n "title": {\n "type": "string"\n }\n },\n "required": [\n "text",\n "title"\n ]\n },\n "date": {\n "type": "string"\n }\n },\n "required": [\n "link",\n "value",\n "date"\n ]\n }\n ]\n}\n'})}),"\n",(0,r.jsx)(n.p,{children:"Example: This json data was generated by a data worflow that gets the new releases of a few projects from GitHub. The system used to build the data workflow is n8n."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-json",children:'[\n {\n "link": {\n "text": "jellyfin/jellyfin",\n "url": "https://github.com/jellyfin/jellyfin/releases/tag/v10.10.7",\n "title": ""\n },\n "value": {\n "text": "v10.10.7",\n "title": "2025-04-05"\n },\n "date": "2025-04-05T19:14:59Z"\n },\n {\n "link": {\n "text": "jellyfin/jellyfin-web",\n "url": "https://github.com/jellyfin/jellyfin-web/releases/tag/v10.10.7",\n "title": ""\n },\n "value": {\n "text": "v10.10.7",\n "title": "2025-04-05"\n },\n "date": "2025-04-05T19:15:00Z"\n },\n {\n "link": {\n "text": "lissy93/dashy",\n "url": "https://github.com/Lissy93/dashy/releases/tag/3.1.1",\n "title": ""\n },\n "value": {\n "text": "3.1.1",\n "title": "2024-05-30"\n },\n "date": "2024-05-30T17:20:53Z"\n },\n {\n "link": {\n "text": "VSCodium/vscodium",\n "url": "https://github.com/VSCodium/vscodium/releases/tag/1.102.14746",\n "title": ""\n },\n "value": {\n "text": "1.102.14746",\n "title": "2025-07-16"\n },\n "date": "2025-07-16T18:27:49Z"\n }\n]\n'})}),"\n",(0,r.jsx)(n.h4,{id:"notes",children:"Notes"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"This widget is designed to render data generated by another system that complies with the schema. The example JSON data above was generated using a n8n workflow, and other ETL or workflow systems can generate similar results."}),"\n",(0,r.jsx)(n.li,{children:"To avoid requests to a different system in each refresh, you can save the input files locally in the user-data folder inside your Dashy installation."}),"\n",(0,r.jsxs)(n.li,{children:["To use json files from a different domain, remember to add ",(0,r.jsx)(n.code,{children:"useProxy: true"})," to the widget configuration. I have not tested this use case because I save all my input data locally on the Dashy server. Please open a ticket if you have an issue in this use case."]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-31",children:"Example"}),"\n",(0,r.jsxs)(n.p,{children:["This widget renders a json file that from a ",(0,r.jsx)(n.code,{children:"json-data"})," directory inside the ",(0,r.jsx)(n.code,{children:"user-data"})," directory on the Dashy server."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: custom-list\n options:\n url: /json-data/github-releases.json\n title: 'Github Releases'\n daysForNew: 5\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-33",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Not needed for files hosted inside the ",(0,r.jsx)(n.code,{children:"user-data"})," directory. Use ",(0,r.jsx)(n.code,{children:"useProxy: true"})," to bypass CORS restrictions when using data from a different server."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": user defined"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": depends on the user defined host."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"custom-search",children:"Custom search"}),"\n",(0,r.jsx)(n.p,{children:"Allows web search using multiple user-defined search engines and other websites."}),"\n",(0,r.jsx)(n.h4,{id:"options-35",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"engines"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"required"}),(0,r.jsxs)(n.td,{children:["An array of search engine objects. Each search engine object should have two required properties: ",(0,r.jsx)(n.strong,{children:"title"})," and ",(0,r.jsx)(n.strong,{children:"url"}),". See the example below."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"placeholder"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"Placeholder text in the search box."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"openingMethod"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["Open search in one of ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"notes-1",children:"Notes"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["The first search engine in the engines array will be treated as the default search engine, and used when the user presses ",(0,r.jsx)(n.code,{children:"Enter"})," in the search box."]}),"\n",(0,r.jsx)(n.li,{children:"Popup blockers can interfere with opening a new search window."}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-32",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"This widget allows searching multiple search engines from dashy."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: 'custom-search'\n options:\n placeholder: Search for something using the buttons below\n engines:\n - title: SearXNG\n url: https://searx.lan/?q=\n - title: Quant\n url: https://www.qwant.com/?q=\n - title: Bing Web\n url: http://www.bing.com/search?q=\n - title: Bing Images\n url: http://www.bing.com/images/search?q=\n - title: Bing Maps\n url: http://www.bing.com/maps/search?q=\n - title: Yandex\n url: https://www.yandex.com/search/?text=\n - title: Passmark\n url: https://www.passmark.com/search/zoomsearch.php?zoom_query=\n - title: IMDB\n url: http://www.imdb.com/find?q=\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-34",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Not needed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": user defined"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": depends on the user defined search engines."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"rescuetime-overview",children:"RescueTime Overview"}),"\n",(0,r.jsx)(n.p,{children:"Show an overview of how you have spent your time for the current day."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/rescue-time"})}),"\n",(0,r.jsx)(n.h4,{id:"options-36",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"required"}),(0,r.jsx)(n.td,{children:"The API-Key generated in the RescueTime UI."})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-33",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: rescue-time\n useProxy: true\n options:\n apiKey: abcdefghijkl_mnop\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-35",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Depends on user subscription"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://www.rescuetime.com",children:"RescueTime"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.rescuetime.com/privacy",children:"RescueTime Privacy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"minecraft-server",children:"Minecraft Server"}),"\n",(0,r.jsx)(n.p,{children:"Show minecraft server status"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://i.ibb.co/hcmd4Wf/minecraft-widget.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-37",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display title for server uses server address if not set."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"server"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Server hostname or ip(:port) will use srv records."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"bedrock"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If server is a bedrock edition server. (default false)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showMods"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display mod list if available"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showPlayers"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display player list if available"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showPlugins"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display plugin list if available"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-34",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: minecraft-status\n options:\n title: Venity Network\n server: play.venitymc.com\n bedrock: true\n showMods: true\n showPlayers: true\n showPlugins: true\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-36",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mcsrvstat.us/",children:"Minecraft Server Status"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://mcsrvstat.us/faq",children:"Minecraft Server Status FAQ"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"self-hosted-services-widgets",children:"Self-Hosted Services Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"system-info",children:"System Info"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]}),"\nDisplays info about the server which Dashy is hosted on. Includes user + host, operating system, uptime and basic memory & load data."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/system-info"})}),"\n",(0,r.jsx)(n.h4,{id:"options-38",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-35",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: system-info\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-37",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:"No external data requests made"}),"\n",(0,r.jsxs)(n.p,{children:["Note that this widget is not available if you are running Dashy in a container or VM. Instead you can use the ",(0,r.jsx)(n.a,{href:"#system-resource-monitoring",children:"System Monitoring"})," widgets to display stats from the host system instead."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cron-monitoring-health-checks",children:"Cron Monitoring (Health Checks)"}),"\n",(0,r.jsxs)(n.p,{children:["Cron job monitoring using ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"Health Checks"}),". Both managed and self-hosted instances are supported."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cron-monitor"})}),"\n",(0,r.jsx)(n.h4,{id:"options-39",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A read-only API key for the project to monitor. You can generate this by selecting a Project --\x3e Settings --\x3e API Access. Note that you must generate a separate key for each project"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you're self-hosting, or using any instance other than the official (healthchecks.io), you will need to specify the host address. E.g. ",(0,r.jsx)(n.code,{children:"https://healthchecks.example.com"})," or ",(0,r.jsx)(n.code,{children:"http://cron-monitoing.local"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-36",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: health-checks\n options:\n apiKey: XXXXXXXXX\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-38",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 20 services, or self-host for unlimited)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"GitHub - HealthChecks"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://healthchecks.io/privacy/",children:"Health-Checks Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-history-netdata",children:"CPU History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent CPU usage history from NetData."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/cpu-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-40",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-37",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-cpu-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-39",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"memory-history-netdata",children:"Memory History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent system RAM usage from NetData, and show as a breakdown of different categories."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://i.ibb.co/2dsSWnk/nd-memory-history.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-41",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-38",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-ram-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-40",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"load-history-netdata",children:"Load History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent load usage in 1, 5 and 15 minute intervals, from NetData."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/load-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-42",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-39",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-load-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-41",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-stats",children:"Pi-Hole Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the number of queries blocked by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/pi-hole"})}),"\n",(0,r.jsx)(n.h4,{id:"options-43",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStatus"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideChart"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideInfo"})})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide any of the three parts of the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-40",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-stats\n options:\n hostname: http://192.168.130.1\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["To avoid storing secrets in plaintext, both ",(0,r.jsx)(n.code,{children:"hostname"})," and ",(0,r.jsx)(n.code,{children:"apiKey"})," can be supplied via environment variables. See ",(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"}),"."]})}),"\n",(0,r.jsx)(n.h4,{id:"info-42",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-Hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-stats-v6",children:"Pi-Hole Stats v6"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the number of queries blocked by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/zftCLJN/pi-hole-stats.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-44",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStatus"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideChart"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideInfo"})})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide any of the three parts of the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-41",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-stats-v6\n options:\n hostname: http://192.168.130.1\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["To avoid storing secrets in plaintext, both ",(0,r.jsx)(n.code,{children:"hostname"})," and ",(0,r.jsx)(n.code,{children:"apiKey"})," can be supplied via environment variables. See ",(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"}),"."]})}),"\n",(0,r.jsx)(n.h4,{id:"info-43",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-Hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-queries",children:"Pi-Hole Queries"}),"\n",(0,r.jsxs)(n.p,{children:["Shows top queries that were blocked and allowed by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/pi-hole-queries"})}),"\n",(0,r.jsx)(n.h4,{id:"options-45",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of queries to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-42",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-top-queries\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-44",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-queries-v6",children:"Pi-Hole Queries v6"}),"\n",(0,r.jsxs)(n.p,{children:["Shows top queries that were blocked and allowed by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/pXR0bdQ/pi-hole-queries.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-46",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of queries to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-43",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-top-queries-v6\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-45",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-recent-traffic",children:"Pi-Hole Recent Traffic"}),"\n",(0,r.jsxs)(n.p,{children:["Shows number of recent traffic, using allowed and blocked queries from ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/pi-hole-trafic"})}),"\n",(0,r.jsx)(n.h4,{id:"options-47",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-44",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-traffic\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-46",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-recent-traffic-v6",children:"Pi-Hole Recent Traffic v6"}),"\n",(0,r.jsxs)(n.p,{children:["Shows number of recent traffic, using allowed and blocked queries from ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://i.ibb.co/7kdxxwx/pi-hole-recent-queries.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-48",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-45",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-traffic-v6\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-47",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"stat-ping-statuses",children:"Stat Ping Statuses"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the current and recent uptime of your running services, via a self-hosted instance of ",(0,r.jsx)(n.a,{href:"https://github.com/statping/statping",children:"StatPing"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"300",src:"https://pixelflare.cc/alicia/dashy/statping"})}),"\n",(0,r.jsx)(n.h4,{id:"options-49",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your StatPing instance, without a trailing slash"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"groupId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided, only Services in the given group are displayed. Defaults to ",(0,r.jsx)(n.code,{children:"0"})," in which case all services are displayed."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showChart"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided and ",(0,r.jsx)(n.code,{children:"false"})," then charts are not displayed. Defaults to ",(0,r.jsx)(n.code,{children:"true"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided and ",(0,r.jsx)(n.code,{children:"false"})," then information summaries are not displayed. Defaults to ",(0,r.jsx)(n.code,{children:"true"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-46",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stat-ping\n options:\n hostname: http://192.168.130.1:8080\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stat-ping\n options:\n hostname: http://192.168.130.1:8080\n groupId: 3\n showChart: false\n showInfo: false\n"})}),"\n",(0,r.jsxs)(n.p,{children:["You can use multiple StatPing widgets with different ",(0,r.jsx)(n.code,{children:"groupId"}),"s."]}),"\n",(0,r.jsx)(n.p,{children:"Note, the Group Id is not directly visible in StatPing UI, you can inspect the group select HTML element or the API response to find out."}),"\n",(0,r.jsx)(n.h4,{id:"info-48",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/statping/statping",children:"GitHub - StatPing"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.statping.com/",children:"StatPing Docs"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"synology-download-station",children:"Synology Download Station"}),"\n",(0,r.jsx)(n.p,{children:"Displays the current downloads/torrents tasks of your Synology NAS"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/synology-downloads"})}),"\n",(0,r.jsx)(n.h4,{id:"options-50",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Synology NAS, without a trailing slash"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The username of a user on your synology NAS. You will see only this user download station tasks if he is not part of the administrator group. Currently don't support OTP protected accounts."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The password of the account specified above."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-47",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: synology-download\n options:\n hostname: http://192.168.1.1:8080\n username: dashy\n password: totally-secure-password\n\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-49",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://www.synology.com/en-us",children:"Synology"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.synology.com/en-us/company/legal/privacy",children:"Synology Privacy Statement"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-block-stats",children:"AdGuard Home Block Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and\ndisplays total number of allowed and blocked queries, plus a pie chart showing breakdown by block type."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-51",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-48",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-stats\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-50",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-filters",children:"AdGuard Home Filters"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, to display the current status of each of your filter lists. Includes filter name, last updated, number of items, and a link to the list."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-filters"})}),"\n",(0,r.jsx)(n.h4,{id:"options-52",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showOnOffStatusOnly"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", will only show aggregated AdGuard filter status (on/off), instead of a list of filters"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-49",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-filter-status\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n showOnOffStatusOnly: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-51",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-dns-info",children:"AdGuard Home DNS Info"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and displays the current status (Enabled / Disabled) of AdGuard DNS. Click show more to view detailed info, including upstream DNS provider, active ports, and the status of DNSSEC, EDNS CS, PTR and IPv6."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-dns"})}),"\n",(0,r.jsx)(n.h4,{id:"options-53",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showFullInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),', the full DNS info will be shown by default, without having to click "Show Info"']})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-50",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-dns-info\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n showFullInfo: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-52",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-top-domains",children:"AdGuard Home Top Domains"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and displays a list of the most queried, and most blocked domains."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/adguard-domains"})}),"\n",(0,r.jsx)(n.h4,{id:"options-54",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the number of results to show, between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"100"}),", defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideBlockedDomains"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show the blocked domains list (queried domains only)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideQueriedDomains"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show the queried domains list (blocked domains only)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-51",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-top-domains\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n limit: 10\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-53",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-user",children:"Nextcloud User"}),"\n",(0,r.jsxs)(n.p,{children:["Nextcloud is a ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/install/#instructions-server",children:"self hosted"})," productivity platform, it can also be used free of charge with ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/sign-up/",children:"hundreds of existing hosting providers"})," that offer a free Nextcloud account."]}),"\n",(0,r.jsx)(n.p,{children:"Displays branding information of a Nextcloud server (logo, url, slogan) and some user details (name, login name, last login, disk space or quota). Use with regular or admin user."}),"\n",(0,r.jsx)(n.p,{children:"Shows quota usage when quota is enabled for the user or disk usage when not enabled."}),"\n",(0,r.jsx)(n.p,{children:"Known issues: the User API incorrectly reports available disk space as total for admin users when quota is not enabled (which usually is the case for admins)."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-user",alt:"nextcloud-user"})}),"\n",(0,r.jsx)(n.h4,{id:"options-55",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-52",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-user\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-54",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-user-statuses",children:"Nextcloud User Statuses"}),"\n",(0,r.jsx)(n.p,{children:"Show user statuses for selected users."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-user-status",alt:"nextcloud-userstatus"})}),"\n",(0,r.jsx)(n.h4,{id:"options-56",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"users"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Nextcloud User IDs to show statuses for, list size between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"100"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showEmpty"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Show statuses without a message, defaults to ",(0,r.jsx)(n.code,{children:"true"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-53",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-userstatus\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n users: ['bob', 'alice']\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-55",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-notifications",children:"Nextcloud Notifications"}),"\n",(0,r.jsx)(n.p,{children:"Displays your notifications and allows deleting them."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/next-cloud-notifications",alt:"nextcloud-notifications"})}),"\n",(0,r.jsx)(n.h4,{id:"options-57",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number|string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit displayed notifications either by count, e.g. ",(0,r.jsx)(n.code,{children:"5"})," to show the 5 most recent, or by age, e.g. ",(0,r.jsx)(n.code,{children:"1d"})," to only show notifications not older than a day. Accepted suffixes for age limit are ",(0,r.jsx)(n.code,{children:"m"}),", ",(0,r.jsx)(n.code,{children:"h"})," and ",(0,r.jsx)(n.code,{children:"d"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-54",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-notifications\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n limit: 6h\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-56",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-system",children:"Nextcloud System"}),"\n",(0,r.jsx)(n.p,{children:"Visualises overall memory utilisation and CPU load averages, shows server versions."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/next-cloud-sysyem",alt:"nextcloud-system"})}),"\n",(0,r.jsx)(n.h4,{id:"options-58",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-55",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-system\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-57",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-stats",children:"Nextcloud Stats"}),"\n",(0,r.jsx)(n.p,{children:"Shows key usage statistics about your Nextcloud server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-stats",alt:"nextcloud-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-59",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-56",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-stats\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-58",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-php-opcache-stats",children:"Nextcloud PHP OPcache Stats"}),"\n",(0,r.jsx)(n.p,{children:"Shows statistics about PHP OPcache performance on your Nextcloud server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-php",alt:"nextcloud-phpopcache"})}),"\n",(0,r.jsx)(n.h4,{id:"options-60",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-57",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-php-opcache\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-59",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ntfy-stream",children:"ntfy stream"}),"\n",(0,r.jsxs)(n.p,{children:["Subscribes to topics on a ",(0,r.jsx)(n.strong,{children:"private"})," ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/",children:"ntfy"})," server, and shows ",(0,r.jsx)(n.strong,{children:"new messages"})," as they arrive."]}),"\n",(0,r.jsx)(n.h4,{id:"options-61",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"server_url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The server url"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"topic"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A topic, or a comma separated list of topics"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"auth"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["An ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/subscribe/api/#authentication",children:"auth string"}),". \u26a0\ufe0f This is sent as query parameter."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A title for the widget."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-58",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: ntfy-stream\n useProxy: false\n options:\n title: NTFY stream\n server_url: https://myntfy.server.tld\n topic: alert,warning,mytopic\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-60",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": Disabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/install/",children:"Self-hosted"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/privacy/",children:"ntfy Privacy Policy"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"proxmox-lists",children:"Proxmox lists"}),"\n",(0,r.jsx)(n.p,{children:"Shows lists of nodes, containers, and VMs in a Proxmox virtual environment cluster, with a status indicator."}),"\n",(0,r.jsx)(n.h4,{id:"options-62",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cluster_url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The URL of the proxmox cluster server. No trailing ",(0,r.jsx)(n.code,{children:"/"}),". for example: ",(0,r.jsx)(n.code,{children:"https://proxmox.lan:8006"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"user_name"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A Proxmox API Username, for example ",(0,r.jsx)(n.code,{children:"root@pam"})," or ",(0,r.jsx)(n.code,{children:"dashy@pve"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token_name"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A Proxmox API token name. You can get a token in the API section of the cluster management interface."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token_uuid"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The value of the token entered above. This is normally a UUID."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"node"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A Proxmox node name. If empty or not supplied, a list of nodes will be shown."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"node_data"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["This is required if a node is selected, Currently this accepts two values, either ",(0,r.jsx)(n.code,{children:"lxc"})," or ",(0,r.jsx)(n.code,{children:"qemu"})," but the widget can be improved to get other types of data from the Proxmox API."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A widget title."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title_as_link"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["When this is set to anything other than 0 or false, the title will be linked to the value entered in the ",(0,r.jsx)(n.code,{children:"cluster_url"})," option."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"footer"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A widget footer."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"footer_as_link"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["When this is set to anything other than 0 or false, the title will be linked to the value entered in the ",(0,r.jsx)(n.code,{children:"cluster_url"})," option."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hide_templates"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"When this is set to anything other than 0 or false, templates will be filtered out of the result list."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-59",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"This will show the list of nodes."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: proxmox-lists\n useProxy: true \n options:\n cluster_url: https://proxmox.lan:8006\n user_name: root@pam\n token_name: dashy\n token_uuid: bfb152df-abcd-abcd-abcd-ccb95a472d01\n"})}),"\n",(0,r.jsx)(n.p,{children:"This will show the list of VMs, with a title and a linked fotter, hiding VM templates."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: proxmox-lists\n useProxy: true \n options:\n cluster_url: https://proxmox.lan:8006\n user_name: root@pam\n token_name: dashy\n token_uuid: bfb152df-abcd-abcd-abcd-ccb95a472d01\n node: proxmox\n node_data: qemu\n title: Proxmox VMs\n title_as_link: false\n footer: Proxmox\n footer_as_link: true\n hide_templates: 1\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-61",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://proxmox.com/en/proxmox-ve",children:"Proxmox Virtual Environment"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://proxmox.com/en/privacy-policy",children:"Proxmox's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"troubleshooting",children:"Troubleshooting"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"404 Error in development mode"}),": The error might disappear in production mode ",(0,r.jsx)(n.code,{children:"yarn start"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"500 Error in production mode"}),": Try adding the certificate authority (CA) certificate of your Proxmox host to Node.js.\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Download the Proxmox CA certificate to your Dashy host."}),"\n",(0,r.jsxs)(n.li,{children:["Export environment variable ",(0,r.jsx)(n.code,{children:"NODE_EXTRA_CA_CERTS"})," and set its value to the path of the downloaded CA certificate. Example: ",(0,r.jsx)(n.code,{children:"export NODE_EXTRA_CA_CERTS=/usr/local/share/ca-certificates/devlab_ca.pem"})]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"sabnzbd",children:"Sabnzbd"}),"\n",(0,r.jsx)(n.p,{children:"Shows queue information regarding your self hosted Sabnzbd server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/sabnzbd",alt:"Sabnzbd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-63",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sabnzbdUrl"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The URL of the Sabnzbd server. No trailing ",(0,r.jsx)(n.code,{children:"/"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for Sabnzbd access. Located under ",(0,r.jsx)(n.code,{children:"Config"})," -> ",(0,r.jsx)(n.code,{children:"General"})," -> ",(0,r.jsx)(n.code,{children:"Security"})," -> ",(0,r.jsx)(n.code,{children:"API Key"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Hides extra server queue details."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideQueue"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Hides the queue list in an expandable dropdown."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-60",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: sabnzbd\n options:\n sabnzbdUrl: 'https://sabnzbd.example.com'\n apiKey: XXXXXXXXXXXXXXXXXX\n hideDetails: false\n hideQueue: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-62",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://sabnzbd.org/",children:"Sabnzbd"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://forums.sabnzbd.org/ucp.php?mode=privacy",children:"Sabnzbd Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"gluetun-vpn-info",children:"Gluetun VPN Info"}),"\n",(0,r.jsx)(n.p,{children:"Display info from the Gluetun VPN container public IP API. This can show the IP and location data for the exit VPN node."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/guletn-vpn"})}),"\n",(0,r.jsx)(n.h4,{id:"options-64",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"visibleFields"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A comma separated list of the fields you want visible in the widget. You can have any number of the following : ",(0,r.jsx)(n.code,{children:"public_ip"}),", ",(0,r.jsx)(n.code,{children:"region"}),", ",(0,r.jsx)(n.code,{children:"country"}),", ",(0,r.jsx)(n.code,{children:"city"}),", ",(0,r.jsx)(n.code,{children:"location"}),", ",(0,r.jsx)(n.code,{children:"organisation"}),", ",(0,r.jsx)(n.code,{children:"postal_code"}),", ",(0,r.jsx)(n.code,{children:"timezone"}),". Defaults to just ",(0,r.jsx)(n.code,{children:"public_ip"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The url to the gluetun HTTP control server. E.g. ",(0,r.jsx)(n.code,{children:"http://gluetun:8000"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-61",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gluetun-status\n useProxy: true\n options:\n hostname: http://server-or-conatiner-hostname:8000\n visibleFields: public_ip,region,country,city,location,organisation,postal_code,timezone\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-63",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/gluetun",children:"Gluetun"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/gluetun/wiki",children:"Gluetun Wiki"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"drone-ci-builds",children:"Drone CI Builds"}),"\n",(0,r.jsxs)(n.p,{children:["Display the last builds from a ",(0,r.jsx)(n.a,{href:"https://www.drone.ci",children:"Drone CI"})," instance. A self-hosted CI system that uses docker."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/drone-ci"})}),"\n",(0,r.jsx)(n.h4,{id:"options-65",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The hostname of the Drone CI instance."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The API key (https://[your-drone-instance]/account)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"integer"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Limit the amounts of listed builds."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"repo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Show only builds of the specified repo"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-62",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: drone-ci\n updateInterval: 30\n options:\n host: https://drone.somedomain.com\n apiKey: my-very-secret-api-key\n limit: 10\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-64",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://www.drone.io",children:"Drone"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.drone.io",children:"Drone"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"filebrowser",children:"Filebrowser"}),"\n",(0,r.jsxs)(n.p,{children:["Displays storage statistics and file listings from a ",(0,r.jsx)(n.a,{href:"https://github.com/gtsteffaniak/filebrowser",children:"Filebrowser Quantum"})," instance. Shows directory size, file/folder counts, favorite files, and recently modified files with quick-access links."]}),"\n",(0,r.jsx)(n.h4,{id:"options-66",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of your Filebrowser instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A long-lived API key (create in Settings \u2192 API Keys)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"source"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The source/scope name to browse. Defaults to the first available source"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"path"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The directory path to display. Defaults to ",(0,r.jsx)(n.code,{children:"/"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"favorites"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"List of filenames to show as quick-access favorites"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showRecent"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Number of recently modified files to display. Defaults to ",(0,r.jsx)(n.code,{children:"5"}),", set to ",(0,r.jsx)(n.code,{children:"0"})," to disable"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Maximum number of files to display per section. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStats"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the storage statistics section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideFavorites"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the favorites section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideRecent"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the recent files section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showDetailedStats"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", shows additional statistics including last modified date, largest file, hidden file count, total items, and file type breakdown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-63",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Basic usage:"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: filebrowser\n useProxy: true\n options:\n hostname: http://filebrowser.local:8080\n apiKey: DASHY_FILEBROWSER_KEY\n source: Documents\n path: /\n showRecent: 5\n favorites:\n - important-notes.txt\n - config.yaml\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"With detailed statistics:"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: filebrowser\n useProxy: true\n options:\n hostname: http://filebrowser.local:8080\n apiKey: DASHY_FILEBROWSER_KEY\n source: Downloads\n showDetailedStats: true\n showRecent: 10\n limit: 15\n"})}),"\n",(0,r.jsx)(n.h4,{id:"widget-sections",children:"Widget Sections"}),"\n",(0,r.jsx)(n.p,{children:"The widget displays up to four sections:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Storage Stats"})," - Directory name, total size, file and folder counts"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Detailed Stats"})," (optional) - Last modified date, largest file, hidden file count, total items, and file type breakdown with badges"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Favorites"})," - Quick-access links to user-specified files"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Recent Files"})," - Most recently modified files sorted by date"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"info-65",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/gtsteffaniak/filebrowser",children:"Filebrowser Quantum"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsx)(n.em,{children:"Self-Hosted"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"linkding",children:"Linkding"}),"\n",(0,r.jsx)(n.p,{children:"Linkding is a self-hosted bookmarking service, which has a clean interface and is simple to set up. This lists the links, filterable by tags."}),"\n",(0,r.jsx)(n.h4,{id:"options-67",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The hostname of the Drone CI instance."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The API key (",(0,r.jsx)(n.a,{href:"https://your-linkding-instance/settings/integrations",children:"https://your-linkding-instance/settings/integrations"}),")."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"tags"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"list of string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter the links by tag."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-64",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: linkding\n updateInterval: 30\n options:\n host: https://lingding.somedomain.com\n apiKey: my-very-secret-api-key\n tags: \n - rpg\n - markdown\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-66",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/sissbruecker/linkding",children:"Linkding"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/sissbruecker/linkding",children:"Linkding"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime-kuma",children:"Uptime Kuma"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," is an easy-to-use self-hosted monitoring tool."]}),"\n",(0,r.jsx)(n.h4,{id:"options-68",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Uptime Kuma instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The API key (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma/wiki/API-Keys",children:"https://github.com/louislam/uptime-kuma/wiki/API-Keys"}),")."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-65",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: uptime-kuma\n useProxy: true\n options:\n apiKey: uk2_99H0Yd3I2pPNIRfn0TqBFu4g5q85R1Mh75yZzw6H\n url: http://192.168.1.106:3691/metrics\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-67",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime-kuma-status-page",children:"Uptime Kuma Status Page"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," is an easy-to-use self-hosted monitoring tool."]}),"\n",(0,r.jsx)(n.h4,{id:"options-69",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Uptime Kuma instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"slug"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The slug of the status page"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monitorNames"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"strins"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Names of monitored services (in the same order as on the kuma uptime status page)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-66",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: uptime-kuma-status-page\n options:\n host: http://localhost:3001\n slug: another-beautiful-status-page\n monitorNames:\n - "Name1"\n - "Name2"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"info-68",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Needed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"tactical-rmm",children:"Tactical RMM"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})," is a self-hosted remote monitoring & management tool."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsxs)(n.a,{href:"https://github.com/user-attachments/assets/152a7205-e5de-401f-bad8-19063ddfaf3c",children:["\n ",(0,r.jsx)(n.img,{src:"https://github.com/user-attachments/assets/5921d46f-d84c-494d-8aaf-6b20cc592640",alt:"Capture",border:"0"})]})}),"\n",(0,r.jsx)(n.h4,{id:"options-70",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The status endpoint URL (",(0,r.jsx)(n.a,{href:"https://api.example.com/core/v2/status/",children:"https://api.example.com/core/v2/status/"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The MON_TOKEN (see ",(0,r.jsx)(n.a,{href:"https://docs.tacticalrmm.com/tipsntricks/#monitor-your-trmm-instance-via-the-built-in-monitoring-endpoint",children:"https://docs.tacticalrmm.com/tipsntricks/#monitor-your-trmm-instance-via-the-built-in-monitoring-endpoint"}),")."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-67",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: trmm\n useProxy: true\n options:\n token: PkPVKMzbmXgeQDlJWb0WXYvsIk3JvZyadURud2cSTdMia6hUbQ\n url: https://api.example.com/core/v2/status/\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-69",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"system-resource-monitoring",children:"System Resource Monitoring"}),"\n",(0,r.jsx)(n.h3,{id:"glances",children:"Glances"}),"\n",(0,r.jsxs)(n.p,{children:["The easiest method for displaying system info and resource usage in Dashy is with ",(0,r.jsx)(n.a,{href:"https://nicolargo.github.io/glances/",children:"Glances"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Glances is a cross-platform monitoring tool developed by ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo",children:"@nicolargo"}),". It's similar to top/htop but with a ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/api.html",children:"Rest API"})," and many ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/gw/index.html",children:"data exporters"})," available. Under the hood, it uses ",(0,r.jsx)(n.a,{href:"https://github.com/giampaolo/psutil",children:"psutil"})," for retrieving system info."]}),"\n",(0,r.jsxs)(n.p,{children:["If you don't already have it installed, either follow the ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/blob/master/README.rst",children:"Installation Guide"})," for your system, or setup ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/docker.html",children:"with Docker"}),", or use the one-line install script: ",(0,r.jsx)(n.code,{children:"curl -L https://bit.ly/glances | /bin/bash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are using Docker to run glances make sure to add the enviroment variable ",(0,r.jsx)(n.code,{children:"-e TZ = {YourTimeZone}"}),". You can get a list of valid timezones by running ",(0,r.jsx)(n.code,{children:"timedatectl list-timezones"})," on any linux system. This is needed so the graphs show the currect time."]}),"\n",(0,r.jsx)(n.p,{children:"Here an example for Docker"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{children:" docker run -d \\\n --name glances \\\n --restart unless-stopped \\\n -v /var/run/docker.sock:/var/run/docker.sock:ro \\\n -p 61208:61208 \\\n --pid host \\\n --privileged \\\n -e GLANCES_OPT=-w \\\n -e PUID=1000 \\\n -e PGID=1000 \\\n -e TZ=Europe/Zurich \\\n nicolargo/glances:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Glances can be launched with the ",(0,r.jsx)(n.code,{children:"glances"})," command. You'll need to run it in web server mode, using the ",(0,r.jsx)(n.code,{children:"-w"})," option for the API to be reachable. If you don't plan on using the Web UI, then you can disable it using ",(0,r.jsx)(n.code,{children:"--disable-webui"}),". See the ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/cmds.html",children:"command reference docs"})," for more info."]}),"\n",(0,r.jsxs)(n.p,{children:["If Glances is running on a Windows system it is recommended to add the following arguments ",(0,r.jsx)(n.code,{children:"--disable-plugin all --enable-plugin cpu,mem,diskio,ip,network,containers,quicklook,load,fs,alert -w"})," This is due to Glances not being that stable on windows, so disabling all plugins that aren't used by Dashy widgets can save on ressources."]}),"\n",(0,r.jsx)(n.h4,{id:"options-71",children:"Options"}),"\n",(0,r.jsxs)(n.p,{children:["All Glance's based widgets require a ",(0,r.jsx)(n.code,{children:"hostname"}),". All other parameters are optional."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL or IP + port to your Glances instance (without a trailing slash)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you have setup basic auth on Glances, specify username here (defaults to ",(0,r.jsx)(n.code,{children:"glances"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you have setup basic auth on Glances, specify password here. ",(0,r.jsx)(n.strong,{children:"Note"}),": since this password is in plaintext, it is important not to reuse it anywhere else"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify an API version, defaults to V ",(0,r.jsx)(n.code,{children:"3"}),". Note that support for older versions is limited"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"For widgets that show a time-series chart, optionally limit the number of data points returned. A higher number will show more historical results, but will take longer to load. A value between 300 - 800 is usually optimal"})]})]})]}),"\n",(0,r.jsxs)(n.p,{children:["Note that if auth is configured, requests must be proxied with ",(0,r.jsx)(n.code,{children:"useProxy: true"})]}),"\n",(0,r.jsx)(n.h4,{id:"info-70",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances",children:"GitHub - Nicolargo/Glances"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"screenshot",children:"Screenshot"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.a,{href:"https://ibb.co/pR6dMZT",children:(0,r.jsx)(n.img,{src:"https://pixelflare.cc/alicia/dashy/monitor-board",alt:"example-screenshot"})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-cpu-usage",children:"Current CPU Usage"}),"\n",(0,r.jsx)(n.p,{children:"Live-updating current CPU usage, as a combined average across all cores"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-cpu"})}),"\n",(0,r.jsx)(n.h4,{id:"example-68",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cpu\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-cpu-usage-speedometer",children:"Current CPU Usage Speedometer"}),"\n",(0,r.jsx)(n.p,{children:"Speedometer styled version of the Current CPU Usage widget"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-cpu-speedometer"})}),"\n",(0,r.jsx)(n.h4,{id:"example-69",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-speedometer\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-usage-per-core",children:"CPU Usage Per Core"}),"\n",(0,r.jsx)(n.p,{children:"Live-updating CPU usage breakdown per core"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cpu-per-core"})}),"\n",(0,r.jsx)(n.h4,{id:"example-70",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cores\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-usage-history",children:"CPU Usage History"}),"\n",(0,r.jsx)(n.p,{children:"Recent CPU usage history, across all cores, and displayed by user and system"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/cpu-history-chart"})}),"\n",(0,r.jsx)(n.h4,{id:"options-72",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results returned, rendering more data points will take longer to load. Defaults to ",(0,r.jsx)(n.code,{children:"100"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-71",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-history\n options:\n hostname: http://192.168.130.2:61208\n limit: 60\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-memory-usage",children:"Current Memory Usage"}),"\n",(0,r.jsx)(n.p,{children:"Real-time memory usage gauge, with more info visible on click"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-mem"})}),"\n",(0,r.jsx)(n.h4,{id:"example-72",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-mem\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-memory-usage-speedometer",children:"Current Memory Usage Speedometer"}),"\n",(0,r.jsx)(n.p,{children:"Speedometer styled version of the Current Memory Usage widget"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-mem-speedometer"})}),"\n",(0,r.jsx)(n.h4,{id:"example-73",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-mem-speedometer\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"memory-usage-history",children:"Memory Usage History"}),"\n",(0,r.jsx)(n.p,{children:"Recent memory usage chart"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/mem-history-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"options-73",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results returned, rendering more data points will take longer to load. Defaults to ",(0,r.jsx)(n.code,{children:"100"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-74",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-mem-history\n options:\n hostname: http://localhost:61208\n limit: 80\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"disk-space",children:"Disk Space"}),"\n",(0,r.jsx)(n.p,{children:"List connected disks, showing free / used space and other info (file system, mount point and space available)"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/diskspace-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-75",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-space\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"disk-io",children:"Disk IO"}),"\n",(0,r.jsx)(n.p,{children:"Shows real-time read and write speeds and operations per sec for each disk"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/disk-io-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-76",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-io\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"system-load",children:"System Load"}),"\n",(0,r.jsx)(n.p,{children:"Shows the number of processes waiting in the run-queue, averaged across all cores. Displays for past 5, 10 and 15 minutes"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/system-load-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-77",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-system-load\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime",children:"Uptime"}),"\n",(0,r.jsx)(n.p,{children:"Displays the system uptime fetched from Glances."}),"\n",(0,r.jsx)(n.h4,{id:"example-78",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-uptime\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"system-load-history",children:"System Load History"}),"\n",(0,r.jsxs)(n.p,{children:["Shows recent historical system load, calculated from the number of processes waiting in the run-queue, in 1, 5 and 15 minute intervals, and averaged across all cores. Optionally specify ",(0,r.jsx)(n.code,{children:"limit"})," to set number of results returned, defaults to ",(0,r.jsx)(n.code,{children:"500"}),", max ",(0,r.jsx)(n.code,{children:"100000"}),", but the higher the number the longer the load and render times will be."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/system-load-history-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-79",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-load-history\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"network-interfaces",children:"Network Interfaces"}),"\n",(0,r.jsx)(n.p,{children:"Lists visible network interfaces, including real-time upload/ download stats"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/network-interfaces-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-80",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-network-interfaces\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"network-traffic",children:"Network Traffic"}),"\n",(0,r.jsxs)(n.p,{children:["Shows amount of data recently uploaded/ downloaded across all network interfaces. Optionally set the ",(0,r.jsx)(n.code,{children:"limit"})," option to specify number historical of data points to return"]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/network-traffic-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-81",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-network-traffic\n options:\n hostname: http://192.168.130.2:61208\n limit: 500\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"resource-usage-alerts",children:"Resource Usage Alerts"}),"\n",(0,r.jsx)(n.p,{children:"Lists recent high resource usage alerts (e.g. CPU, mem, IO, load, temp)"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/w01NX5R/gl-alerts.png"})}),"\n",(0,r.jsx)(n.h4,{id:"example-82",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-alerts\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ip-address",children:"IP Address"}),"\n",(0,r.jsx)(n.p,{children:"Shows public and private IP address. Note that the ip plugin is not available on all instances of Glances."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/ip-addresses-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-83",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-ip-address\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-temp",children:"CPU Temp"}),"\n",(0,r.jsx)(n.p,{children:"Displays temperature data from system CPUs."}),"\n",(0,r.jsxs)(n.p,{children:["Note: This widget uses the ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/blob/develop/glances/plugins/glances_sensors.py",children:(0,r.jsx)(n.code,{children:"sensors"})})," plugin, which is disabled by default, and may cause ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/issues/1664#issuecomment-632063558",children:"performance issues"}),".\nYou'll need to enable the sensors plugin to use this widget, using: ",(0,r.jsx)(n.code,{children:"--enable-plugin sensors"})," when you start Glances."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cpu-temp-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"options-74",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Use ",(0,r.jsx)(n.code,{children:"C"})," to display temperatures in Celsius or ",(0,r.jsx)(n.code,{children:"F"})," to use Fahrenheit. Defaults to ",(0,r.jsx)(n.code,{children:"C"}),"."]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-84",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-temp\n options:\n hostname: http://192.168.130.2:61208\n units: C\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"compact-metrics",children:"Compact Metrics"}),"\n",(0,r.jsx)(n.p,{children:"A multi-system overview widget that displays CPU, memory and disk usage for multiple Glances instances in a compact table. Click on a row to see detailed metrics for that system."}),"\n",(0,r.jsx)(n.h4,{id:"options-75",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"systems"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["An array of systems to monitor, each with ",(0,r.jsx)(n.code,{children:"header"})," (display name) and ",(0,r.jsx)(n.code,{children:"url"})," (Glances base URL)"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Glances API version. Defaults to ",(0,r.jsx)(n.code,{children:"4"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If Glances is password-protected, specify the username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If Glances is password-protected, specify the password"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-85",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-compact-metrics\n options:\n systems:\n - header: Server 1\n url: http://192.168.1.10:61208\n - header: Server 2\n url: http://192.168.1.11:61208\n - header: NAS\n url: http://192.168.1.20:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"dynamic-widgets",children:"Dynamic Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"iframe-widget",children:"Iframe Widget"}),"\n",(0,r.jsx)(n.p,{children:"Embed any webpage into your dashboard as a widget."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/iframe"})}),"\n",(0,r.jsx)(n.h4,{id:"options-76",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to the webpage to embed"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"frameHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If needed, specify height of iframe in ",(0,r.jsx)(n.code,{children:"px"}),". E.g. ",(0,r.jsx)(n.code,{children:"400"}),", defaults to auto"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-86",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: iframe\n options:\n url: https://fiatleak.com/\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"html-embedded-widget",children:"HTML Embedded Widget"}),"\n",(0,r.jsx)(n.p,{children:"Many websites and apps provide their own embeddable widgets. These can be used with Dashy using the Embed widget, which lets you dynamically embed and HTML, CSS or JavaScript contents."}),"\n",(0,r.jsxs)(n.p,{children:["\u26a0\ufe0f ",(0,r.jsx)(n.strong,{children:"NOTE:"})," Use with extreme caution. Embedding a script from an untrustworthy source may have serious unintended consequences."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/html-embed"})}),"\n",(0,r.jsx)(n.h4,{id:"options-77",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"html"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"HTML contents to render in the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"script"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Raw JavaScript code to execute (caution)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"scriptSrc"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A URL to JavaScript content (caution)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"css"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Any stylings for widget contents"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-87",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: embed\n options:\n scriptSrc: https://cdn.speedcheck.org/basic/scbjs.min.js\n html: |\n
\n
\n \n Speedcheck\n \n
\n
\n'})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: embed\n options:\n css: \'.coinmarketcap-currency-widget { color: var(--widget-text-color); }\'\n html: \'
\'\n scriptSrc: \'https://files.coinmarketcap.com/static/widget/currency.js\'\n'})}),"\n",(0,r.jsx)(n.p,{children:"You can also use this widget to display an image, wither locally or from a remote origin."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: embed\n options:\n html: ''\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"api-response",children:"API Response"}),"\n",(0,r.jsx)(n.p,{children:"Directly output plain-text response from any API-enabled service."}),"\n",(0,r.jsx)(n.p,{children:"// Coming soon..."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"prometheus-data",children:"Prometheus Data"}),"\n",(0,r.jsx)(n.p,{children:"Display data from any service with a Prometheus exporter."}),"\n",(0,r.jsx)(n.p,{children:"// Coming soon..."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"data-feed",children:"Data Feed"}),"\n",(0,r.jsxs)(n.p,{children:["Show live data from an RSS-enabled service. The only required parameter is ",(0,r.jsx)(n.code,{children:"rssUrl"}),", which is the URL to the ATOM feed. See ",(0,r.jsx)(n.a,{href:"#rss-feed",children:"RSS Widget"})," for full list of available options."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"700",src:"https://pixelflare.cc/alicia/dashy/data-feed"})}),"\n",(0,r.jsx)(n.h4,{id:"example-88",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: rss-feed\n options:\n rssUrl: https://notes.aliciasykes.com/feed\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"usage--customizations",children:"Usage & Customizations"}),"\n",(0,r.jsx)(n.h3,{id:"widget-usage-guide",children:"Widget Usage Guide"}),"\n",(0,r.jsx)(n.p,{children:"Like items, widgets are placed under sections. You may have one or more widgets per section."}),"\n",(0,r.jsx)(n.p,{children:"In your YAML config file, this will look something like:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Today\n icon: far fa-calendar-day\n widgets:\n - type: clock\n options:\n format: en-GB\n - type: weather\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n city: London\n units: metric\n"})}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:'In this example, there is a single section, named "Today", using a Calendar icon from Font-Awesome. It has 2 widgets, a clock and the current weather.'}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"continuous-updates",children:"Continuous Updates"}),"\n",(0,r.jsxs)(n.p,{children:["By default, a widget which displays dynamic data from an external source, will only fetch results on page load. If you would like to keep data updated at all times, you can enable ",(0,r.jsx)(n.strong,{children:"Continuous Updates"}),". This is done by setting a time value in the ",(0,r.jsx)(n.code,{children:"updateInterval"})," field."]}),"\n",(0,r.jsxs)(n.p,{children:["The value of ",(0,r.jsx)(n.code,{children:"updateInterval"})," is optional, and is specified and seconds. It must be more than ",(0,r.jsx)(n.code,{children:"10"})," and less than ",(0,r.jsx)(n.code,{children:"7200"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example, the following widget displaying stats from Pi-Hole will update ever 20 seconds."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"widgets:\n- type: pi-hole-stats\n updateInterval: 20\n options:\n hostname: http://192.168.130.2\n"})}),"\n",(0,r.jsx)(n.p,{children:"Note that if you have many widgets, and set them to continuously update frequently, you will notice a hit to performance. A widget that relies on data from an external API, will also consume your usage quota faster, if set to keep updating."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"proxying-requests",children:"Proxying Requests"}),"\n",(0,r.jsx)(n.p,{children:"If a widget fails to make a data request, and the console shows a CORS error, this means the server is blocking client-side requests."}),"\n",(0,r.jsxs)(n.p,{children:["Dashy has a built-in CORS proxy (",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/services/cors-proxy.js",children:(0,r.jsx)(n.code,{children:"services/cors-proxy.js"})}),"), which will be used automatically by some widgets, or can be forced to use by other by setting the ",(0,r.jsx)(n.code,{children:"useProxy"})," option."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"widgets:\n- type: pi-hole-stats\n useProxy: true\n options:\n hostname: http://pi-hole.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"Alternatively, and more securely, you can set the auth headers on your service to accept requests from Dashy. For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"handling-secrets",children:"Handling Secrets"}),"\n",(0,r.jsxs)(n.p,{children:["Some widgets require you to pass potentially sensitive info such as API keys. The ",(0,r.jsx)(n.code,{children:"conf.yml"})," is not ideal for this, as it's stored in plaintext. Instead, for secrets you should use environment variables."]}),"\n",(0,r.jsxs)(n.p,{children:["In your widget options, set the value to the name of an environment variable starting with ",(0,r.jsx)(n.code,{children:"DASHY_"})," (or ",(0,r.jsx)(n.code,{children:"VITE_APP_"})," / ",(0,r.jsx)(n.code,{children:"VUE_APP_"})," for backwards compatibility). The Dashy server will substitute it with the matching ",(0,r.jsx)(n.code,{children:"process.env"})," value when proxying the request."]}),"\n",(0,r.jsxs)(n.p,{children:["The widget must be set to route through the Dashy server with ",(0,r.jsx)(n.code,{children:"useProxy: true"}),". Without this, the placeholder is sent directly to the upstream API and will fail. Substitution covers the request URL, headers and body, so any auth pattern (Bearer, Basic, query param, POST body) works."]}),"\n",(0,r.jsxs)(n.p,{children:["For more information about setting and managing your environment variables, see ",(0,r.jsx)(n.a,{href:"/docs/management#passing-in-environmental-variables",children:"Management Docs --\x3e Environmental Variables"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather\n useProxy: true\n options:\n apiKey: DASHY_WEATHER_TOKEN\n city: London\n units: metric\n hideDetails: true\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then set ",(0,r.jsx)(n.code,{children:"DASHY_WEATHER_TOKEN='xxx'"})," in your container or local environment, and restart Dashy. To rotate the value, just update the env var and restart. No rebuild required."]}),"\n",(0,r.jsxs)(n.admonition,{type:"note",children:[(0,r.jsxs)(n.p,{children:["Only env vars starting with ",(0,r.jsx)(n.code,{children:"DASHY_"}),", ",(0,r.jsx)(n.code,{children:"VITE_APP_"})," or ",(0,r.jsx)(n.code,{children:"VUE_APP_"})," are eligible for substitution. Other server-side env vars are never exposed."]}),(0,r.jsxs)(n.p,{children:["If you build Dashy from source yourself, ",(0,r.jsx)(n.code,{children:"VITE_APP_*"})," and ",(0,r.jsx)(n.code,{children:"DASHY_*"})," vars set at build time are also baked into the bundle by Vite, which works without ",(0,r.jsx)(n.code,{children:"useProxy: true"}),"."]})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"setting-timeout",children:"Setting Timeout"}),"\n",(0,r.jsxs)(n.p,{children:["If the endpoint you are requesting data from is slow to respond, you may see a timeout error in the console. This can easily be fixed by specifying the ",(0,r.jsx)(n.code,{children:"timeout"})," property on the offending widget. This should be an integer value, in milliseconds. By default timeout is ",(0,r.jsx)(n.code,{children:"2500"})," ms (2\xbd seconds)."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cpu\n timeout: 8000\n options:\n hostname: https://glances.dns-device.local\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adding-labels",children:"Adding Labels"}),"\n",(0,r.jsxs)(n.p,{children:["If you have multiple widgets of the same type in a single section, it may not be clear what each one is. To overcome this, you can add a custom label to any given widget, using the ",(0,r.jsx)(n.code,{children:"label"})," property."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- name: CPU Usage\n icon: fas fa-tachometer\n widgets:\n - type: gl-current-cpu\n label: Meida Server\n options:\n hostname: http://media-server.lan:61208\n - type: gl-current-cpu\n label: Firewall\n options:\n hostname: http://firewall.lan:61208\n - type: gl-current-cpu\n label: File Sync Server\n options:\n hostname: http://file-sync.lan:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ignoring-errors",children:"Ignoring Errors"}),"\n",(0,r.jsx)(n.p,{children:"When there's an error fetching or displaying a widgets data, then it will be highlighted in yellow, and a message displayed on the UI."}),"\n",(0,r.jsxs)(n.p,{children:["In some instances, this is a false positive, and the widget is actually functioning correctly. If this is the case, you can disable the UI error message of a given widget by setting: ",(0,r.jsx)(n.code,{children:"ignoreErrors: true"})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-io\n ignoreErrors: true\n options:\n hostname: https://glances.dns-device.local\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"widget-styling",children:"Widget Styling"}),"\n",(0,r.jsx)(n.p,{children:"Like elsewhere in Dashy, all colours can be easily modified with CSS variables."}),"\n",(0,r.jsx)(n.p,{children:"Widgets use the following color variables, which can be overridden if desired:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-text-color"})," - Text color, defaults to ",(0,r.jsx)(n.code,{children:"--primary"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-background-color"})," - Background color, defaults to ",(0,r.jsx)(n.code,{children:"--background-darker"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-accent-color"})," - Accent color, defaults to ",(0,r.jsx)(n.code,{children:"--background"})]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more info on how to apply custom variables, see the ",(0,r.jsx)(n.a,{href:"/docs/theming#setting-custom-css-in-the-ui",children:"Theming Docs"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"customizing-charts",children:"Customizing Charts"}),"\n",(0,r.jsxs)(n.p,{children:["For widgets that contain charts, you can set an array of colors under ",(0,r.jsx)(n.code,{children:"chartColors"}),".\nTo specify the chart height, set ",(0,r.jsx)(n.code,{children:"chartHeight"})," to an integer (in ",(0,r.jsx)(n.code,{children:"px"}),"), defaults to ",(0,r.jsx)(n.code,{children:"300"}),".\nFor example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-load-history\n options:\n hostname: http://192.168.130.2:61208\n chartColors: ['#9b5de5', '#f15bb5', '#00bbf9', '#00f5d4']\n chartHeight: 450\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"language-translations",children:"Language Translations"}),"\n",(0,r.jsx)(n.p,{children:"Since most of the content displayed within widgets is fetched from an external API, unless that API supports multiple languages, translating dynamic content is not possible."}),"\n",(0,r.jsx)(n.p,{children:"However, any hard-coded content is translatable, and all dates and times will display in your local format."}),"\n",(0,r.jsxs)(n.p,{children:["For more info about multi-language support, see the ",(0,r.jsx)(n.a,{href:"/docs/multi-language-support",children:"Internationalization Docs"}),"."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"widget-ui-options",children:"Widget UI Options"}),"\n",(0,r.jsx)(n.p,{children:"Widgets can be opened in full-page view, by clicking the Arrow icon (top-right). The URL in your address bar will also update, and visiting that web address directly will take you straight to that widget."}),"\n",(0,r.jsx)(n.p,{children:"You can reload the data of any widget, by clicking the Refresh Data icon (also in top-right). This will only affect the widget where the action was triggered from."}),"\n",(0,r.jsxs)(n.p,{children:["All ",(0,r.jsx)(n.a,{href:"/docs/configuring#section",children:"config options"})," that can be applied to sections, can also be applied to widget sections. For example, to make a widget section double the width, set ",(0,r.jsx)(n.code,{children:"displayData.cols: 2"})," within the parent section. You can collapse a widget (by clicking the section title), and collapse state will be saved locally."]}),"\n",(0,r.jsxs)(n.p,{children:["Widgets cannot currently be edited through the UI. This feature is in development, and will be released soon. In the meantime, you can either use the JSON config editor, or use ",(0,r.jsx)(n.a,{href:"https://github.com/coder/code-server",children:"VS Code Server"}),", or just SSH into your box and edit the conf.yml file directly."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"build-your-own-widget",children:"Build your own Widget"}),"\n",(0,r.jsx)(n.p,{children:"Widgets are built in a modular fashion, making it easy for anyone to create their own custom components."}),"\n",(0,r.jsxs)(n.p,{children:["For a full tutorial on creating your own widget, you can follow ",(0,r.jsx)(n.a,{href:"/docs/development-guides#building-a-widget",children:"this guide"}),", or take a look at ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/commit/3da76ce2999f57f76a97454c0276301e39957b8e",children:"here"})," for a code example."]}),"\n",(0,r.jsxs)(n.p,{children:["Alternatively, for displaying simple data, you could also just use the either the ",(0,r.jsx)(n.a,{href:"#iframe-widget",children:"iframe"}),", ",(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"embed"}),", ",(0,r.jsx)(n.a,{href:"#data-feed",children:"data feed"})," or ",(0,r.jsx)(n.a,{href:"#api-response",children:"API response"})," widgets."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"requesting-a-widget",children:"Requesting a Widget"}),"\n",(0,r.jsx)(n.p,{children:"Suggestions for widget ideas are welcome. But there is no guarantee that I will build your widget idea."}),"\n",(0,r.jsx)(n.p,{children:"Please only request widgets for services that:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Have a publicly accessible API"}),"\n",(0,r.jsx)(n.li,{children:"Are CORS and HTTPS enabled"}),"\n",(0,r.jsx)(n.li,{children:"Are free to use, or have a free plan"}),"\n",(0,r.jsx)(n.li,{children:"Allow for use in their Terms of Service"}),"\n",(0,r.jsx)(n.li,{children:"Would be useful for other users"}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["You can suggest a widget ",(0,r.jsx)(n.a,{href:"https://git.io/Jygo3",children:"here"}),", please star the repo before submitting a ticket. If you are a monthly GitHub sponsor, I will happily build out a custom widget for any service that meets the above criteria, usually within 2 weeks of initial request."]}),"\n",(0,r.jsxs)(n.p,{children:["For services that are not officially supported, it is likely still possible to display data using either the ",(0,r.jsx)(n.a,{href:"#iframe-widget",children:"iframe"}),", ",(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"embed"})," or ",(0,r.jsx)(n.a,{href:"#api-response",children:"API response"})," widgets. For more advanced features, like charts and action buttons, you could also build your own widget, using ",(0,r.jsx)(n.a,{href:"/docs/development-guides#building-a-widget",children:"this tutorial"}),", it's fairly straight forward, and you can use an ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/components/Widgets",children:"existing widget"})," (or ",(0,r.jsx)(n.a,{href:"https://git.io/JygKI",children:"this example"}),") as a template."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"troubleshooting-widget-errors",children:"Troubleshooting Widget Errors"}),"\n",(0,r.jsxs)(n.p,{children:["If an error occurs when fetching or rendering results, you will see a short message in the UI. If that message doesn't adequately explain the problem, then you can ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open the browser console"})," to see more details."]}),"\n",(0,r.jsx)(n.p,{children:"Before proceeding, ensure that if the widget requires auth your API is correct, and for custom widgets, double check that the URL and protocol is correct."}),"\n",(0,r.jsx)(n.p,{children:"If you're able to, you can find more information about why the request may be failing in the Dev Tools under the Network tab, and you can ensure your endpoint is correct and working using a tool like Postman."}),"\n",(0,r.jsx)(n.h4,{id:"cors-errors",children:"CORS Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The most common issue is a CORS error. This is a browser security mechanism which prevents the client-side app (Dashy) from from accessing resources on a remote origin, without that server's explicit permission (e.g. with headers like Access-Control-Allow-Origin). See the MDN Docs for more info: ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"There are several ways to fix a CORS error:"}),"\n",(0,r.jsx)(n.h4,{id:"option-1---ensure-correct-protocol",children:"Option 1 - Ensure Correct Protocol"}),"\n",(0,r.jsx)(n.p,{children:"You will get a CORS error if you try and access a http service from a https source. So ensure that the URL you are requesting has the right protocol, and is correctly formatted."}),"\n",(0,r.jsx)(n.h4,{id:"option-2---set-headers",children:"Option 2 - Set Headers"}),"\n",(0,r.jsxs)(n.p,{children:["If you have control over the destination (e.g. for a self-hosted service), then you can simply apply the correct headers.\nAdd the ",(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, with the value of either ",(0,r.jsx)(n.code,{children:"*"})," to allow requests from anywhere, or more securely, the host of where Dashy is served from. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://url-of-dashy.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: *\n"})}),"\n",(0,r.jsx)(n.h4,{id:"option-3---proxying-request",children:"Option 3 - Proxying Request"}),"\n",(0,r.jsxs)(n.p,{children:["You can route requests through Dashy's built-in CORS proxy. Instructions and more details can be found ",(0,r.jsx)(n.a,{href:"#proxying-requests",children:"here"}),". If you don't have control over the target origin, and you are running Dashy either through Docker, with the Node server or on Netlify, then this solution will work for you."]}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.code,{children:"useProxy: true"})," option to the failing widget."]}),"\n",(0,r.jsx)(n.h4,{id:"option-4---use-a-plugin",children:"Option 4 - Use a plugin"}),"\n",(0,r.jsxs)(n.p,{children:["For testing purposes, you can use an addon, which will disable the CORS checks. You can get the Allow-CORS extension for ",(0,r.jsx)(n.a,{href:"https://chrome.google.com/webstore/detail/allow-cors-access-control/lhobafahddgcelffkeicbaginigeejlf?hl=en-US",children:"Chrome"})," or ",(0,r.jsx)(n.a,{href:"https://addons.mozilla.org/en-US/firefox/addon/access-control-allow-origin/",children:"Firefox"}),", more details ",(0,r.jsx)(n.a,{href:"https://mybrowseraddon.com/access-control-allow-origin.html",children:"here"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"raising-an-issue",children:"Raising an Issue"}),"\n",(0,r.jsxs)(n.p,{children:["If you need to submit a bug report for a failing widget, then please include the full console output (see ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"how"}),") as well as the relevant parts of your config file. Before sending the request, ensure you've read the docs. If you're new to GitHub, an haven't previously contributed to the project, then please fist star the repo to avoid your ticket being closed by the anti-spam bot."]})]})}function x(e={}){const{wrapper:n}={...(0,d.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(o,{...e})}):o(e)}},8453(e,n,s){s.d(n,{R:()=>l,x:()=>t});var i=s(6540);const r={},d=i.createContext(r);function l(e){const n=i.useContext(d);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function t(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:l(e.components),i.createElement(d.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[5552],{8803(e,n,s){s.r(n),s.d(n,{assets:()=>c,contentTitle:()=>t,default:()=>x,frontMatter:()=>l,metadata:()=>i,toc:()=>h});const i=JSON.parse('{"id":"widgets","title":"Widgets","description":"Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API.","source":"@site/docs/widgets.md","sourceDirName":".","slug":"/widgets","permalink":"/docs/widgets","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/widgets.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Icons","permalink":"/docs/icons"},"next":{"title":"Theming","permalink":"/docs/theming"}}');var r=s(4848),d=s(8453);const l={},t="Widgets",c={},h=[{value:"Contents",id:"contents",level:2},{value:"General Widgets",id:"general-widgets",level:2},{value:"Clock",id:"clock",level:3},{value:"Options",id:"options",level:4},{value:"Example",id:"example",level:4},{value:"Info",id:"info",level:4},{value:"Weather",id:"weather",level:3},{value:"Options",id:"options-1",level:4},{value:"Example",id:"example-1",level:4},{value:"Info",id:"info-1",level:4},{value:"Weather Forecast",id:"weather-forecast",level:3},{value:"Options",id:"options-2",level:4},{value:"Example",id:"example-2",level:4},{value:"Info",id:"info-2",level:4},{value:"RSS Feed",id:"rss-feed",level:3},{value:"Options",id:"options-3",level:4},{value:"Example",id:"example-3",level:4},{value:"Info",id:"info-3",level:4},{value:"Image",id:"image",level:3},{value:"Options",id:"options-4",level:4},{value:"Example",id:"example-4",level:4},{value:"Info",id:"info-4",level:4},{value:"Public IP",id:"public-ip",level:3},{value:"Options",id:"options-5",level:4},{value:"Example",id:"example-5",level:4},{value:"Supported providers",id:"supported-providers",level:4},{value:"Info",id:"info-5",level:4},{value:"IP Blacklist",id:"ip-blacklist",level:3},{value:"Options",id:"options-6",level:4},{value:"Example",id:"example-6",level:4},{value:"Info",id:"info-6",level:4},{value:"Domain Monitor",id:"domain-monitor",level:3},{value:"Options",id:"options-7",level:4},{value:"Example",id:"example-7",level:4},{value:"Info",id:"info-7",level:4},{value:"Crypto Watch List",id:"crypto-watch-list",level:3},{value:"Options",id:"options-8",level:4},{value:"Example",id:"example-8",level:4},{value:"Info",id:"info-8",level:4},{value:"Crypto Token Price History",id:"crypto-token-price-history",level:3},{value:"Options",id:"options-9",level:4},{value:"Example",id:"example-9",level:4},{value:"Info",id:"info-9",level:4},{value:"Wallet Balance",id:"wallet-balance",level:3},{value:"Options",id:"options-10",level:4},{value:"Example",id:"example-10",level:4},{value:"Info",id:"info-10",level:4},{value:"Code Stats",id:"code-stats",level:3},{value:"Options",id:"options-11",level:4},{value:"Example",id:"example-11",level:4},{value:"Info",id:"info-11",level:4},{value:"Mullvad Status",id:"mullvad-status",level:3},{value:"Options",id:"options-12",level:4},{value:"Example",id:"example-12",level:4},{value:"Info",id:"info-12",level:4},{value:"addy.io",id:"addyio",level:3},{value:"Options",id:"options-13",level:4},{value:"Example",id:"example-13",level:4},{value:"Info",id:"info-13",level:4},{value:"Vulnerability Feed",id:"vulnerability-feed",level:3},{value:"Options",id:"options-14",level:4},{value:"Example",id:"example-14",level:4},{value:"Info",id:"info-14",level:4},{value:"Exchange Rates",id:"exchange-rates",level:3},{value:"Options",id:"options-15",level:4},{value:"Example",id:"example-15",level:4},{value:"Info",id:"info-15",level:4},{value:"Public Holidays",id:"public-holidays",level:3},{value:"Options",id:"options-16",level:4},{value:"Example",id:"example-16",level:4},{value:"Info",id:"info-16",level:4},{value:"Covid-19 Status",id:"covid-19-status",level:3},{value:"Options",id:"options-17",level:4},{value:"Example",id:"example-17",level:4},{value:"Info",id:"info-17",level:4},{value:"Sports Scores",id:"sports-scores",level:3},{value:"Options",id:"options-18",level:4},{value:"Example",id:"example-18",level:4},{value:"Info",id:"info-18",level:4},{value:"News Headlines",id:"news-headlines",level:3},{value:"Options",id:"options-19",level:4},{value:"Example",id:"example-19",level:4},{value:"Info",id:"info-19",level:4},{value:"TFL Status",id:"tfl-status",level:3},{value:"Options",id:"options-20",level:4},{value:"Example",id:"example-20",level:4},{value:"Info",id:"info-20",level:4},{value:"Stock Price History",id:"stock-price-history",level:3},{value:"Options",id:"options-21",level:4},{value:"Example",id:"example-21",level:4},{value:"Info",id:"info-21",level:4},{value:"ETH Gas Prices",id:"eth-gas-prices",level:3},{value:"Options",id:"options-22",level:4},{value:"Example",id:"example-22",level:4},{value:"Info",id:"info-22",level:4},{value:"Joke",id:"joke",level:3},{value:"Options",id:"options-23",level:4},{value:"Example",id:"example-23",level:4},{value:"Info",id:"info-23",level:4},{value:"Chuck Norris quotes",id:"chuck-norris-quotes",level:3},{value:"Options",id:"options-24",level:4},{value:"Example",id:"example-24",level:4},{value:"Info",id:"info-24",level:4},{value:"XKCD Comics",id:"xkcd-comics",level:3},{value:"Options",id:"options-25",level:4},{value:"Example",id:"example-25",level:4},{value:"Info",id:"info-25",level:4},{value:"Flight Data",id:"flight-data",level:3},{value:"Options",id:"options-26",level:4},{value:"Example",id:"example-26",level:4},{value:"Info",id:"info-26",level:4},{value:"Astronomy Picture of the Day",id:"astronomy-picture-of-the-day",level:3},{value:"Options",id:"options-27",level:4},{value:"Example",id:"example-27",level:4},{value:"Info",id:"info-27",level:4},{value:"GitHub Trending",id:"github-trending",level:3},{value:"Options",id:"options-28",level:4},{value:"Example",id:"example-28",level:4},{value:"Info",id:"info-28",level:4},{value:"GitHub Profile Stats",id:"github-profile-stats",level:3},{value:"Options",id:"options-29",level:4},{value:"Example",id:"example-29",level:4},{value:"Info",id:"info-29",level:4},{value:"HealthChecks Status",id:"healthchecks-status",level:3},{value:"Options",id:"options-30",level:4},{value:"Info",id:"info-30",level:4},{value:"Hackernews Trending",id:"hackernews-trending",level:3},{value:"Options",id:"options-31",level:4},{value:"Example",id:"example-30",level:5},{value:"MVG Departure",id:"mvg-departure",level:3},{value:"Options",id:"options-32",level:4},{value:"Info",id:"info-31",level:4},{value:"MVG Connection",id:"mvg-connection",level:3},{value:"Options",id:"options-33",level:4},{value:"Info",id:"info-32",level:4},{value:"Custom List",id:"custom-list",level:3},{value:"Options",id:"options-34",level:4},{value:"Json Schema",id:"json-schema",level:4},{value:"Notes",id:"notes",level:4},{value:"Example",id:"example-31",level:4},{value:"Info",id:"info-33",level:4},{value:"Custom search",id:"custom-search",level:3},{value:"Options",id:"options-35",level:4},{value:"Notes",id:"notes-1",level:4},{value:"Example",id:"example-32",level:4},{value:"Info",id:"info-34",level:4},{value:"RescueTime Overview",id:"rescuetime-overview",level:3},{value:"Options",id:"options-36",level:4},{value:"Example",id:"example-33",level:4},{value:"Info",id:"info-35",level:4},{value:"Minecraft Server",id:"minecraft-server",level:3},{value:"Options",id:"options-37",level:4},{value:"Example",id:"example-34",level:4},{value:"Info",id:"info-36",level:4},{value:"Self-Hosted Services Widgets",id:"self-hosted-services-widgets",level:2},{value:"System Info",id:"system-info",level:3},{value:"Options",id:"options-38",level:4},{value:"Example",id:"example-35",level:4},{value:"Info",id:"info-37",level:4},{value:"Cron Monitoring (Health Checks)",id:"cron-monitoring-health-checks",level:3},{value:"Options",id:"options-39",level:4},{value:"Example",id:"example-36",level:4},{value:"Info",id:"info-38",level:4},{value:"CPU History (NetData)",id:"cpu-history-netdata",level:3},{value:"Options",id:"options-40",level:4},{value:"Example",id:"example-37",level:4},{value:"Info",id:"info-39",level:4},{value:"Memory History (NetData)",id:"memory-history-netdata",level:3},{value:"Options",id:"options-41",level:4},{value:"Example",id:"example-38",level:4},{value:"Info",id:"info-40",level:4},{value:"Load History (NetData)",id:"load-history-netdata",level:3},{value:"Options",id:"options-42",level:4},{value:"Example",id:"example-39",level:4},{value:"Info",id:"info-41",level:4},{value:"Pi-Hole Stats",id:"pi-hole-stats",level:3},{value:"Options",id:"options-43",level:4},{value:"Example",id:"example-40",level:4},{value:"Info",id:"info-42",level:4},{value:"Pi-Hole Stats v6",id:"pi-hole-stats-v6",level:3},{value:"Options",id:"options-44",level:4},{value:"Example",id:"example-41",level:4},{value:"Info",id:"info-43",level:4},{value:"Pi-Hole Queries",id:"pi-hole-queries",level:3},{value:"Options",id:"options-45",level:4},{value:"Example",id:"example-42",level:4},{value:"Info",id:"info-44",level:4},{value:"Pi-Hole Queries v6",id:"pi-hole-queries-v6",level:3},{value:"Options",id:"options-46",level:4},{value:"Example",id:"example-43",level:4},{value:"Info",id:"info-45",level:4},{value:"Pi-Hole Recent Traffic",id:"pi-hole-recent-traffic",level:3},{value:"Options",id:"options-47",level:4},{value:"Example",id:"example-44",level:4},{value:"Info",id:"info-46",level:4},{value:"Pi-Hole Recent Traffic v6",id:"pi-hole-recent-traffic-v6",level:3},{value:"Options",id:"options-48",level:4},{value:"Example",id:"example-45",level:4},{value:"Info",id:"info-47",level:4},{value:"Stat Ping Statuses",id:"stat-ping-statuses",level:3},{value:"Options",id:"options-49",level:4},{value:"Example",id:"example-46",level:4},{value:"Info",id:"info-48",level:4},{value:"Synology Download Station",id:"synology-download-station",level:3},{value:"Options",id:"options-50",level:4},{value:"Example",id:"example-47",level:4},{value:"Info",id:"info-49",level:4},{value:"AdGuard Home Block Stats",id:"adguard-home-block-stats",level:3},{value:"Options",id:"options-51",level:4},{value:"Example",id:"example-48",level:4},{value:"Info",id:"info-50",level:4},{value:"AdGuard Home Filters",id:"adguard-home-filters",level:3},{value:"Options",id:"options-52",level:4},{value:"Example",id:"example-49",level:4},{value:"Info",id:"info-51",level:4},{value:"AdGuard Home DNS Info",id:"adguard-home-dns-info",level:3},{value:"Options",id:"options-53",level:4},{value:"Example",id:"example-50",level:4},{value:"Info",id:"info-52",level:4},{value:"AdGuard Home Top Domains",id:"adguard-home-top-domains",level:3},{value:"Options",id:"options-54",level:4},{value:"Example",id:"example-51",level:4},{value:"Info",id:"info-53",level:4},{value:"Nextcloud User",id:"nextcloud-user",level:3},{value:"Options",id:"options-55",level:4},{value:"Example",id:"example-52",level:4},{value:"Info",id:"info-54",level:4},{value:"Nextcloud User Statuses",id:"nextcloud-user-statuses",level:3},{value:"Options",id:"options-56",level:4},{value:"Example",id:"example-53",level:4},{value:"Info",id:"info-55",level:4},{value:"Nextcloud Notifications",id:"nextcloud-notifications",level:3},{value:"Options",id:"options-57",level:4},{value:"Example",id:"example-54",level:4},{value:"Info",id:"info-56",level:4},{value:"Nextcloud System",id:"nextcloud-system",level:3},{value:"Options",id:"options-58",level:4},{value:"Example",id:"example-55",level:4},{value:"Info",id:"info-57",level:4},{value:"Nextcloud Stats",id:"nextcloud-stats",level:3},{value:"Options",id:"options-59",level:4},{value:"Example",id:"example-56",level:4},{value:"Info",id:"info-58",level:4},{value:"Nextcloud PHP OPcache Stats",id:"nextcloud-php-opcache-stats",level:3},{value:"Options",id:"options-60",level:4},{value:"Example",id:"example-57",level:4},{value:"Info",id:"info-59",level:4},{value:"ntfy stream",id:"ntfy-stream",level:3},{value:"Options",id:"options-61",level:4},{value:"Example",id:"example-58",level:4},{value:"Info",id:"info-60",level:4},{value:"Proxmox lists",id:"proxmox-lists",level:3},{value:"Options",id:"options-62",level:4},{value:"Example",id:"example-59",level:4},{value:"Info",id:"info-61",level:4},{value:"Troubleshooting",id:"troubleshooting",level:4},{value:"Sabnzbd",id:"sabnzbd",level:3},{value:"Options",id:"options-63",level:4},{value:"Example",id:"example-60",level:4},{value:"Info",id:"info-62",level:4},{value:"Gluetun VPN Info",id:"gluetun-vpn-info",level:3},{value:"Options",id:"options-64",level:4},{value:"Example",id:"example-61",level:4},{value:"Info",id:"info-63",level:4},{value:"Drone CI Builds",id:"drone-ci-builds",level:3},{value:"Options",id:"options-65",level:4},{value:"Example",id:"example-62",level:4},{value:"Info",id:"info-64",level:4},{value:"Filebrowser",id:"filebrowser",level:3},{value:"Options",id:"options-66",level:4},{value:"Example",id:"example-63",level:4},{value:"Widget Sections",id:"widget-sections",level:4},{value:"Info",id:"info-65",level:4},{value:"Linkding",id:"linkding",level:3},{value:"Options",id:"options-67",level:4},{value:"Example",id:"example-64",level:4},{value:"Info",id:"info-66",level:4},{value:"Uptime Kuma",id:"uptime-kuma",level:3},{value:"Options",id:"options-68",level:4},{value:"Example",id:"example-65",level:4},{value:"Info",id:"info-67",level:4},{value:"Uptime Kuma Status Page",id:"uptime-kuma-status-page",level:3},{value:"Options",id:"options-69",level:4},{value:"Example",id:"example-66",level:4},{value:"Info",id:"info-68",level:4},{value:"Tactical RMM",id:"tactical-rmm",level:3},{value:"Options",id:"options-70",level:4},{value:"Example",id:"example-67",level:4},{value:"Info",id:"info-69",level:4},{value:"System Resource Monitoring",id:"system-resource-monitoring",level:2},{value:"Glances",id:"glances",level:3},{value:"Options",id:"options-71",level:4},{value:"Info",id:"info-70",level:4},{value:"Screenshot",id:"screenshot",level:4},{value:"Current CPU Usage",id:"current-cpu-usage",level:3},{value:"Example",id:"example-68",level:4},{value:"Current CPU Usage Speedometer",id:"current-cpu-usage-speedometer",level:3},{value:"Example",id:"example-69",level:4},{value:"CPU Usage Per Core",id:"cpu-usage-per-core",level:3},{value:"Example",id:"example-70",level:4},{value:"CPU Usage History",id:"cpu-usage-history",level:3},{value:"Options",id:"options-72",level:4},{value:"Example",id:"example-71",level:4},{value:"Current Memory Usage",id:"current-memory-usage",level:3},{value:"Example",id:"example-72",level:4},{value:"Current Memory Usage Speedometer",id:"current-memory-usage-speedometer",level:3},{value:"Example",id:"example-73",level:4},{value:"Memory Usage History",id:"memory-usage-history",level:3},{value:"Options",id:"options-73",level:4},{value:"Example",id:"example-74",level:4},{value:"Disk Space",id:"disk-space",level:3},{value:"Example",id:"example-75",level:4},{value:"Disk IO",id:"disk-io",level:3},{value:"Example",id:"example-76",level:4},{value:"System Load",id:"system-load",level:3},{value:"Example",id:"example-77",level:4},{value:"Uptime",id:"uptime",level:3},{value:"Example",id:"example-78",level:4},{value:"System Load History",id:"system-load-history",level:3},{value:"Example",id:"example-79",level:4},{value:"Network Interfaces",id:"network-interfaces",level:3},{value:"Example",id:"example-80",level:4},{value:"Network Traffic",id:"network-traffic",level:3},{value:"Example",id:"example-81",level:4},{value:"Resource Usage Alerts",id:"resource-usage-alerts",level:3},{value:"Example",id:"example-82",level:4},{value:"IP Address",id:"ip-address",level:3},{value:"Example",id:"example-83",level:4},{value:"CPU Temp",id:"cpu-temp",level:3},{value:"Options",id:"options-74",level:4},{value:"Example",id:"example-84",level:4},{value:"Compact Metrics",id:"compact-metrics",level:3},{value:"Options",id:"options-75",level:4},{value:"Example",id:"example-85",level:4},{value:"Dynamic Widgets",id:"dynamic-widgets",level:2},{value:"Iframe Widget",id:"iframe-widget",level:3},{value:"Options",id:"options-76",level:4},{value:"Example",id:"example-86",level:4},{value:"HTML Embedded Widget",id:"html-embedded-widget",level:3},{value:"Options",id:"options-77",level:4},{value:"Example",id:"example-87",level:4},{value:"API Response",id:"api-response",level:3},{value:"Prometheus Data",id:"prometheus-data",level:3},{value:"Data Feed",id:"data-feed",level:3},{value:"Example",id:"example-88",level:4},{value:"Usage & Customizations",id:"usage--customizations",level:2},{value:"Widget Usage Guide",id:"widget-usage-guide",level:3},{value:"Continuous Updates",id:"continuous-updates",level:3},{value:"Proxying Requests",id:"proxying-requests",level:3},{value:"Handling Secrets",id:"handling-secrets",level:3},{value:"Setting Timeout",id:"setting-timeout",level:3},{value:"Adding Labels",id:"adding-labels",level:3},{value:"Ignoring Errors",id:"ignoring-errors",level:3},{value:"Widget Styling",id:"widget-styling",level:3},{value:"Customizing Charts",id:"customizing-charts",level:3},{value:"Language Translations",id:"language-translations",level:3},{value:"Widget UI Options",id:"widget-ui-options",level:3},{value:"Build your own Widget",id:"build-your-own-widget",level:3},{value:"Requesting a Widget",id:"requesting-a-widget",level:3},{value:"Troubleshooting Widget Errors",id:"troubleshooting-widget-errors",level:3},{value:"CORS Errors",id:"cors-errors",level:4},{value:"Option 1 - Ensure Correct Protocol",id:"option-1---ensure-correct-protocol",level:4},{value:"Option 2 - Set Headers",id:"option-2---set-headers",level:4},{value:"Option 3 - Proxying Request",id:"option-3---proxying-request",level:4},{value:"Option 4 - Use a plugin",id:"option-4---use-a-plugin",level:4},{value:"Raising an Issue",id:"raising-an-issue",level:3}];function o(e){const n={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",h5:"h5",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,d.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"widgets",children:"Widgets"})}),"\n",(0,r.jsx)(n.p,{children:"Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API."}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#general-widgets",children:"General Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#clock",children:"Clock"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather",children:"Weather"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather-forecast",children:"Weather Forecast"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#rss-feed",children:"RSS Feed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#image",children:"Image"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-ip",children:"Public IP Address"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ip-blacklist",children:"IP Blacklist Checker"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#domain-monitor",children:"Domain Monitor"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#crypto-watch-list",children:"Crypto Watch List"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#crypto-token-price-history",children:"Crypto Price History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#wallet-balance",children:"Crypto Wallet Balance"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#code-stats",children:"Code Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mullvad-status",children:"Mullvad Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#addyio",children:"Email Aliases (addy.io)"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#vulnerability-feed",children:"Vulnerability Feed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#exchange-rates",children:"Exchange Rates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-holidays",children:"Public Holidays"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#covid-19-status",children:"Covid-19 Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sports-scores",children:"Sports Scores"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#news-headlines",children:"News Headlines"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tfl-status",children:"TFL Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#stock-price-history",children:"Stock Price History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#eth-gas-prices",children:"ETH Gas Prices"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#joke",children:"Joke of the Day"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#chuck-norris-quotes",children:"Chuck Norris quotes"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#xkcd-comics",children:"XKCD Comics"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#flight-data",children:"Flight Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#astronomy-picture-of-the-day",children:"NASA APOD"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#github-trending",children:"GitHub Trending"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#github-profile-stats",children:"GitHub Profile Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthchecks-status",children:"Healthchecks Status"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#hackernews-trending",children:"Hackernews Trending"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mvg-departure",children:"Mvg Departure"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mvg-connection",children:"Mvg Connection"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#custom-search",children:"Custom search"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#rescuetime-overview",children:"Rescuetime overview"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#minecraft-server",children:"Minecraft Server"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#self-hosted-services-widgets",children:"Self-Hosted Services Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-info",children:"System Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cron-monitoring-health-checks",children:"Cron Monitoring"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-history-netdata",children:"CPU History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#memory-history-netdata",children:"Memory History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#load-history-netdata",children:"System Load History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-stats",children:"Pi-Hole Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-stats-v6",children:"Pi-Hole Stats v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-queries",children:"Pi-Hole Queries"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-queries-v6",children:"Pi-Hole Queries v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-recent-traffic",children:"Pi-Hole Recent Traffic"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#pi-hole-recent-traffic-v6",children:"Pi-Hole Recent Traffic v6"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#stat-ping-statuses",children:"Stat Ping Statuses"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#synology-download-station",children:"Synology Download Station"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-block-stats",children:"AdGuard Home Block Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-filters",children:"AdGuard Home Filters"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-dns-info",children:"AdGuard Home DNS Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adguard-home-top-domains",children:"AdGuard Home Top Domains"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-user",children:"Nextcloud User"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-user-statuses",children:"Nextcloud User Statuses"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-notifications",children:"Nextcloud Notifications"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-system",children:"Nextcloud System"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-stats",children:"Nextcloud Stats"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#nextcloud-php-opcache-stats",children:"Nextcloud PHP OPcache"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ntfy-stream",children:"Ntfy stream"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#proxmox-lists",children:"Proxmox lists"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sabnzbd",children:"Sabnzbd"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#gluetun-vpn-info",children:"Gluetun VPN Info"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#drone-ci-builds",children:"Drone CI Build"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#filebrowser",children:"Filebrowser"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#linkding",children:"Linkding"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#uptime-kuma",children:"Uptime Kuma"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#uptime-kuma-status-page",children:"Uptime Kuma Status Page"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#tactical-rmm",children:"Tactical RMM"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#system-resource-monitoring",children:"System Resource Monitoring"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#current-cpu-usage",children:"CPU Usage Current"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-usage-per-core",children:"CPU Usage Per Core"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-usage-history",children:"CPU Usage History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#current-memory-usage",children:"Memory Usage Current"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#memory-usage-history",children:"Memory Usage History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disk-space",children:"Disk Space"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#disk-io",children:"Disk IO"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-load",children:"System Load"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#system-load-history",children:"System Load History"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-interfaces",children:"Network Interfaces"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#network-traffic",children:"Network Traffic"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#resource-usage-alerts",children:"Resource Usage Alerts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ip-address",children:"Public & Private IP"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cpu-temp",children:"CPU Temperature"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#compact-metrics",children:"Compact Metrics"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#dynamic-widgets",children:"Dynamic Widgets"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#iframe-widget",children:"Iframe Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"HTML Embed Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#api-response",children:"API Response"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#prometheus-data",children:"Prometheus Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#data-feed",children:"Data Feed"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.a,{href:"#usage--customizations",children:"Usage & Customizations"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-usage-guide",children:"Widget Usage Guide"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#continuous-updates",children:"Continuous Updates"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#proxying-requests",children:"Proxying Requests"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#setting-timeout",children:"Setting Timeout"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#adding-labels",children:"Adding Labels"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ignoring-errors",children:"Ignoring Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-styling",children:"Custom CSS Styling"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#customizing-charts",children:"Customizing Charts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#language-translations",children:"Language Translations"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-ui-options",children:"Widget UI Options"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#build-your-own-widget",children:"Building a Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#requesting-a-widget",children:"Requesting a Widget"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#troubleshooting-widget-errors",children:"Troubleshooting"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.h2,{id:"general-widgets",children:"General Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"clock",children:"Clock"}),"\n",(0,r.jsx)(n.p,{children:"A simple, live-updating time and date widget with time-zone support. All fields are optional."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/clock"})}),"\n",(0,r.jsx)(n.h4,{id:"options",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"timeZone"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The time zone to display date and time in.",(0,r.jsx)(n.br,{})," Specified as Region/City, for example: ",(0,r.jsx)(n.code,{children:"Australia/Melbourne"}),". See the ",(0,r.jsx)(n.a,{href:"https://timezonedb.com/time-zones",children:"Time Zone DB"})," for a full list of supported TZs. Defaults to the browser / device's local time"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"format"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["A country code for displaying the date and time in local format.",(0,r.jsx)(n.br,{}),"Specified as ",(0,r.jsx)(n.code,{children:"[ISO-3166]-[ISO-639]"}),", for example: ",(0,r.jsx)(n.code,{children:"en-AU"}),". See ",(0,r.jsx)(n.a,{href:"https://www.fincher.org/Utilities/CountryLanguageList.shtml",children:"here"})," for a full list of locales. Defaults to the browser / device's region"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"customCityName"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"By default the city from the time-zone is shown, but setting this value will override that text"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDate"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the date and city will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideSeconds"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", seconds will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"use12Hour"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", 12 hour time will be displayed. Defaults to the settings suggested by the current ",(0,r.jsx)(n.code,{children:"format"})," and ",(0,r.jsx)(n.code,{children:"timeZone"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: clock\n options:\n timeZone: Europe/London\n format: en-GB\n hideDate: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No external data requests."})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"weather",children:"Weather"}),"\n",(0,r.jsx)(n.p,{children:"A simple, live-updating local weather component, showing temperature, conditions and more info."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/weather"})}),"\n",(0,r.jsx)(n.h4,{id:"options-1",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your OpenWeatherMap API key. You can get one for free at ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"city"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A city name to use for fetching weather. This can also be a state code or country code, following the ISO-3166 format"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cityId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An OpenWeatherMap numeric city ID, used to disambiguate cities that share a name. You can find the ID in the URL of the city's page on ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," (e.g. ",(0,r.jsx)(n.code,{children:"2643743"})," for London, GB). If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," option"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The units to use for displaying data, can be either ",(0,r.jsx)(n.code,{children:"metric"})," or ",(0,r.jsx)(n.code,{children:"imperial"}),". Defaults to ",(0,r.jsx)(n.code,{children:"metric"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the additional details (wind, humidity, pressure, etc) will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lat"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To show weather for a specific location, you can provide the latitude and longitude coordinates. If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lon"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To show weather for a specific location, you can provide the latitude and longitude coordinates. If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-1",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n city: London\n units: metric\n hideDetails: true\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-1",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"weather-forecast",children:"Weather Forecast"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the weather (temperature and conditions) for the next few days for a given location. Note that this requires either the free ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"OpenWeatherMap Student Plan"}),", or the Premium Plan."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/weather-forecast"})}),"\n",(0,r.jsx)(n.h4,{id:"options-2",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your OpenWeatherMap API key. You can get one at ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," or for free via the ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"OWM Student Plan"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"city"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A city name to use for fetching weather. This can also be a state code or country code, following the ISO-3166 format"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cityId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An OpenWeatherMap numeric city ID, used to disambiguate cities that share a name. You can find the ID in the URL of the city's page on ",(0,r.jsx)(n.a,{href:"https://openweathermap.org/",children:"openweathermap.org"})," (e.g. ",(0,r.jsx)(n.code,{children:"2643743"})," for London, GB). If provided, this will override the ",(0,r.jsx)(n.code,{children:"city"})," option"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lat"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Latitude for a specific location. If provided alongside ",(0,r.jsx)(n.code,{children:"lon"}),", this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lon"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Longitude for a specific location. If provided alongside ",(0,r.jsx)(n.code,{children:"lat"}),", this will override the ",(0,r.jsx)(n.code,{children:"city"})," and ",(0,r.jsx)(n.code,{children:"cityId"})," options"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of days to display of forecast info to display. Defaults to ",(0,r.jsx)(n.code,{children:"4"}),", max ",(0,r.jsx)(n.code,{children:"16"})," days"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The units to use for displaying data, can be either ",(0,r.jsx)(n.code,{children:"metric"})," or ",(0,r.jsx)(n.code,{children:"imperial"}),". Defaults to ",(0,r.jsx)(n.code,{children:"metric"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the additional details (wind, humidity, pressure, etc) will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-2",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather-forecast\n options:\n city: California\n numDays: 6\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n units: imperial\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-2",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udd34 Premium (free for personal use only)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://openweather.co.uk/privacy-policy",children:"OWM Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"rss-feed",children:"RSS Feed"}),"\n",(0,r.jsx)(n.p,{children:"Display news and updates from any RSS-enabled service."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/rss"})}),"\n",(0,r.jsx)(n.h4,{id:"options-3",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"rssUrl"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL location of your RSS feed"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An API key for ",(0,r.jsx)(n.a,{href:"https://rss2json.com/",children:"rss2json"}),". It's free, and will allow you to make 10,000 requests per day, you can sign up ",(0,r.jsx)(n.a,{href:"https://rss2json.com/sign-up",children:"here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The number of posts to return. If you haven't specified an API key, this will be limited to 10"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"orderBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["How results should be sorted. Can be either ",(0,r.jsx)(n.code,{children:"pubDate"}),", ",(0,r.jsx)(n.code,{children:"author"})," or ",(0,r.jsx)(n.code,{children:"title"}),". Defaults to ",(0,r.jsx)(n.code,{children:"pubDate"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"orderDirection"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Order direction of feed items to return. Can be either ",(0,r.jsx)(n.code,{children:"asc"})," or ",(0,r.jsx)(n.code,{children:"desc"}),". Defaults to ",(0,r.jsx)(n.code,{children:"desc"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"parseLocally"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If true parse the rss feed locally instead of using the rss2json API."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-3",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: rss-feed\n options:\n rssUrl: https://www.schneier.com/blog/atom.xml\n apiKey: xxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-3",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan (up to 10,000 requests / day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://rss2json.com/privacy-policy",children:"Rss2Json Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"image",children:"Image"}),"\n",(0,r.jsx)(n.p,{children:"Displays an image."}),"\n",(0,r.jsxs)(n.p,{children:["This may be useful if you have a service (such as Grafana - ",(0,r.jsx)(n.a,{href:"https://mattionline.de/grafana-api-export-graph-as-png/",children:"see example"}),"), which periodically exports charts or other data as an image."]}),"\n",(0,r.jsxs)(n.p,{children:["You can also store images within Dashy's public directory (using a Docker volume), and reference them directly. E.g. ",(0,r.jsx)(n.code,{children:"-v ./path/to/my-homelab-logo.png:/app/public/logo.png"}),", then in the widget ",(0,r.jsx)(n.code,{children:"imagePath: /logo.png"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Similarly, any web service that serves up widgets as image can be used. E.g. you could show current star chart for a GitHub repo, with: ",(0,r.jsx)(n.code,{children:"imagePath: https://starchart.cc/Lissy93/dashy.svg"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you'd like to embed a live screenshot, of all or just part of a website, then this can be done using ",(0,r.jsx)(n.a,{href:"https://apiflash.com/",children:"API Flash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Or what about showing a photo of the day? Try ",(0,r.jsx)(n.code,{children:"https://source.unsplash.com/random/400x300"})," or ",(0,r.jsx)(n.code,{children:"https://picsum.photos/400/300"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"300",src:"https://pixelflare.cc/alicia/dashy/image"})}),"\n",(0,r.jsx)(n.h4,{id:"options-4",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imagePath"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The path (local or remote) of the image to display"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imageWidth"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify a fixed width for rendered image. Accepts either integer value in ",(0,r.jsx)(n.code,{children:"px"}),", or any string value with units (e.g. ",(0,r.jsx)(n.code,{children:"420"}),", ",(0,r.jsx)(n.code,{children:"100px"}),", ",(0,r.jsx)(n.code,{children:"6.9rem"}),") (defaults to ",(0,r.jsx)(n.code,{children:"auto"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"imageHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify a fixed height for rendered image. Accepts either integer value in ",(0,r.jsx)(n.code,{children:"px"}),", or any string value with units (e.g. ",(0,r.jsx)(n.code,{children:"420"}),", ",(0,r.jsx)(n.code,{children:"100px"}),", ",(0,r.jsx)(n.code,{children:"6.9rem"}),") (defaults to ",(0,r.jsx)(n.code,{children:"auto"}),")"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-4",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: image\n options:\n imagePath: https://i.ibb.co/yhbt6CY/dashy.png\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-4",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:"Unless image fetched from remote source, no external data request is made."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"public-ip",children:"Public IP"}),"\n",(0,r.jsx)(n.p,{children:"Displays your public IP address, ISP and approximate location."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/public-ip"})}),"\n",(0,r.jsx)(n.h4,{id:"options-5",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"provider"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["One of ",(0,r.jsx)(n.code,{children:"ipinfo"})," ",(0,r.jsx)(n.em,{children:"(default)"}),", ",(0,r.jsx)(n.code,{children:"freeipapi"}),", ",(0,r.jsx)(n.code,{children:"ipquery"}),", ",(0,r.jsx)(n.code,{children:"ip-api"}),", or ",(0,r.jsx)(n.code,{children:"ipgeolocation"}),". See the table below"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Required for ",(0,r.jsx)(n.code,{children:"ipgeolocation"}),". Optional for ",(0,r.jsx)(n.code,{children:"ipinfo"})," (a ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/signup",children:"free token"})," raises the rate limit from ~1k/day to ~50k/month)"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLocation"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"})," to hide the flag, ISP name and city/region \u2014 only the IP address is shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-5",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"Default (no options needed)"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Using ",(0,r.jsx)(n.code,{children:"ip-api"})," via the proxy (gets server IP, because of ",(0,r.jsx)(n.code,{children:"useProxy"}),")"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n useProxy: true\n options:\n provider: ip-api\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Using ",(0,r.jsx)(n.code,{children:"ipgeolocation"})," with a key:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-ip\n options:\n provider: ipgeolocation\n apiKey: xxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["Setting ",(0,r.jsx)(n.code,{children:"useProxy: true"})," makes the lookup happen from Dashy's server instead of your browser, so the upstream API returns the public IP of the machine running Dashy, whereas without ",(0,r.jsx)(n.code,{children:"useProxy"})," it will show your public IP."]})}),"\n",(0,r.jsx)(n.h4,{id:"supported-providers",children:"Supported providers"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Provider"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Key"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Proxy"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Notes"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"ipinfo"})," ",(0,r.jsx)(n.em,{children:"(default)"})]}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed / \ud83d\udfe0 Optional"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Service from ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/",children:"ipinfo.io"}),". Keyless gives ~1k/day; a ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/signup",children:"free token"})," raises this to ~50k/month. Sometimes blocked by adblockers."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"freeipapi"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Keyless, 60 req/min via ",(0,r.jsx)(n.a,{href:"https://freeipapi.com/",children:"freeipapi.com"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ipquery"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Modern keyless API from ",(0,r.jsx)(n.a,{href:"https://ipquery.io/",children:"ipquery.io"}),". Includes VPN/Tor/datacenter risk flags in the raw response."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ip-api"})}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["\ud83d\udd34 ",(0,r.jsx)(n.strong,{children:"Required"})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.a,{href:"https://ip-api.com/",children:"ip-api.com"})," is HTTP-only on the free tier, so the proxy is needed to avoid mixed-content errors."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"ipgeolocation"})}),(0,r.jsxs)(n.td,{children:["\ud83d\udd34 ",(0,r.jsx)(n.strong,{children:"Required"})]}),(0,r.jsx)(n.td,{children:"\ud83d\udfe2 Not needed"}),(0,r.jsxs)(n.td,{children:["Get a free key from ",(0,r.jsx)(n.a,{href:"https://ipgeolocation.io/signup.html",children:"ipgeolocation.io"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"info-5",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled (only ",(0,r.jsx)(n.code,{children:"provider: ip-api"})," needs the proxy)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional (only ",(0,r.jsx)(n.code,{children:"ipgeolocation"})," requires a key, and ",(0,r.jsx)(n.code,{children:"ipinfo"})," is optional)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": See the policy for whichever provider you choose: ",(0,r.jsx)(n.a,{href:"https://freeipapi.com/",children:"freeipapi.com"}),", ",(0,r.jsx)(n.a,{href:"https://ipinfo.io/privacy-policy",children:"ipinfo.io"}),", ",(0,r.jsx)(n.a,{href:"https://ipquery.io/",children:"ipquery.io"}),", ",(0,r.jsx)(n.a,{href:"https://ip-api.com/docs/legal",children:"ip-api.com"}),", ",(0,r.jsx)(n.a,{href:"https://ipgeolocation.io/privacy.html",children:"IPGeoLocation"}),"."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ip-blacklist",children:"IP Blacklist"}),"\n",(0,r.jsxs)(n.p,{children:["Notice certain web pages aren't loading? This widget quickly shows which blacklists your IP address (or host, or email) appears on, using data from ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/",children:"blacklistchecker.com"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/ip-blacklist"})}),"\n",(0,r.jsx)(n.h4,{id:"options-6",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"ipAddress"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The IP to check. This can also be a domain/ host name or even an email address. If left blank, Dashy will use your current public IP address."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["You can get your free API key from ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/keys",children:"blacklistchecker.com"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-6",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: blacklist-check\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n ipAddress: 1.1.1.1\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-6",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://blacklistchecker.com/privacy",children:"BlacklistChecker Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"domain-monitor",children:"Domain Monitor"}),"\n",(0,r.jsxs)(n.p,{children:["Keep an eye on the expiry dates of your domain names, using public whois records fetched from ",(0,r.jsx)(n.a,{href:"https://whoapi.com/",children:"whoapi.com"}),". Click the domain name to view additional info, like registrar, name servers and date last updated."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/domain-monitor"})}),"\n",(0,r.jsx)(n.h4,{id:"options-7",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"domain"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The domain to check"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["You can get your free API key from ",(0,r.jsx)(n.a,{href:"https://my.whoapi.com/user/signup",children:"my.whoapi.com"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showFullInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If set to true, the toggle-full-info panel will be open by default"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-7",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: domain-monitor\n options:\n domain: example.com\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n\n - type: domain-monitor\n options:\n domain: example2.com\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-7",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free Plan (10,000 requests)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://whoapi.com/privacy-policy/",children:"WhoAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"crypto-watch-list",children:"Crypto Watch List"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of price changes of your favorite crypto assets. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/",children:"CoinGecko"}),". All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/crypto-prices"})}),"\n",(0,r.jsx)(n.h4,{id:"options-8",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"assets"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An array of cryptocurrencies, coins and tokens. See ",(0,r.jsx)(n.a,{href:"https://api.coingecko.com/api/v3/asset_platforms",children:"list of supported assets"}),". If none are specified, then the top coins by ",(0,r.jsx)(n.code,{children:"sortBy"})," (defaults to market cap) will be returned"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"currency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The fiat currency to display price in, expressed as an ISO-4217 alpha code (see ",(0,r.jsx)(n.a,{href:"https://www.iban.com/currency-codes",children:"list of currencies"}),"). Defaults to ",(0,r.jsx)(n.code,{children:"USD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The method of sorting results. Can be ",(0,r.jsx)(n.code,{children:"marketCap"}),", ",(0,r.jsx)(n.code,{children:"volume"})," or ",(0,r.jsx)(n.code,{children:"alphabetical"}),". Defaults to ",(0,r.jsx)(n.code,{children:"marketCap"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Number of results to return, useful when no assets are specified. Defaults to either ",(0,r.jsx)(n.code,{children:"all"})," or ",(0,r.jsx)(n.code,{children:"100"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-8",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: crypto-watch-list\n options:\n limit: 10\n"})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: crypto-watch-list\n options:\n currency: GBP\n sortBy: marketCap\n assets:\n - bitcoin\n - ethereum\n - monero\n - cosmos\n - polkadot\n - dogecoin\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-8",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"crypto-token-price-history",children:"Crypto Token Price History"}),"\n",(0,r.jsxs)(n.p,{children:["Shows recent price history for a given crypto asset, using price data fetched from ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/",children:"CoinGecko"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/crypto-price-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-9",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"asset"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Name of a crypto asset, coin or token to fetch price data for, see ",(0,r.jsx)(n.a,{href:"https://api.coingecko.com/api/v3/asset_platforms",children:"list of supported assets"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"currency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The fiat currency to display results in, expressed as an ISO-4217 alpha code (see ",(0,r.jsx)(n.a,{href:"https://www.iban.com/currency-codes",children:"list of currencies"}),"). Defaults to ",(0,r.jsx)(n.code,{children:"USD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of days of price history to render. Defaults to ",(0,r.jsx)(n.code,{children:"7"}),", min: ",(0,r.jsx)(n.code,{children:"1"}),", max: ",(0,r.jsx)(n.code,{children:"30"})," days"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value. Defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"})," which inherits dashboard primary color"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-9",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: crypto-price-chart\n options:\n asset: bitcoin\n currency: GBP\n numDays: 7\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-9",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.coingecko.com/en/privacy",children:"CoinGecko Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"wallet-balance",children:"Wallet Balance"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of your crypto balances and see recent transactions. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.blockcypher.com/dev/",children:"BlockCypher"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/crypto-wallet"})}),"\n",(0,r.jsx)(n.h4,{id:"options-10",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"coin"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Symbol of coin or asset, e.g. ",(0,r.jsx)(n.code,{children:"btc"}),", ",(0,r.jsx)(n.code,{children:"eth"})," or ",(0,r.jsx)(n.code,{children:"doge"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"address"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Address to monitor. This is your wallet's ",(0,r.jsx)(n.strong,{children:"public"})," / receiving address"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"network"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To use a different network, other than mainnet. Defaults to ",(0,r.jsx)(n.code,{children:"main"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of transactions to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"}),", set to large number to show all"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-10",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: wallet-balance\n options:\n coin: btc\n address: 3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-10",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.blockcypher.com/privacy.html",children:"BlockCypher Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"code-stats",children:"Code Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Display your coding summary. ",(0,r.jsx)(n.a,{href:"https://codestats.net/",children:"Code::Stats"})," is a free and open source app that aggregates statistics about your programming activity. Dashy supports both the public instance, as well as self-hosted versions."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/code-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-11",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Your CodeStats username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If your self-hosting CodeStats, then supply the host name. By default it will use the public hosted instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monthsToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the number of months to render in the historical data chart. Defaults to ",(0,r.jsx)(n.code,{children:"6"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMeta"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the meta section (username, level, all-time and recent XP)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideHistory"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the historical calendar heat map"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLanguages"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the programming languages pie chart"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMachines"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide the machines percentage chart"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-11",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: code-stats\n options:\n username: alicia\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-11",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://codestats.net/tos#privacy",children:"Code::Stats Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mullvad-status",children:"Mullvad Status"}),"\n",(0,r.jsxs)(n.p,{children:["Shows your Mullvad VPN connection status, as well as server info. Fetched from ",(0,r.jsx)(n.a,{href:"https://mullvad.net/en/check/",children:"am.i.mullvad.net"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/mullvad"})}),"\n",(0,r.jsx)(n.h4,{id:"options-12",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No Options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-12",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mullvad-status\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-12",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://mullvad.net/en/help/privacy-policy/",children:"Mullvad Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"addyio",children:"addy.io"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://addy.io/",children:"addy.io"})," is a free and open source mail forwarding service. Use it to protect your real email address, by using a different alias for each of your online accounts, and have all emails land in your normal inbox(es). Supports custom domains, email replies, PGP-encryption, multiple recipients and more"]}),"\n",(0,r.jsx)(n.p,{children:"This widget display email addresses / aliases from addy.io. Click an email address to copy to clipboard, or use the toggle switch to enable/ disable it. Shows usage stats (bandwidth, used aliases etc), as well as total messages received, blocked and sent. Works with both self-hosted and managed instances of addy.io."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/addy"})}),"\n",(0,r.jsx)(n.h4,{id:"options-13",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your addy.io API Key / Personal Access Token. You can generate this under ",(0,r.jsx)(n.a,{href:"https://app.addy.io/settings/api",children:"API Settings"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If your self-hosting addy.io, then supply the host name. By default it will use the public hosted instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you're using an API version that is not version ",(0,r.jsx)(n.code,{children:"v1"}),", then specify it here"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of emails shown per page. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortBy"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the sort order for email addresses. Defaults to ",(0,r.jsx)(n.code,{children:"updated_at"}),". Can be either: ",(0,r.jsx)(n.code,{children:"local_part"}),", ",(0,r.jsx)(n.code,{children:"domain"}),", ",(0,r.jsx)(n.code,{children:"email"}),", ",(0,r.jsx)(n.code,{children:"emails_forwarded"}),", ",(0,r.jsx)(n.code,{children:"emails_blocked"}),", ",(0,r.jsx)(n.code,{children:"emails_replied"}),", ",(0,r.jsx)(n.code,{children:"emails_sent"}),", ",(0,r.jsx)(n.code,{children:"created_at"}),", ",(0,r.jsx)(n.code,{children:"updated_at"})," or ",(0,r.jsx)(n.code,{children:"deleted_at"}),". Precede with a ",(0,r.jsx)(n.code,{children:"-"})," character to reverse order."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"searchTerm"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A search term to filter results by, will search the email, description and domain"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"disableControls"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Prevent any changes being made to account through the widget. User will not be able to enable or disable aliases through UI when this option is set"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideMeta"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show account meta info (forward/ block count, quota usage etc)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideAliases"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show email address / alias list. Will only show account meta info"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-13",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:' - type: addy\n options:\n apiKey: "xxxxxxxxxxxxxxxxxxxxxxxx\\\n xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\\\n xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"\n limit: 5\n sortBy: created_at\n disableControls: true\n'})}),"\n",(0,r.jsx)(n.h4,{id:"info-13",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free for Self-Hosted / Free Plan available on managed instance or $1/month for premium"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://addy.io/privacy/",children:"addy.io Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"vulnerability-feed",children:"Vulnerability Feed"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of recent security advisories and vulnerabilities, with optional filtering by score, exploits, vendor and product. Using data from ",(0,r.jsx)(n.a,{href:"https://nvd.nist.gov/developers/vulnerabilities",children:"NIST Vulnerability API"}),". All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cve"})}),"\n",(0,r.jsx)(n.h4,{id:"options-14",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cveTag"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVE records that include the provided cveTag. Options are ",(0,r.jsx)(n.strong,{children:"disputed"}),", ",(0,r.jsx)(n.strong,{children:"unsupported-when-assigned"})," or ",(0,r.jsx)(n.strong,{children:"exclusively-hosted-service"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of results to fetch. Can be between ",(0,r.jsx)(n.code,{children:"5"})," and ",(0,r.jsx)(n.code,{children:"30"}),", defaults to ",(0,r.jsx)(n.code,{children:"5"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV2Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv2 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV3Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv3 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cvssV4Severity"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["This parameter returns only the CVEs that match the provided CVSSv4 qualitative severity rating. Options are ",(0,r.jsx)(n.strong,{children:"LOW"}),", ",(0,r.jsx)(n.strong,{children:"MEDIUM"}),", ",(0,r.jsx)(n.strong,{children:"HIGH"})," or ",(0,r.jsx)(n.strong,{children:"CRITICAL"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"keywordSearch"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"This parameter returns only the CVEs where a word or phrase is found in the current description"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Without a key you're limited to 5 requests every 30 seconds. You can get a free API key from ",(0,r.jsx)(n.a,{href:"https://nvd.nist.gov/developers/request-an-api-key",children:"here"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-14",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: cve-vulnerabilities\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: cve-vulnerabilities\n options:\n cveTag: disputed\n cvssV2Severity: CRITICAL\n limit: 30\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-14",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required (free apiKey recommended for multiple widgets)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.cvedetails.com/privacy.php",children:"CVE Details Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"exchange-rates",children:"Exchange Rates"}),"\n",(0,r.jsx)(n.p,{children:"Display current FX rates in your native currency. Hover over a row to view more info, or click to show rates in that currency."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/currencies"})}),"\n",(0,r.jsx)(n.h4,{id:"options-15",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"inputCurrency"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The base currency to show results in. Specified as a 3-letter ISO-4217 code, see ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/docs/supported-currencies",children:"here"})," for the full list of supported currencies, and their symbols"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"outputCurrencies"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["List or currencies to show results for. Specified as a 3-letter ISO-4217 code, see ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/docs/supported-currencies",children:"here"})," for the full list of supported currencies, and their symbols"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/",children:"exchangerate-api.com"}),", usually a 24-digit alpha-numeric string. You can sign up for a free account ",(0,r.jsx)(n.a,{href:"https://app.exchangerate-api.com/sign-up",children:"here"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-15",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: exchange-rates\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxx\n inputCurrency: GBP\n outputCurrencies:\n - USD\n - JPY\n - HKD\n - KPW\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-15",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 100,000 requests/ month)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.exchangerate-api.com/terms",children:"ExchangeRateAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"public-holidays",children:"Public Holidays"}),"\n",(0,r.jsxs)(n.p,{children:["Counting down to the next day off work? This widget displays upcoming public holidays for your country. Data is fetched from ",(0,r.jsx)(n.a,{href:"http://kayaposoft.com/enrico/",children:"Enrico"})]}),"\n",(0,r.jsxs)(n.p,{children:["Note, config for this widget is case-sensitive (see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1268",children:"#1268"}),")"]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/holidays"})}),"\n",(0,r.jsx)(n.h4,{id:"options-16",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"country"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The country to fetch holiday data for, specified as a country code, e.g. ",(0,r.jsx)(n.code,{children:"GB"})," or ",(0,r.jsx)(n.code,{children:"US"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"state"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["restrict a country to a specific state defined by ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/ISO_3166-2",children:"ISO_3166-2"}),", e.g. ",(0,r.jsx)(n.code,{children:"LND"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"holidayType"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The type of holidays to fetch. Can be: ",(0,r.jsx)(n.code,{children:"all"}),", ",(0,r.jsx)(n.code,{children:"public_holiday"}),", ",(0,r.jsx)(n.code,{children:"observance"}),", ",(0,r.jsx)(n.code,{children:"school_holiday"}),", ",(0,r.jsx)(n.code,{children:"other_day"})," or ",(0,r.jsx)(n.code,{children:"extra_working_day"}),". Defaults to ",(0,r.jsx)(n.code,{children:"public_holiday"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monthsToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of months in advance to show. Min: ",(0,r.jsx)(n.code,{children:"1"}),", max: ",(0,r.jsx)(n.code,{children:"24"}),". Defaults to ",(0,r.jsx)(n.code,{children:"12"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The language in which the events should be. Usually local languages and english are available. Default to first available in the country. e.g. ",(0,r.jsx)(n.code,{children:"en"})," or ",(0,r.jsx)(n.code,{children:"fr"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-16",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: public-holidays\n options:\n country: GB\n state: LND\n holidayType: all\n monthsToShow: 12\n lang: en\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-16",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/jurajmajer/enrico",children:"jurajmajer/enrico"}),") or Managed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"covid-19-status",children:"Covid-19 Status"}),"\n",(0,r.jsxs)(n.p,{children:["Keep track of the current COVID-19 status. Optionally also show cases by country, and a time-series chart. Uses live data from various sources, computed by ",(0,r.jsx)(n.a,{href:"https://disease.sh/",children:"disease.sh"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/covid"})}),"\n",(0,r.jsx)(n.h4,{id:"options-17",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showChart"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Also display a time-series chart showing number of recent cases"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showCountries"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Also display a list of cases per country"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"numDays"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Specify number of days worth of history to render on the chart"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"countries"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string[]"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["An array of countries to display, specified by their ",(0,r.jsx)(n.a,{href:"https://www.iso.org/obp/ui",children:"ISO-3 codes"}),". Leave blank to show all, sorted by most cases. ",(0,r.jsx)(n.code,{children:"showCountries"})," must be set to ",(0,r.jsx)(n.code,{children:"true"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If showing all countries, set a limit for number of results to return. Defaults to ",(0,r.jsx)(n.code,{children:"10"}),", no maximum"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-17",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: covid-stats\n"})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: covid-stats\n options:\n showChart: true\n showCountries: true\n countries:\n - GBR\n - USA\n - IND\n - RUS\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-17",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/disease-sh/api",children:"disease-sh/api"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Conditions"}),": ",(0,r.jsx)(n.a,{href:"https://github.com/disease-sh/api/blob/master/TERMS",children:"Terms of Use"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"sports-scores",children:"Sports Scores"}),"\n",(0,r.jsxs)(n.p,{children:["Show recent scores and upcoming matches from your favorite sports team. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/",children:"TheSportsDB.com"}),". From the UI, you can click any other team to view their scores and upcoming games, or click a league name to see all teams."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/sports"})}),"\n",(0,r.jsx)(n.h4,{id:"options-18",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"teamId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The ID of a team to fetch scores from. You can search for your team on the ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/teams_main.php",children:"Teams Page"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"leagueId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Alternatively, provide a league ID to fetch all games from. You can find the ID on the ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/Sport/Leagues",children:"Leagues Page"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"pastOrFuture"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"past"})," to show scores for recent games, or ",(0,r.jsx)(n.code,{children:"future"})," to show upcoming games. Defaults to ",(0,r.jsx)(n.code,{children:"past"}),". You can change this within the UI"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Optionally specify your API key, which you can sign up for at ",(0,r.jsx)(n.a,{href:"https://www.thesportsdb.com/",children:"TheSportsDB.com"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["To limit output to a certain number of matches, defaults to ",(0,r.jsx)(n.code,{children:"15"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideImage"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"})," to not render the team / match banner image, defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-18",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: sports-scores\n options:\n teamId: 133636\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-18",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 30 requests / minute, limited endpoints)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"news-headlines",children:"News Headlines"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the latest news, click to read full article. Date is fetched from various news sources using ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/en",children:"Currents API"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/news"})}),"\n",(0,r.jsx)(n.h4,{id:"options-19",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your API key for CurrentsAPI. This is free, and you can ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/en/register",children:"get one here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"country"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Fetch news only from a certain country or region. Specified as a country code, e.g. ",(0,r.jsx)(n.code,{children:"GB"})," or ",(0,r.jsx)(n.code,{children:"US"}),". See ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/regions",children:"here"})," for a list of supported regions"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"category"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Only return news from within a given category, e.g. ",(0,r.jsx)(n.code,{children:"sports"}),", ",(0,r.jsx)(n.code,{children:"programming"}),", ",(0,r.jsx)(n.code,{children:"world"}),", ",(0,r.jsx)(n.code,{children:"science"}),". The ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/categories",children:"following categories"})," are supported"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the language for returned articles as a 2-digit ISO code (limited article support). The ",(0,r.jsx)(n.a,{href:"https://api.currentsapi.services/v1/available/languages",children:"following languages"})," are supported, defaults to ",(0,r.jsx)(n.code,{children:"en"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results. Can be between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"200"}),", defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"keywords"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Only return articles that contain an exact match within their title or description"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideImages"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", then article image thumbnails will not be displayed"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-19",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: news-headlines\n options:\n apiKey: xxxxxxx\n category: world\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-19",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 600 requests / day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://currentsapi.services/privacy",children:"CurrentsAPI Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"tfl-status",children:"TFL Status"}),"\n",(0,r.jsx)(n.p,{children:"Shows real-time tube status of the London Underground. All fields are optional."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/tfl"})}),"\n",(0,r.jsx)(n.h4,{id:"options-20",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showAll"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default, details for lines with a Good Service are not visible, but you can click More Details to see all. Setting this option to ",(0,r.jsx)(n.code,{children:"true"})," will show all lines on initial page load"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sortAlphabetically"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default lines are sorted by current status, set this option to ",(0,r.jsx)(n.code,{children:"true"})," to instead sort them alphabetically"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"linesToShow"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"By default all lines are shown. If you're only interested in the status of a few lines, then pass in an array of lines to show, specified by name"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-20",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: tfl-status\n"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: tfl-status\n options:\n showAll: true\n sortAlphabetically: true\n linesToShow:\n - District\n - Jubilee\n - Central\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-20",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://tfl.gov.uk/corporate/privacy-and-cookies/",children:"TFL Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"stock-price-history",children:"Stock Price History"}),"\n",(0,r.jsx)(n.p,{children:"Shows recent price history for a given publicly-traded stock or share"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/stocks"})}),"\n",(0,r.jsx)(n.h4,{id:"options-21",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/",children:"Alpha Vantage"}),", you can get a free API key ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/support/#api-key",children:"here"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"stock"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The stock symbol for the asset to fetch data for"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"priceTime"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The time to fetch price for. Can be ",(0,r.jsx)(n.code,{children:"high"}),", ",(0,r.jsx)(n.code,{children:"low"}),", ",(0,r.jsx)(n.code,{children:"open"})," or ",(0,r.jsx)(n.code,{children:"close"}),". Defaults to ",(0,r.jsx)(n.code,{children:"high"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value. Defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"})," which inherits dashboard primary color"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-21",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stock-price-chart\n options:\n stock: NET\n apiKey: PGUWSWD6CZTXMT8N\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-21",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 500 requests/day)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.alphavantage.co/privacy/",children:"AlphaVantage Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"eth-gas-prices",children:"ETH Gas Prices"}),"\n",(0,r.jsxs)(n.p,{children:["Renders the current Gas cost of transactions on the Ethereum network (in both GWEI and USD), along with recent historical prices. Useful for spotting a good time to transact. Uses data from ",(0,r.jsx)(n.a,{href:"https://ethgas.watch/",children:"ethgas.watch"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/eth-gas"})}),"\n",(0,r.jsx)(n.h4,{id:"options-22",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-22",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: eth-gas-prices\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-22",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/wslyvh/ethgaswatch",children:"wslyvh/ethgaswatch"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"joke",children:"Joke"}),"\n",(0,r.jsxs)(n.p,{children:["Renders a programming or generic joke. Data is fetched from the ",(0,r.jsx)(n.a,{href:"https://github.com/Sv443/JokeAPI",children:"JokesAPI"})," by @Sv443. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/joke"})}),"\n",(0,r.jsx)(n.h4,{id:"options-23",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"category"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set the category of jokes to return. Use a string to specify a single category, or an array to pass in multiple options. Available options are: ",(0,r.jsx)(n.code,{children:"all"}),", ",(0,r.jsx)(n.code,{children:"programming"}),", ",(0,r.jsx)(n.code,{children:"pun"}),", ",(0,r.jsx)(n.code,{children:"dark"}),", ",(0,r.jsx)(n.code,{children:"spooky"}),", ",(0,r.jsx)(n.code,{children:"christmas"})," and ",(0,r.jsx)(n.code,{children:"misc"}),". An up-to-date list of supported categories can be found ",(0,r.jsx)(n.a,{href:"https://v2.jokeapi.dev/categories",children:"here"}),". Defaults to ",(0,r.jsx)(n.code,{children:"all"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"safeMode"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set to ",(0,r.jsx)(n.code,{children:"true"}),", to prevent the fetching of any NSFW jokes. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"language"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the language for returned jokes. The following languages are supported: ",(0,r.jsx)(n.code,{children:"en"}),", ",(0,r.jsx)(n.code,{children:"cs"}),", ",(0,r.jsx)(n.code,{children:"de"}),", ",(0,r.jsx)(n.code,{children:"es"}),", ",(0,r.jsx)(n.code,{children:"fr"})," and ",(0,r.jsx)(n.code,{children:"pt"}),", and an up-to-date list of supported languages can be found ",(0,r.jsx)(n.a,{href:"https://v2.jokeapi.dev/languages",children:"here"}),". By default, your system language will be used, if it's supported, otherwise English"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-23",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: joke\n options:\n safeMode: true\n language: en\n category: Programming\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-23",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/Sv443/JokeAPI",children:"Sv443/JokeAPI"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://sv443.net/privacypolicy/en",children:"SV443's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"chuck-norris-quotes",children:"Chuck Norris quotes"}),"\n",(0,r.jsxs)(n.p,{children:["Renders a Chuck Norris quote. Data is fetched from the ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/",children:"ChuckNorrisAPI"})," by @matchilling. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://tbd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-24",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"categories"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Set the category of jokes to return. Use a string to specify a single category, or an array to pass in multiple options. Available options are: ",(0,r.jsx)(n.code,{children:"animal"}),",",(0,r.jsx)(n.code,{children:"career"}),",",(0,r.jsx)(n.code,{children:"celebrity"}),",",(0,r.jsx)(n.code,{children:"dev"}),",",(0,r.jsx)(n.code,{children:"explicit"}),",",(0,r.jsx)(n.code,{children:"fashion"}),",",(0,r.jsx)(n.code,{children:"food"}),",",(0,r.jsx)(n.code,{children:"history"}),",",(0,r.jsx)(n.code,{children:"money"}),",",(0,r.jsx)(n.code,{children:"movie"}),",",(0,r.jsx)(n.code,{children:"music"}),",",(0,r.jsx)(n.code,{children:"political"}),",",(0,r.jsx)(n.code,{children:"religion"}),",",(0,r.jsx)(n.code,{children:"science"}),",",(0,r.jsx)(n.code,{children:"sport"})," and ",(0,r.jsx)(n.code,{children:"travel"}),". An up-to-date list of supported categories can be found ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/jokes/categories",children:"here"}),". Defaults to not explicitely set and therefore any of the categories can come up."]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-24",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: chucknorris\n options:\n categories: history,sport\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-24",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://api.chucknorris.io/privacy",children:"matchilling's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"xkcd-comics",children:"XKCD Comics"}),"\n",(0,r.jsxs)(n.p,{children:["Have a laugh with the daily comic from ",(0,r.jsx)(n.a,{href:"https://xkcd.com/",children:"XKCD"}),". A classic webcomic website covering everything from Linux, math, romance, science and language. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/xkcd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-25",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"comic"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string / number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Choose which comic to display. Set to either ",(0,r.jsx)(n.code,{children:"random"}),", ",(0,r.jsx)(n.code,{children:"latest"})," or the series number of a specific comic, like ",(0,r.jsx)(n.code,{children:"627"}),". Defaults to ",(0,r.jsx)(n.code,{children:"latest"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-25",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: xkcd-comic\n options:\n comic: latest\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-25",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"flight-data",children:"Flight Data"}),"\n",(0,r.jsxs)(n.p,{children:["Displays airport departure and arrival flights, using data from ",(0,r.jsx)(n.a,{href:"https://www.aerodatabox.com/",children:"AeroDataBox"}),". Useful if you live near an airport and often wonder where the flight overhead is going to. Hover over a row for more flight data."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/flights"})}),"\n",(0,r.jsx)(n.h4,{id:"options-26",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"airport"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The airport to show flight data from. Should be specified as a 4-character ICAO-code, a full list of which can be found ",(0,r.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/ICAO_airport_code",children:"here"})," (example: ",(0,r.jsx)(n.code,{children:"KBJC"})," or ",(0,r.jsx)(n.code,{children:"EGKK"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A valid ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/",children:"RapidAPI"})," Key, with ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/aerodatabox/api/aerodatabox/",children:"AeroDataBox"})," enabled (check in your ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/developer/billing/subscriptions-and-usage",children:"Subscription Dashboard"}),"). This API is free to sign up for and use"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"For busy airports, you may wish to limit the number of results visible"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"direction"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["By default, both departure and arrival flights will be fetched, if you would like to only show flights in one direction, set this to wither ",(0,r.jsx)(n.code,{children:"departure"})," or ",(0,r.jsx)(n.code,{children:"arrival"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-26",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: flight-data\n options:\n airport: EGLC\n apiKey: XXXXX\n limit: 12\n direction: all\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-26",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 150 requests / month)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance Only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.aerodatabox.com/#h.p_cxtiyzwf_wqd",children:"AeroDataBox"})," and ",(0,r.jsx)(n.a,{href:"https://rapidapi.com/privacy/",children:"RapidAPI Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"astronomy-picture-of-the-day",children:"Astronomy Picture of the Day"}),"\n",(0,r.jsxs)(n.p,{children:["Show the NASA Astronomy Picture of the Day. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://apod.nasa.gov/apod/",children:"APOD"})," using ",(0,r.jsx)(n.a,{href:"https://github.com/lissy93/go-apod",children:"@Lissy93/go-apod"})," / hosted at ",(0,r.jsx)(n.a,{href:"https://apod.as93.net/",children:"apod.as93.net"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://storage.googleapis.com/as93-screenshots/dashy/apod.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-27",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-27",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: apod\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-27",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/lissy93/go-apod",children:"@Lissy93/go-apod"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.nasa.gov/about/highlights/HP_Privacy.html",children:"NASA's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"github-trending",children:"GitHub Trending"}),"\n",(0,r.jsxs)(n.p,{children:["Displays currently trending projects on GitHub. Optionally specify a language and time-frame. Data is fetched from ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/gh-trending-no-cors",children:"Lissy93/gh-trending-no-cors"})," using the GitHub API. All fields are optional."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/github-trending"})}),"\n",(0,r.jsx)(n.h4,{id:"options-28",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"lang"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["A programming language to fetch trending repos from that category. E.g. ",(0,r.jsx)(n.code,{children:"javascript"})," or ",(0,r.jsx)(n.code,{children:"go"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"since"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The timeframe to use when calculating trends. Can be either ",(0,r.jsx)(n.code,{children:"daily"}),", ",(0,r.jsx)(n.code,{children:"weekly"})," or ",(0,r.jsx)(n.code,{children:"monthly"}),". Defaults to ",(0,r.jsx)(n.code,{children:"daily"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Optionally limit the number of results. Max ",(0,r.jsx)(n.code,{children:"25"}),", default is ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-28",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: github-trending-repos\n options:\n limit: 8\n since: weekly\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-28",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/gh-trending-no-cors",children:"Lissy93/gh-trending-no-cors"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"github-profile-stats",children:"GitHub Profile Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Display stats from your GitHub profile, using embedded cards from ",(0,r.jsx)(n.a,{href:"https://github.com/anuraghazra/github-readme-stats",children:"anuraghazra/github-readme-stats"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/github-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-29",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The GitHub username to fetch info for. E.g. ",(0,r.jsx)(n.code,{children:"lissy93"}),". (Not required if ",(0,r.jsx)(n.code,{children:"hideProfileCard"})," and ",(0,r.jsx)(n.code,{children:"hideLanguagesCard"})," are both set to ",(0,r.jsx)(n.code,{children:"true"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideProfileCard"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the users profile card will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideLanguagesCard"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", the users top languages card will not be shown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"repos"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you'd like to also display stats for some GitHub repositories, then add an array or repo names here. Specified as ",(0,r.jsx)(n.code,{children:"[username]/[repo-name]"}),", e.g. ",(0,r.jsx)(n.code,{children:"lissy93/dashy"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-29",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: github-profile-stats\n options:\n username: Lissy93\n hideLanguagesCard: true\n repos:\n - lissy93/dashy\n - lissy93/personal-security-checklist\n - lissy93/twitter-sentiment-visualisation\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-29",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/anuraghazra/github-readme-stats",children:"anuraghazra/github-readme-stats"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.github.com/en/github/site-policy/github-privacy-statement",children:"GitHub's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"healthchecks-status",children:"HealthChecks Status"}),"\n",(0,r.jsx)(n.p,{children:"Display status of one or more HealthChecks project(s). Works with healthchecks.io and your selfhosted instance."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/healthchecks"})}),"\n",(0,r.jsx)(n.h4,{id:"options-30",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["The base url of your instance, default is ",(0,r.jsx)(n.code,{children:"https://healthchecks.io"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," or ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"One or more API keys for your healthcheck projects. (Read-only works fine)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: HealthChecks\n options:\n host: https://healthcheck.your-domain.de\n apiKey: \n - abcdefg...\n - zxywvu...\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-30",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Paid / Self-hosted"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"healthchecks/healthchecks"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://healthchecks.io/privacy/",children:"Healthchecks.io Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"hackernews-trending",children:"Hackernews Trending"}),"\n",(0,r.jsx)(n.p,{children:"Display new and trending Posts from Hackernews"}),"\n",(0,r.jsx)(n.h4,{id:"options-31",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"stories"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["HN Stories to display defaults to ",(0,r.jsx)(n.code,{children:"topstories"}),". Options are: ",(0,r.jsx)(n.code,{children:"beststories"}),", ",(0,r.jsx)(n.code,{children:"topstories"})," or ",(0,r.jsx)(n.code,{children:"newstories"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"int"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The size of the list of Posts to show."})]})]})]}),"\n",(0,r.jsx)(n.h5,{id:"example-30",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: hackernews-trending\n options:\n stories: newstories\n limit: 10\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mvg-departure",children:"MVG Departure"}),"\n",(0,r.jsx)(n.p,{children:"Display departure time of a MVG (M\xfcnchner Verkehrs Gesellschaft) station."}),"\n",(0,r.jsxs)(n.p,{children:["From ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/impressum.html",children:"https://www.mvg.de/impressum.html"}),":"]}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:"[...] Die Verarbeitung unserer Inhalte oder Daten durch Dritte erfordert unsere ausdr\xfcckliche Zustimmung. F\xfcr private, nicht-kommerzielle Zwecke, wird eine gem\xe4\xdfigte Nutzung ohne unsere ausdr\xfcckliche Zustimmung geduldet. Jegliche Form von Data-Mining stellt keine gem\xe4\xdfigte Nutzung dar.[...]"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"In other words: Private, noncomercial, moderate use of the API is tolerated. They don\u2019t consider data mining as moderate use. (This is not a legal advice)"}),"\n",(0,r.jsx)(n.h4,{id:"options-32",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"location"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The name of the location (exact) or the location id, startin with ",(0,r.jsx)(n.code,{children:"de:09162:"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"integer"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Limit number of entries, defaults to 10."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A custom title to be displayed."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"header"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"bool"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Shall the title be shown?"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.line"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for given line(s)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.product"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific product (TRAM, UBAHN, SBAHN, BUS)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific destination(s)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mvg\n options:\n location: Marienplatz\n limit: 5\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-31",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Private use only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mvg.de",children:"MVG"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"mvg-connection",children:"MVG Connection"}),"\n",(0,r.jsx)(n.p,{children:"Display the next connection for two addresses/coordinates, stations or POI within Munich using MVG MVG (M\xfcnchner Verkehrs Gesellschaft)."}),"\n",(0,r.jsxs)(n.p,{children:["From ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/impressum.html",children:"https://www.mvg.de/impressum.html"}),":"]}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:"[...] Die Verarbeitung unserer Inhalte oder Daten durch Dritte erfordert unsere ausdr\xfcckliche Zustimmung. F\xfcr private, nicht-kommerzielle Zwecke, wird eine gem\xe4\xdfigte Nutzung ohne unsere ausdr\xfcckliche Zustimmung geduldet. Jegliche Form von Data-Mining stellt keine gem\xe4\xdfigte Nutzung dar.[...]"}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"In other words: Private, noncomercial, moderate use of the API is tolerated. They don\u2019t consider data mining as moderate use. (This is not a legal advice)"}),"\n",(0,r.jsx)(n.h4,{id:"options-33",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"origin"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Origin of the connection."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Destination of the connection."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A custom title to be displayed."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"header"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"bool"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Shall the title be shown?"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.line"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for given line(s)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.product"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific product (TRAM, UBAHN, SBAHN, BUS)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"filters.destination"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string/object"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter results for specific destination(s)"})]})]})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: mvg-connection\n options:\n from: Marienplatz\n from: Dachauer Stra\xdfe 123\n header: true\n filters:\n product: [UBAHN]\n line: [U1,U2,U4,U5]\n\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-32",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free / Private use only"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mvg.de",children:"MVG"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"custom-list",children:"Custom List"}),"\n",(0,r.jsx)(n.p,{children:"Renders custom schema-compliant JOSN in a list."}),"\n",(0,r.jsx)(n.h4,{id:"options-34",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"text"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A string containing the url of a json file."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"text"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A title for the widget. Can be helpful if stacking multiple lists in the same section."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"daysForNew"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsx)(n.td,{children:"Used to highlight new items."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"json-schema",children:"Json Schema"}),"\n",(0,r.jsx)(n.p,{children:"The input file should comply with the following schema:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-json",children:'{\n "$schema": "http://json-schema.org/draft-04/schema#",\n "type": "array",\n "items": [\n {\n "type": "object",\n "properties": {\n "link": {\n "type": "object",\n "properties": {\n "text": {\n "type": "string"\n },\n "url": {\n "type": "string"\n },\n "title": {\n "type": "string"\n }\n },\n "required": [\n "text",\n "url",\n "title"\n ]\n },\n "value": {\n "type": "object",\n "properties": {\n "text": {\n "type": "string"\n },\n "title": {\n "type": "string"\n }\n },\n "required": [\n "text",\n "title"\n ]\n },\n "date": {\n "type": "string"\n }\n },\n "required": [\n "link",\n "value",\n "date"\n ]\n }\n ]\n}\n'})}),"\n",(0,r.jsx)(n.p,{children:"Example: This json data was generated by a data worflow that gets the new releases of a few projects from GitHub. The system used to build the data workflow is n8n."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-json",children:'[\n {\n "link": {\n "text": "jellyfin/jellyfin",\n "url": "https://github.com/jellyfin/jellyfin/releases/tag/v10.10.7",\n "title": ""\n },\n "value": {\n "text": "v10.10.7",\n "title": "2025-04-05"\n },\n "date": "2025-04-05T19:14:59Z"\n },\n {\n "link": {\n "text": "jellyfin/jellyfin-web",\n "url": "https://github.com/jellyfin/jellyfin-web/releases/tag/v10.10.7",\n "title": ""\n },\n "value": {\n "text": "v10.10.7",\n "title": "2025-04-05"\n },\n "date": "2025-04-05T19:15:00Z"\n },\n {\n "link": {\n "text": "lissy93/dashy",\n "url": "https://github.com/Lissy93/dashy/releases/tag/3.1.1",\n "title": ""\n },\n "value": {\n "text": "3.1.1",\n "title": "2024-05-30"\n },\n "date": "2024-05-30T17:20:53Z"\n },\n {\n "link": {\n "text": "VSCodium/vscodium",\n "url": "https://github.com/VSCodium/vscodium/releases/tag/1.102.14746",\n "title": ""\n },\n "value": {\n "text": "1.102.14746",\n "title": "2025-07-16"\n },\n "date": "2025-07-16T18:27:49Z"\n }\n]\n'})}),"\n",(0,r.jsx)(n.h4,{id:"notes",children:"Notes"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"This widget is designed to render data generated by another system that complies with the schema. The example JSON data above was generated using a n8n workflow, and other ETL or workflow systems can generate similar results."}),"\n",(0,r.jsx)(n.li,{children:"To avoid requests to a different system in each refresh, you can save the input files locally in the user-data folder inside your Dashy installation."}),"\n",(0,r.jsxs)(n.li,{children:["To use json files from a different domain, remember to add ",(0,r.jsx)(n.code,{children:"useProxy: true"})," to the widget configuration. I have not tested this use case because I save all my input data locally on the Dashy server. Please open a ticket if you have an issue in this use case."]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-31",children:"Example"}),"\n",(0,r.jsxs)(n.p,{children:["This widget renders a json file that from a ",(0,r.jsx)(n.code,{children:"json-data"})," directory inside the ",(0,r.jsx)(n.code,{children:"user-data"})," directory on the Dashy server."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: custom-list\n options:\n url: /json-data/github-releases.json\n title: 'Github Releases'\n daysForNew: 5\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-33",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Not needed for files hosted inside the ",(0,r.jsx)(n.code,{children:"user-data"})," directory. Use ",(0,r.jsx)(n.code,{children:"useProxy: true"})," to bypass CORS restrictions when using data from a different server."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": user defined"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": depends on the user defined host."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"custom-search",children:"Custom search"}),"\n",(0,r.jsx)(n.p,{children:"Allows web search using multiple user-defined search engines and other websites."}),"\n",(0,r.jsx)(n.h4,{id:"options-35",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"engines"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"required"}),(0,r.jsxs)(n.td,{children:["An array of search engine objects. Each search engine object should have two required properties: ",(0,r.jsx)(n.strong,{children:"title"})," and ",(0,r.jsx)(n.strong,{children:"url"}),". See the example below."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"placeholder"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"Placeholder text in the search box."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"openingMethod"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["Open search in one of ",(0,r.jsx)(n.code,{children:"newtab"}),", ",(0,r.jsx)(n.code,{children:"sametab"})," or ",(0,r.jsx)(n.code,{children:"workspace"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"notes-1",children:"Notes"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["The first search engine in the engines array will be treated as the default search engine, and used when the user presses ",(0,r.jsx)(n.code,{children:"Enter"})," in the search box."]}),"\n",(0,r.jsx)(n.li,{children:"Popup blockers can interfere with opening a new search window."}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"example-32",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"This widget allows searching multiple search engines from dashy."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: 'custom-search'\n options:\n placeholder: Search for something using the buttons below\n engines:\n - title: SearXNG\n url: https://searx.lan/?q=\n - title: Quant\n url: https://www.qwant.com/?q=\n - title: Bing Web\n url: http://www.bing.com/search?q=\n - title: Bing Images\n url: http://www.bing.com/images/search?q=\n - title: Bing Maps\n url: http://www.bing.com/maps/search?q=\n - title: Yandex\n url: https://www.yandex.com/search/?text=\n - title: Passmark\n url: https://www.passmark.com/search/zoomsearch.php?zoom_query=\n - title: IMDB\n url: http://www.imdb.com/find?q=\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-34",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Not needed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": user defined"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": depends on the user defined search engines."]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"rescuetime-overview",children:"RescueTime Overview"}),"\n",(0,r.jsx)(n.p,{children:"Show an overview of how you have spent your time for the current day."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/rescue-time"})}),"\n",(0,r.jsx)(n.h4,{id:"options-36",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"required"}),(0,r.jsx)(n.td,{children:"The API-Key generated in the RescueTime UI."})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-33",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: rescue-time\n useProxy: true\n options:\n apiKey: abcdefghijkl_mnop\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-35",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Depends on user subscription"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://www.rescuetime.com",children:"RescueTime"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.rescuetime.com/privacy",children:"RescueTime Privacy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"minecraft-server",children:"Minecraft Server"}),"\n",(0,r.jsx)(n.p,{children:"Show minecraft server status"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://i.ibb.co/hcmd4Wf/minecraft-widget.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-37",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display title for server uses server address if not set."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"server"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Server hostname or ip(:port) will use srv records."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"bedrock"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If server is a bedrock edition server. (default false)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showMods"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display mod list if available"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showPlayers"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display player list if available"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showPlugins"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Display plugin list if available"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-34",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: minecraft-status\n options:\n title: Venity Network\n server: play.venitymc.com\n bedrock: true\n showMods: true\n showPlayers: true\n showPlugins: true\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-36",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://mcsrvstat.us/",children:"Minecraft Server Status"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://mcsrvstat.us/faq",children:"Minecraft Server Status FAQ"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"self-hosted-services-widgets",children:"Self-Hosted Services Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"system-info",children:"System Info"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.mvg.de/datenschutz-mvg.html",children:"MVG Datenschutz"})]}),"\nDisplays info about the server which Dashy is hosted on. Includes user + host, operating system, uptime and basic memory & load data."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/system-info"})}),"\n",(0,r.jsx)(n.h4,{id:"options-38",children:"Options"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"No config options."})}),"\n",(0,r.jsx)(n.h4,{id:"example-35",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: system-info\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-37",children:"Info"}),"\n",(0,r.jsx)(n.p,{children:"No external data requests made"}),"\n",(0,r.jsxs)(n.p,{children:["Note that this widget is not available if you are running Dashy in a container or VM. Instead you can use the ",(0,r.jsx)(n.a,{href:"#system-resource-monitoring",children:"System Monitoring"})," widgets to display stats from the host system instead."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cron-monitoring-health-checks",children:"Cron Monitoring (Health Checks)"}),"\n",(0,r.jsxs)(n.p,{children:["Cron job monitoring using ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"Health Checks"}),". Both managed and self-hosted instances are supported."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cron-monitor"})}),"\n",(0,r.jsx)(n.h4,{id:"options-39",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A read-only API key for the project to monitor. You can generate this by selecting a Project --\x3e Settings --\x3e API Access. Note that you must generate a separate key for each project"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you're self-hosting, or using any instance other than the official (healthchecks.io), you will need to specify the host address. E.g. ",(0,r.jsx)(n.code,{children:"https://healthchecks.example.com"})," or ",(0,r.jsx)(n.code,{children:"http://cron-monitoing.local"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-36",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: health-checks\n options:\n apiKey: XXXXXXXXX\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-38",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe0 Free plan (up to 20 services, or self-host for unlimited)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Managed Instance or Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/healthchecks/healthchecks",children:"GitHub - HealthChecks"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://healthchecks.io/privacy/",children:"Health-Checks Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-history-netdata",children:"CPU History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent CPU usage history from NetData."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/cpu-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-40",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-37",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-cpu-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-39",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"memory-history-netdata",children:"Memory History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent system RAM usage from NetData, and show as a breakdown of different categories."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://i.ibb.co/2dsSWnk/nd-memory-history.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-41",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-38",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-ram-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-40",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"load-history-netdata",children:"Load History (NetData)"}),"\n",(0,r.jsx)(n.p,{children:"Pull recent load usage in 1, 5 and 15 minute intervals, from NetData."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/load-history"})}),"\n",(0,r.jsx)(n.h4,{id:"options-42",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your NetData instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The height of rendered chart in px. Defaults to ",(0,r.jsx)(n.code,{children:"300"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColor"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"chartColors"})})]}),(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.code,{children:"string"})," / ",(0,r.jsx)(n.code,{children:"array"})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Color of the chart value(s) as hex codes. ",(0,r.jsx)(n.code,{children:"chartColor"})," is a single value (defaults to ",(0,r.jsx)(n.code,{children:"--widget-text-color"}),"), whereas ",(0,r.jsx)(n.code,{children:"chartColors"})," is an array of colors"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-39",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nd-load-history\n options:\n host: http://192.168.1.1:19999\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-41",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/netdata/netdata",children:"GitHub - NetData"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.netdata.cloud/data-privacy/",children:"NetData Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-stats",children:"Pi-Hole Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the number of queries blocked by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/pi-hole"})}),"\n",(0,r.jsx)(n.h4,{id:"options-43",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStatus"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideChart"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideInfo"})})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide any of the three parts of the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-40",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-stats\n options:\n hostname: http://192.168.130.1\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["To avoid storing secrets in plaintext, both ",(0,r.jsx)(n.code,{children:"hostname"})," and ",(0,r.jsx)(n.code,{children:"apiKey"})," can be supplied via environment variables. See ",(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"}),"."]})}),"\n",(0,r.jsx)(n.h4,{id:"info-42",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-Hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-stats-v6",children:"Pi-Hole Stats v6"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the number of queries blocked by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/zftCLJN/pi-hole-stats.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-44",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsxs)(n.td,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStatus"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideChart"})})," / ",(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideInfo"})})]}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Optionally hide any of the three parts of the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-41",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-stats-v6\n options:\n hostname: http://192.168.130.1\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.admonition,{type:"tip",children:(0,r.jsxs)(n.p,{children:["To avoid storing secrets in plaintext, both ",(0,r.jsx)(n.code,{children:"hostname"})," and ",(0,r.jsx)(n.code,{children:"apiKey"})," can be supplied via environment variables. See ",(0,r.jsx)(n.a,{href:"#handling-secrets",children:"Handling Secrets"}),"."]})}),"\n",(0,r.jsx)(n.h4,{id:"info-43",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-Hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-queries",children:"Pi-Hole Queries"}),"\n",(0,r.jsxs)(n.p,{children:["Shows top queries that were blocked and allowed by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/pi-hole-queries"})}),"\n",(0,r.jsx)(n.h4,{id:"options-45",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of queries to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-42",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-top-queries\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-44",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-queries-v6",children:"Pi-Hole Queries v6"}),"\n",(0,r.jsxs)(n.p,{children:["Shows top queries that were blocked and allowed by ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/pXR0bdQ/pi-hole-queries.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-46",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"count"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The number of queries to display. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-43",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-top-queries-v6\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-45",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-recent-traffic",children:"Pi-Hole Recent Traffic"}),"\n",(0,r.jsxs)(n.p,{children:["Shows number of recent traffic, using allowed and blocked queries from ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/pi-hole-trafic"})}),"\n",(0,r.jsx)(n.h4,{id:"options-47",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password. It is ",(0,r.jsx)(n.strong,{children:"NOT"})," your pi-hole admin interface or server password. It can be found in ",(0,r.jsx)(n.code,{children:"/etc/pihole/setupVars.conf"}),", and is a 64-character located on the line that starts with ",(0,r.jsx)(n.code,{children:"WEBPASSWORD"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-44",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-traffic\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-46",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"pi-hole-recent-traffic-v6",children:"Pi-Hole Recent Traffic v6"}),"\n",(0,r.jsxs)(n.p,{children:["Shows number of recent traffic, using allowed and blocked queries from ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/",children:"Pi-Hole"}),". Use this version of the widget if you have a v6+ Pi-Hole instance."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://i.ibb.co/7kdxxwx/pi-hole-recent-queries.png"})}),"\n",(0,r.jsx)(n.h4,{id:"options-48",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Pi-Hole instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Your Pi-Hole web password or application password. It ",(0,r.jsx)(n.strong,{children:"IS"})," your Pi-Hole admin interface password ",(0,r.jsx)(n.strong,{children:"UNLESS"})," you have 2FA turned on (in contrast to the old widget). If you have 2FA turned on you will need to create an application password. Refer to Pi-Hole documentation for how to create an application password."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-45",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: pi-hole-traffic-v6\n options:\n hostname: https://pi-hole.local\n apiKey: xxxxxxxxxxxxxxxxxxxxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-47",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udd34 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/pi-hole/pi-hole",children:"GitHub - Pi-hole"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://pi-hole.net/privacy/",children:"Pi-Hole Privacy Guide"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"stat-ping-statuses",children:"Stat Ping Statuses"}),"\n",(0,r.jsxs)(n.p,{children:["Displays the current and recent uptime of your running services, via a self-hosted instance of ",(0,r.jsx)(n.a,{href:"https://github.com/statping/statping",children:"StatPing"})]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"300",src:"https://pixelflare.cc/alicia/dashy/statping"})}),"\n",(0,r.jsx)(n.h4,{id:"options-49",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your StatPing instance, without a trailing slash"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"groupId"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided, only Services in the given group are displayed. Defaults to ",(0,r.jsx)(n.code,{children:"0"})," in which case all services are displayed."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showChart"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided and ",(0,r.jsx)(n.code,{children:"false"})," then charts are not displayed. Defaults to ",(0,r.jsx)(n.code,{children:"true"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"Optional"}),(0,r.jsxs)(n.td,{children:["If provided and ",(0,r.jsx)(n.code,{children:"false"})," then information summaries are not displayed. Defaults to ",(0,r.jsx)(n.code,{children:"true"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-46",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stat-ping\n options:\n hostname: http://192.168.130.1:8080\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: stat-ping\n options:\n hostname: http://192.168.130.1:8080\n groupId: 3\n showChart: false\n showInfo: false\n"})}),"\n",(0,r.jsxs)(n.p,{children:["You can use multiple StatPing widgets with different ",(0,r.jsx)(n.code,{children:"groupId"}),"s."]}),"\n",(0,r.jsx)(n.p,{children:"Note, the Group Id is not directly visible in StatPing UI, you can inspect the group select HTML element or the API response to find out."}),"\n",(0,r.jsx)(n.h4,{id:"info-48",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/statping/statping",children:"GitHub - StatPing"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://docs.statping.com/",children:"StatPing Docs"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"synology-download-station",children:"Synology Download Station"}),"\n",(0,r.jsx)(n.p,{children:"Displays the current downloads/torrents tasks of your Synology NAS"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/synology-downloads"})}),"\n",(0,r.jsx)(n.h4,{id:"options-50",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your Synology NAS, without a trailing slash"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The username of a user on your synology NAS. You will see only this user download station tasks if he is not part of the administrator group. Currently don't support OTP protected accounts."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The password of the account specified above."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-47",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: synology-download\n options:\n hostname: http://192.168.1.1:8080\n username: dashy\n password: totally-secure-password\n\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-49",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://www.synology.com/en-us",children:"Synology"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.synology.com/en-us/company/legal/privacy",children:"Synology Privacy Statement"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-block-stats",children:"AdGuard Home Block Stats"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and\ndisplays total number of allowed and blocked queries, plus a pie chart showing breakdown by block type."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-51",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-48",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-stats\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-50",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-filters",children:"AdGuard Home Filters"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, to display the current status of each of your filter lists. Includes filter name, last updated, number of items, and a link to the list."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-filters"})}),"\n",(0,r.jsx)(n.h4,{id:"options-52",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showOnOffStatusOnly"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),", will only show aggregated AdGuard filter status (on/off), instead of a list of filters"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-49",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-filter-status\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n showOnOffStatusOnly: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-51",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-dns-info",children:"AdGuard Home DNS Info"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and displays the current status (Enabled / Disabled) of AdGuard DNS. Click show more to view detailed info, including upstream DNS provider, active ports, and the status of DNSSEC, EDNS CS, PTR and IPv6."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/adguard-dns"})}),"\n",(0,r.jsx)(n.h4,{id:"options-53",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showFullInfo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If set to ",(0,r.jsx)(n.code,{children:"true"}),', the full DNS info will be shown by default, without having to click "Show Info"']})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-50",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-dns-info\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n showFullInfo: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-52",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adguard-home-top-domains",children:"AdGuard Home Top Domains"}),"\n",(0,r.jsxs)(n.p,{children:["Fetches data from your ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"})," instance, and displays a list of the most queried, and most blocked domains."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"600",src:"https://pixelflare.cc/alicia/dashy/adguard-domains"})}),"\n",(0,r.jsx)(n.h4,{id:"options-54",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to your AdGuard Home instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your username here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If you've got auth enabled on AdGuard, provide your password here"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify the number of results to show, between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"100"}),", defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideBlockedDomains"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show the blocked domains list (queried domains only)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideQueriedDomains"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Don't show the queried domains list (blocked domains only)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-51",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: adguard-top-domains\n useProxy: true\n options:\n hostname: http://127.0.0.1\n username: admin\n password: test\n limit: 10\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-53",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/adguard-home/overview.html",children:"AdGuard Home"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://adguard.com/en/privacy.html",children:"AdGuard Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-user",children:"Nextcloud User"}),"\n",(0,r.jsxs)(n.p,{children:["Nextcloud is a ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/install/#instructions-server",children:"self hosted"})," productivity platform, it can also be used free of charge with ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/sign-up/",children:"hundreds of existing hosting providers"})," that offer a free Nextcloud account."]}),"\n",(0,r.jsx)(n.p,{children:"Displays branding information of a Nextcloud server (logo, url, slogan) and some user details (name, login name, last login, disk space or quota). Use with regular or admin user."}),"\n",(0,r.jsx)(n.p,{children:"Shows quota usage when quota is enabled for the user or disk usage when not enabled."}),"\n",(0,r.jsx)(n.p,{children:"Known issues: the User API incorrectly reports available disk space as total for admin users when quota is not enabled (which usually is the case for admins)."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-user",alt:"nextcloud-user"})}),"\n",(0,r.jsx)(n.h4,{id:"options-55",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-52",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-user\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-54",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-user-statuses",children:"Nextcloud User Statuses"}),"\n",(0,r.jsx)(n.p,{children:"Show user statuses for selected users."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-user-status",alt:"nextcloud-userstatus"})}),"\n",(0,r.jsx)(n.h4,{id:"options-56",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"users"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["Nextcloud User IDs to show statuses for, list size between ",(0,r.jsx)(n.code,{children:"1"})," and ",(0,r.jsx)(n.code,{children:"100"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showEmpty"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Show statuses without a message, defaults to ",(0,r.jsx)(n.code,{children:"true"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-53",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-userstatus\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n users: ['bob', 'alice']\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-55",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-notifications",children:"Nextcloud Notifications"}),"\n",(0,r.jsx)(n.p,{children:"Displays your notifications and allows deleting them."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/next-cloud-notifications",alt:"nextcloud-notifications"})}),"\n",(0,r.jsx)(n.h4,{id:"options-57",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number|string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit displayed notifications either by count, e.g. ",(0,r.jsx)(n.code,{children:"5"})," to show the 5 most recent, or by age, e.g. ",(0,r.jsx)(n.code,{children:"1d"})," to only show notifications not older than a day. Accepted suffixes for age limit are ",(0,r.jsx)(n.code,{children:"m"}),", ",(0,r.jsx)(n.code,{children:"h"})," and ",(0,r.jsx)(n.code,{children:"d"}),"."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-54",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-notifications\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n limit: 6h\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-56",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-system",children:"Nextcloud System"}),"\n",(0,r.jsx)(n.p,{children:"Visualises overall memory utilisation and CPU load averages, shows server versions."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/next-cloud-sysyem",alt:"nextcloud-system"})}),"\n",(0,r.jsx)(n.h4,{id:"options-58",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-55",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-system\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-57",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-stats",children:"Nextcloud Stats"}),"\n",(0,r.jsx)(n.p,{children:"Shows key usage statistics about your Nextcloud server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-stats",alt:"nextcloud-stats"})}),"\n",(0,r.jsx)(n.h4,{id:"options-59",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-56",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-stats\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-58",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"nextcloud-php-opcache-stats",children:"Nextcloud PHP OPcache Stats"}),"\n",(0,r.jsx)(n.p,{children:"Shows statistics about PHP OPcache performance on your Nextcloud server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/nextcloud-php",alt:"nextcloud-phpopcache"})}),"\n",(0,r.jsx)(n.h4,{id:"options-60",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Nextcloud server"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Must be a Nextcloud admin user"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"Nextcloud app-password (create one in Settings -> Security)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-57",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: nextcloud-php-opcache\n useProxy: true\n options:\n hostname: https://nextcloud.example.com\n username: alice\n password: xxxxx-xxxxx-xxxxx-xxxxx\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-59",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://nextcloud.com",children:"Nextcloud"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://nextcloud.com/privacy",children:"Nextcloud Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ntfy-stream",children:"ntfy stream"}),"\n",(0,r.jsxs)(n.p,{children:["Subscribes to topics on a ",(0,r.jsx)(n.strong,{children:"private"})," ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/",children:"ntfy"})," server, and shows ",(0,r.jsx)(n.strong,{children:"new messages"})," as they arrive."]}),"\n",(0,r.jsx)(n.h4,{id:"options-61",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"server_url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The server url"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"topic"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A topic, or a comma separated list of topics"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"auth"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["An ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/subscribe/api/#authentication",children:"auth string"}),". \u26a0\ufe0f This is sent as query parameter."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A title for the widget."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-58",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: ntfy-stream\n useProxy: false\n options:\n title: NTFY stream\n server_url: https://myntfy.server.tld\n topic: alert,warning,mytopic\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-60",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": Disabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/install/",children:"Self-hosted"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsx)(n.a,{href:"https://docs.ntfy.sh/privacy/",children:"ntfy Privacy Policy"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"proxmox-lists",children:"Proxmox lists"}),"\n",(0,r.jsx)(n.p,{children:"Shows lists of nodes, containers, and VMs in a Proxmox virtual environment cluster, with a status indicator."}),"\n",(0,r.jsx)(n.h4,{id:"options-62",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"cluster_url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The URL of the proxmox cluster server. No trailing ",(0,r.jsx)(n.code,{children:"/"}),". for example: ",(0,r.jsx)(n.code,{children:"https://proxmox.lan:8006"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"user_name"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A Proxmox API Username, for example ",(0,r.jsx)(n.code,{children:"root@pam"})," or ",(0,r.jsx)(n.code,{children:"dashy@pve"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token_name"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A Proxmox API token name. You can get a token in the API section of the cluster management interface."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token_uuid"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The value of the token entered above. This is normally a UUID."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"node"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A Proxmox node name. If empty or not supplied, a list of nodes will be shown."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"node_data"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["This is required if a node is selected, Currently this accepts two values, either ",(0,r.jsx)(n.code,{children:"lxc"})," or ",(0,r.jsx)(n.code,{children:"qemu"})," but the widget can be improved to get other types of data from the Proxmox API."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A widget title."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"title_as_link"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["When this is set to anything other than 0 or false, the title will be linked to the value entered in the ",(0,r.jsx)(n.code,{children:"cluster_url"})," option."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"footer"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"A widget footer."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"footer_as_link"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsxs)(n.td,{children:["When this is set to anything other than 0 or false, the title will be linked to the value entered in the ",(0,r.jsx)(n.code,{children:"cluster_url"})," option."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hide_templates"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:"optional"}),(0,r.jsx)(n.td,{children:"When this is set to anything other than 0 or false, templates will be filtered out of the result list."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-59",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:"This will show the list of nodes."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: proxmox-lists\n useProxy: true \n options:\n cluster_url: https://proxmox.lan:8006\n user_name: root@pam\n token_name: dashy\n token_uuid: bfb152df-abcd-abcd-abcd-ccb95a472d01\n"})}),"\n",(0,r.jsx)(n.p,{children:"This will show the list of VMs, with a title and a linked fotter, hiding VM templates."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: proxmox-lists\n useProxy: true \n options:\n cluster_url: https://proxmox.lan:8006\n user_name: root@pam\n token_name: dashy\n token_uuid: bfb152df-abcd-abcd-abcd-ccb95a472d01\n node: proxmox\n node_data: qemu\n title: Proxmox VMs\n title_as_link: false\n footer: Proxmox\n footer_as_link: true\n hide_templates: 1\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-61",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://proxmox.com/en/proxmox-ve",children:"Proxmox Virtual Environment"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://proxmox.com/en/privacy-policy",children:"Proxmox's Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"troubleshooting",children:"Troubleshooting"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"404 Error in development mode"}),": The error might disappear in production mode ",(0,r.jsx)(n.code,{children:"yarn start"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"500 Error in production mode"}),": Try adding the certificate authority (CA) certificate of your Proxmox host to Node.js.\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Download the Proxmox CA certificate to your Dashy host."}),"\n",(0,r.jsxs)(n.li,{children:["Export environment variable ",(0,r.jsx)(n.code,{children:"NODE_EXTRA_CA_CERTS"})," and set its value to the path of the downloaded CA certificate. Example: ",(0,r.jsx)(n.code,{children:"export NODE_EXTRA_CA_CERTS=/usr/local/share/ca-certificates/devlab_ca.pem"})]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"sabnzbd",children:"Sabnzbd"}),"\n",(0,r.jsx)(n.p,{children:"Shows queue information regarding your self hosted Sabnzbd server."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"450",src:"https://pixelflare.cc/alicia/dashy/sabnzbd",alt:"Sabnzbd"})}),"\n",(0,r.jsx)(n.h4,{id:"options-63",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"sabnzbdUrl"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The URL of the Sabnzbd server. No trailing ",(0,r.jsx)(n.code,{children:"/"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["API key for Sabnzbd access. Located under ",(0,r.jsx)(n.code,{children:"Config"})," -> ",(0,r.jsx)(n.code,{children:"General"})," -> ",(0,r.jsx)(n.code,{children:"Security"})," -> ",(0,r.jsx)(n.code,{children:"API Key"}),"."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideDetails"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Hides extra server queue details."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideQueue"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Hides the queue list in an expandable dropdown."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-60",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:" - type: sabnzbd\n options:\n sabnzbdUrl: 'https://sabnzbd.example.com'\n apiKey: XXXXXXXXXXXXXXXXXX\n hideDetails: false\n hideQueue: false\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-62",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://sabnzbd.org/",children:"Sabnzbd"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://forums.sabnzbd.org/ucp.php?mode=privacy",children:"Sabnzbd Privacy Policy"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"gluetun-vpn-info",children:"Gluetun VPN Info"}),"\n",(0,r.jsx)(n.p,{children:"Display info from the Gluetun VPN container public IP API. This can show the IP and location data for the exit VPN node."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/guletn-vpn"})}),"\n",(0,r.jsx)(n.h4,{id:"options-64",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"visibleFields"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["A comma separated list of the fields you want visible in the widget. You can have any number of the following : ",(0,r.jsx)(n.code,{children:"public_ip"}),", ",(0,r.jsx)(n.code,{children:"region"}),", ",(0,r.jsx)(n.code,{children:"country"}),", ",(0,r.jsx)(n.code,{children:"city"}),", ",(0,r.jsx)(n.code,{children:"location"}),", ",(0,r.jsx)(n.code,{children:"organisation"}),", ",(0,r.jsx)(n.code,{children:"postal_code"}),", ",(0,r.jsx)(n.code,{children:"timezone"}),". Defaults to just ",(0,r.jsx)(n.code,{children:"public_ip"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The url to the gluetun HTTP control server. E.g. ",(0,r.jsx)(n.code,{children:"http://gluetun:8000"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-61",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gluetun-status\n useProxy: true\n options:\n hostname: http://server-or-conatiner-hostname:8000\n visibleFields: public_ip,region,country,city,location,organisation,postal_code,timezone\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-63",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/gluetun",children:"Gluetun"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/qdm12/gluetun/wiki",children:"Gluetun Wiki"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"drone-ci-builds",children:"Drone CI Builds"}),"\n",(0,r.jsxs)(n.p,{children:["Display the last builds from a ",(0,r.jsx)(n.a,{href:"https://www.drone.ci",children:"Drone CI"})," instance. A self-hosted CI system that uses docker."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"380",src:"https://pixelflare.cc/alicia/dashy/drone-ci"})}),"\n",(0,r.jsx)(n.h4,{id:"options-65",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The hostname of the Drone CI instance."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The API key (https://[your-drone-instance]/account)."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"integer"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Limit the amounts of listed builds."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"repo"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Show only builds of the specified repo"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-62",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: drone-ci\n updateInterval: 30\n options:\n host: https://drone.somedomain.com\n apiKey: my-very-secret-api-key\n limit: 10\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-64",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://www.drone.io",children:"Drone"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://www.drone.io",children:"Drone"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"filebrowser",children:"Filebrowser"}),"\n",(0,r.jsxs)(n.p,{children:["Displays storage statistics and file listings from a ",(0,r.jsx)(n.a,{href:"https://github.com/gtsteffaniak/filebrowser",children:"Filebrowser Quantum"})," instance. Shows directory size, file/folder counts, favorite files, and recently modified files with quick-access links."]}),"\n",(0,r.jsx)(n.h4,{id:"options-66",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of your Filebrowser instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"A long-lived API key (create in Settings \u2192 API Keys)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"source"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"The source/scope name to browse. Defaults to the first available source"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"path"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["The directory path to display. Defaults to ",(0,r.jsx)(n.code,{children:"/"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"favorites"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"List of filenames to show as quick-access favorites"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showRecent"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Number of recently modified files to display. Defaults to ",(0,r.jsx)(n.code,{children:"5"}),", set to ",(0,r.jsx)(n.code,{children:"0"})," to disable"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Maximum number of files to display per section. Defaults to ",(0,r.jsx)(n.code,{children:"10"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideStats"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the storage statistics section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideFavorites"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the favorites section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hideRecent"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", hides the recent files section"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"showDetailedStats"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"boolean"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If ",(0,r.jsx)(n.code,{children:"true"}),", shows additional statistics including last modified date, largest file, hidden file count, total items, and file type breakdown. Defaults to ",(0,r.jsx)(n.code,{children:"false"})]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-63",children:"Example"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Basic usage:"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: filebrowser\n useProxy: true\n options:\n hostname: http://filebrowser.local:8080\n apiKey: DASHY_FILEBROWSER_KEY\n source: Documents\n path: /\n showRecent: 5\n favorites:\n - important-notes.txt\n - config.yaml\n"})}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"With detailed statistics:"})}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: filebrowser\n useProxy: true\n options:\n hostname: http://filebrowser.local:8080\n apiKey: DASHY_FILEBROWSER_KEY\n source: Downloads\n showDetailedStats: true\n showRecent: 10\n limit: 15\n"})}),"\n",(0,r.jsx)(n.h4,{id:"widget-sections",children:"Widget Sections"}),"\n",(0,r.jsx)(n.p,{children:"The widget displays up to four sections:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Storage Stats"})," - Directory name, total size, file and folder counts"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Detailed Stats"})," (optional) - Last modified date, largest file, hidden file count, total items, and file type breakdown with badges"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Favorites"})," - Quick-access links to user-specified files"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Recent Files"})," - Most recently modified files sorted by date"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"info-65",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/gtsteffaniak/filebrowser",children:"Filebrowser Quantum"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsx)(n.em,{children:"Self-Hosted"})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"linkding",children:"Linkding"}),"\n",(0,r.jsx)(n.p,{children:"Linkding is a self-hosted bookmarking service, which has a clean interface and is simple to set up. This lists the links, filterable by tags."}),"\n",(0,r.jsx)(n.h4,{id:"options-67",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The hostname of the Drone CI instance."})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The API key (",(0,r.jsx)(n.a,{href:"https://your-linkding-instance/settings/integrations",children:"https://your-linkding-instance/settings/integrations"}),")."]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"tags"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"list of string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Filter the links by tag."})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-64",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: linkding\n updateInterval: 30\n options:\n host: https://lingding.somedomain.com\n apiKey: my-very-secret-api-key\n tags: \n - rpg\n - markdown\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-66",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/sissbruecker/linkding",children:"Linkding"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/sissbruecker/linkding",children:"Linkding"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime-kuma",children:"Uptime Kuma"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," is an easy-to-use self-hosted monitoring tool."]}),"\n",(0,r.jsx)(n.h4,{id:"options-68",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Uptime Kuma instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiKey"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The API key (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma/wiki/API-Keys",children:"https://github.com/louislam/uptime-kuma/wiki/API-Keys"}),")."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-65",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: uptime-kuma\n useProxy: true\n options:\n apiKey: uk2_99H0Yd3I2pPNIRfn0TqBFu4g5q85R1Mh75yZzw6H\n url: http://192.168.1.106:3691/metrics\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-67",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime-kuma-status-page",children:"Uptime Kuma Status Page"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," is an easy-to-use self-hosted monitoring tool."]}),"\n",(0,r.jsx)(n.h4,{id:"options-69",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"host"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL of the Uptime Kuma instance"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"slug"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The slug of the status page"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"monitorNames"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"strins"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Names of monitored services (in the same order as on the kuma uptime status page)"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-66",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: uptime-kuma-status-page\n options:\n host: http://localhost:3001\n slug: another-beautiful-status-page\n monitorNames:\n - "Name1"\n - "Name2"\n'})}),"\n",(0,r.jsx)(n.h4,{id:"info-68",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Not Needed"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/louislam/uptime-kuma",children:"Uptime Kuma"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"tactical-rmm",children:"Tactical RMM"}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})," is a self-hosted remote monitoring & management tool."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsxs)(n.a,{href:"https://github.com/user-attachments/assets/152a7205-e5de-401f-bad8-19063ddfaf3c",children:["\n ",(0,r.jsx)(n.img,{src:"https://github.com/user-attachments/assets/5921d46f-d84c-494d-8aaf-6b20cc592640",alt:"Capture",border:"0"})]})}),"\n",(0,r.jsx)(n.h4,{id:"options-70",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The status endpoint URL (",(0,r.jsx)(n.a,{href:"https://api.example.com/core/v2/status/",children:"https://api.example.com/core/v2/status/"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"token"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["The MON_TOKEN (see ",(0,r.jsx)(n.a,{href:"https://docs.tacticalrmm.com/tipsntricks/#monitor-your-trmm-instance-via-the-built-in-monitoring-endpoint",children:"https://docs.tacticalrmm.com/tipsntricks/#monitor-your-trmm-instance-via-the-built-in-monitoring-endpoint"}),")."]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-67",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: trmm\n useProxy: true\n options:\n token: PkPVKMzbmXgeQDlJWb0WXYvsIk3JvZyadURud2cSTdMia6hUbQ\n url: https://api.example.com/core/v2/status/\n"})}),"\n",(0,r.jsx)(n.h4,{id:"info-69",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe0 Proxied"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe2 Required"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})," )"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": ",(0,r.jsxs)(n.em,{children:["See ",(0,r.jsx)(n.a,{href:"https://github.com/amidaware/tacticalrmm",children:"Tactical RMM"})]})]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"system-resource-monitoring",children:"System Resource Monitoring"}),"\n",(0,r.jsx)(n.h3,{id:"glances",children:"Glances"}),"\n",(0,r.jsxs)(n.p,{children:["The easiest method for displaying system info and resource usage in Dashy is with ",(0,r.jsx)(n.a,{href:"https://nicolargo.github.io/glances/",children:"Glances"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Glances is a cross-platform monitoring tool developed by ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo",children:"@nicolargo"}),". It's similar to top/htop but with a ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/api.html",children:"Rest API"})," and many ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/gw/index.html",children:"data exporters"})," available. Under the hood, it uses ",(0,r.jsx)(n.a,{href:"https://github.com/giampaolo/psutil",children:"psutil"})," for retrieving system info."]}),"\n",(0,r.jsxs)(n.p,{children:["If you don't already have it installed, either follow the ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/blob/master/README.rst",children:"Installation Guide"})," for your system, or setup ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/docker.html",children:"with Docker"}),", or use the one-line install script: ",(0,r.jsx)(n.code,{children:"curl -L https://bit.ly/glances | /bin/bash"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are using Docker to run glances make sure to add the enviroment variable ",(0,r.jsx)(n.code,{children:"-e TZ = {YourTimeZone}"}),". You can get a list of valid timezones by running ",(0,r.jsx)(n.code,{children:"timedatectl list-timezones"})," on any linux system. This is needed so the graphs show the currect time."]}),"\n",(0,r.jsx)(n.p,{children:"Here an example for Docker"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{children:" docker run -d \\\n --name glances \\\n --restart unless-stopped \\\n -v /var/run/docker.sock:/var/run/docker.sock:ro \\\n -p 61208:61208 \\\n --pid host \\\n --privileged \\\n -e GLANCES_OPT=-w \\\n -e PUID=1000 \\\n -e PGID=1000 \\\n -e TZ=Europe/Zurich \\\n nicolargo/glances:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Glances can be launched with the ",(0,r.jsx)(n.code,{children:"glances"})," command. You'll need to run it in web server mode, using the ",(0,r.jsx)(n.code,{children:"-w"})," option for the API to be reachable. If you don't plan on using the Web UI, then you can disable it using ",(0,r.jsx)(n.code,{children:"--disable-webui"}),". See the ",(0,r.jsx)(n.a,{href:"https://glances.readthedocs.io/en/latest/cmds.html",children:"command reference docs"})," for more info."]}),"\n",(0,r.jsxs)(n.p,{children:["If Glances is running on a Windows system it is recommended to add the following arguments ",(0,r.jsx)(n.code,{children:"--disable-plugin all --enable-plugin cpu,mem,diskio,ip,network,containers,quicklook,load,fs,alert -w"})," This is due to Glances not being that stable on windows, so disabling all plugins that aren't used by Dashy widgets can save on ressources."]}),"\n",(0,r.jsx)(n.h4,{id:"options-71",children:"Options"}),"\n",(0,r.jsxs)(n.p,{children:["All Glance's based widgets require a ",(0,r.jsx)(n.code,{children:"hostname"}),". All other parameters are optional."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"hostname"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL or IP + port to your Glances instance (without a trailing slash)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you have setup basic auth on Glances, specify username here (defaults to ",(0,r.jsx)(n.code,{children:"glances"}),")"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If you have setup basic auth on Glances, specify password here. ",(0,r.jsx)(n.strong,{children:"Note"}),": since this password is in plaintext, it is important not to reuse it anywhere else"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Specify an API version, defaults to V ",(0,r.jsx)(n.code,{children:"3"}),". Note that support for older versions is limited"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"For widgets that show a time-series chart, optionally limit the number of data points returned. A higher number will show more historical results, but will take longer to load. A value between 300 - 800 is usually optimal"})]})]})]}),"\n",(0,r.jsxs)(n.p,{children:["Note that if auth is configured, requests must be proxied with ",(0,r.jsx)(n.code,{children:"useProxy: true"})]}),"\n",(0,r.jsx)(n.h4,{id:"info-70",children:"Info"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"CORS"}),": \ud83d\udfe2 Enabled"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Auth"}),": \ud83d\udfe0 Optional"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Price"}),": \ud83d\udfe2 Free"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Host"}),": Self-Hosted (see ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances",children:"GitHub - Nicolargo/Glances"}),")"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Privacy"}),": \u26ab No Policy Available"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"screenshot",children:"Screenshot"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.a,{href:"https://ibb.co/pR6dMZT",children:(0,r.jsx)(n.img,{src:"https://pixelflare.cc/alicia/dashy/monitor-board",alt:"example-screenshot"})})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-cpu-usage",children:"Current CPU Usage"}),"\n",(0,r.jsx)(n.p,{children:"Live-updating current CPU usage, as a combined average across all cores"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-cpu"})}),"\n",(0,r.jsx)(n.h4,{id:"example-68",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cpu\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-cpu-usage-speedometer",children:"Current CPU Usage Speedometer"}),"\n",(0,r.jsx)(n.p,{children:"Speedometer styled version of the Current CPU Usage widget"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-cpu-speedometer"})}),"\n",(0,r.jsx)(n.h4,{id:"example-69",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-speedometer\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-usage-per-core",children:"CPU Usage Per Core"}),"\n",(0,r.jsx)(n.p,{children:"Live-updating CPU usage breakdown per core"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cpu-per-core"})}),"\n",(0,r.jsx)(n.h4,{id:"example-70",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cores\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-usage-history",children:"CPU Usage History"}),"\n",(0,r.jsx)(n.p,{children:"Recent CPU usage history, across all cores, and displayed by user and system"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/cpu-history-chart"})}),"\n",(0,r.jsx)(n.h4,{id:"options-72",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results returned, rendering more data points will take longer to load. Defaults to ",(0,r.jsx)(n.code,{children:"100"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-71",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-history\n options:\n hostname: http://192.168.130.2:61208\n limit: 60\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-memory-usage",children:"Current Memory Usage"}),"\n",(0,r.jsx)(n.p,{children:"Real-time memory usage gauge, with more info visible on click"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-mem"})}),"\n",(0,r.jsx)(n.h4,{id:"example-72",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-mem\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"current-memory-usage-speedometer",children:"Current Memory Usage Speedometer"}),"\n",(0,r.jsx)(n.p,{children:"Speedometer styled version of the Current Memory Usage widget"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/current-mem-speedometer"})}),"\n",(0,r.jsx)(n.h4,{id:"example-73",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-mem-speedometer\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"memory-usage-history",children:"Memory Usage History"}),"\n",(0,r.jsx)(n.p,{children:"Recent memory usage chart"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/mem-history-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"options-73",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"limit"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Limit the number of results returned, rendering more data points will take longer to load. Defaults to ",(0,r.jsx)(n.code,{children:"100"})]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-74",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-mem-history\n options:\n hostname: http://localhost:61208\n limit: 80\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"disk-space",children:"Disk Space"}),"\n",(0,r.jsx)(n.p,{children:"List connected disks, showing free / used space and other info (file system, mount point and space available)"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/diskspace-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-75",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-space\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"disk-io",children:"Disk IO"}),"\n",(0,r.jsx)(n.p,{children:"Shows real-time read and write speeds and operations per sec for each disk"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/disk-io-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-76",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-io\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"system-load",children:"System Load"}),"\n",(0,r.jsx)(n.p,{children:"Shows the number of processes waiting in the run-queue, averaged across all cores. Displays for past 5, 10 and 15 minutes"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/system-load-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-77",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-system-load\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"uptime",children:"Uptime"}),"\n",(0,r.jsx)(n.p,{children:"Displays the system uptime fetched from Glances."}),"\n",(0,r.jsx)(n.h4,{id:"example-78",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-uptime\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"system-load-history",children:"System Load History"}),"\n",(0,r.jsxs)(n.p,{children:["Shows recent historical system load, calculated from the number of processes waiting in the run-queue, in 1, 5 and 15 minute intervals, and averaged across all cores. Optionally specify ",(0,r.jsx)(n.code,{children:"limit"})," to set number of results returned, defaults to ",(0,r.jsx)(n.code,{children:"500"}),", max ",(0,r.jsx)(n.code,{children:"100000"}),", but the higher the number the longer the load and render times will be."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"500",src:"https://pixelflare.cc/alicia/dashy/system-load-history-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-79",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-load-history\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"network-interfaces",children:"Network Interfaces"}),"\n",(0,r.jsx)(n.p,{children:"Lists visible network interfaces, including real-time upload/ download stats"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/network-interfaces-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-80",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-network-interfaces\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"network-traffic",children:"Network Traffic"}),"\n",(0,r.jsxs)(n.p,{children:["Shows amount of data recently uploaded/ downloaded across all network interfaces. Optionally set the ",(0,r.jsx)(n.code,{children:"limit"})," option to specify number historical of data points to return"]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/network-traffic-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-81",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-network-traffic\n options:\n hostname: http://192.168.130.2:61208\n limit: 500\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"resource-usage-alerts",children:"Resource Usage Alerts"}),"\n",(0,r.jsx)(n.p,{children:"Lists recent high resource usage alerts (e.g. CPU, mem, IO, load, temp)"}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://i.ibb.co/w01NX5R/gl-alerts.png"})}),"\n",(0,r.jsx)(n.h4,{id:"example-82",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-alerts\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ip-address",children:"IP Address"}),"\n",(0,r.jsx)(n.p,{children:"Shows public and private IP address. Note that the ip plugin is not available on all instances of Glances."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/ip-addresses-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"example-83",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-ip-address\n options:\n hostname: http://192.168.130.2:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"cpu-temp",children:"CPU Temp"}),"\n",(0,r.jsx)(n.p,{children:"Displays temperature data from system CPUs."}),"\n",(0,r.jsxs)(n.p,{children:["Note: This widget uses the ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/blob/develop/glances/plugins/glances_sensors.py",children:(0,r.jsx)(n.code,{children:"sensors"})})," plugin, which is disabled by default, and may cause ",(0,r.jsx)(n.a,{href:"https://github.com/nicolargo/glances/issues/1664#issuecomment-632063558",children:"performance issues"}),".\nYou'll need to enable the sensors plugin to use this widget, using: ",(0,r.jsx)(n.code,{children:"--enable-plugin sensors"})," when you start Glances."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/cpu-temp-glances"})}),"\n",(0,r.jsx)(n.h4,{id:"options-74",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsx)(n.tbody,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"units"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Use ",(0,r.jsx)(n.code,{children:"C"})," to display temperatures in Celsius or ",(0,r.jsx)(n.code,{children:"F"})," to use Fahrenheit. Defaults to ",(0,r.jsx)(n.code,{children:"C"}),"."]})]})})]}),"\n",(0,r.jsx)(n.h4,{id:"example-84",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-cpu-temp\n options:\n hostname: http://192.168.130.2:61208\n units: C\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"compact-metrics",children:"Compact Metrics"}),"\n",(0,r.jsx)(n.p,{children:"A multi-system overview widget that displays CPU, memory and disk usage for multiple Glances instances in a compact table. Click on a row to see detailed metrics for that system."}),"\n",(0,r.jsx)(n.h4,{id:"options-75",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"systems"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"array"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsxs)(n.td,{children:["An array of systems to monitor, each with ",(0,r.jsx)(n.code,{children:"header"})," (display name) and ",(0,r.jsx)(n.code,{children:"url"})," (Glances base URL)"]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"apiVersion"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["Glances API version. Defaults to ",(0,r.jsx)(n.code,{children:"4"})]})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"username"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If Glances is password-protected, specify the username"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"password"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"If Glances is password-protected, specify the password"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-85",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-compact-metrics\n options:\n systems:\n - header: Server 1\n url: http://192.168.1.10:61208\n - header: Server 2\n url: http://192.168.1.11:61208\n - header: NAS\n url: http://192.168.1.20:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"dynamic-widgets",children:"Dynamic Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"iframe-widget",children:"Iframe Widget"}),"\n",(0,r.jsx)(n.p,{children:"Embed any webpage into your dashboard as a widget."}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/iframe"})}),"\n",(0,r.jsx)(n.h4,{id:"options-76",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"url"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:"Required"}),(0,r.jsx)(n.td,{children:"The URL to the webpage to embed"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"frameHeight"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"number"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsxs)(n.td,{children:["If needed, specify height of iframe in ",(0,r.jsx)(n.code,{children:"px"}),". E.g. ",(0,r.jsx)(n.code,{children:"400"}),", defaults to auto"]})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-86",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: iframe\n options:\n url: https://fiatleak.com/\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"html-embedded-widget",children:"HTML Embedded Widget"}),"\n",(0,r.jsx)(n.p,{children:"Many websites and apps provide their own embeddable widgets. These can be used with Dashy using the Embed widget, which lets you dynamically embed and HTML, CSS or JavaScript contents."}),"\n",(0,r.jsxs)(n.p,{children:["\u26a0\ufe0f ",(0,r.jsx)(n.strong,{children:"NOTE:"})," Use with extreme caution. Embedding a script from an untrustworthy source may have serious unintended consequences."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"400",src:"https://pixelflare.cc/alicia/dashy/html-embed"})}),"\n",(0,r.jsx)(n.h4,{id:"options-77",children:"Options"}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,r.jsxs)(n.table,{children:[(0,r.jsx)(n.thead,{children:(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Field"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Type"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Required"})}),(0,r.jsx)(n.th,{children:(0,r.jsx)(n.strong,{children:"Description"})})]})}),(0,r.jsxs)(n.tbody,{children:[(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"html"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"HTML contents to render in the widget"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"script"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Raw JavaScript code to execute (caution)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"scriptSrc"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"A URL to JavaScript content (caution)"})]}),(0,r.jsxs)(n.tr,{children:[(0,r.jsx)(n.td,{children:(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"css"})})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.code,{children:"string"})}),(0,r.jsx)(n.td,{children:(0,r.jsx)(n.em,{children:"Optional"})}),(0,r.jsx)(n.td,{children:"Any stylings for widget contents"})]})]})]}),"\n",(0,r.jsx)(n.h4,{id:"example-87",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: embed\n options:\n scriptSrc: https://cdn.speedcheck.org/basic/scbjs.min.js\n html: |\n
\n
\n \n Speedcheck\n \n
\n
\n'})}),"\n",(0,r.jsx)(n.p,{children:"Or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'- type: embed\n options:\n css: \'.coinmarketcap-currency-widget { color: var(--widget-text-color); }\'\n html: \'
\'\n scriptSrc: \'https://files.coinmarketcap.com/static/widget/currency.js\'\n'})}),"\n",(0,r.jsx)(n.p,{children:"You can also use this widget to display an image, wither locally or from a remote origin."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: embed\n options:\n html: ''\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"api-response",children:"API Response"}),"\n",(0,r.jsx)(n.p,{children:"Directly output plain-text response from any API-enabled service."}),"\n",(0,r.jsx)(n.p,{children:"// Coming soon..."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"prometheus-data",children:"Prometheus Data"}),"\n",(0,r.jsx)(n.p,{children:"Display data from any service with a Prometheus exporter."}),"\n",(0,r.jsx)(n.p,{children:"// Coming soon..."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"data-feed",children:"Data Feed"}),"\n",(0,r.jsxs)(n.p,{children:["Show live data from an RSS-enabled service. The only required parameter is ",(0,r.jsx)(n.code,{children:"rssUrl"}),", which is the URL to the ATOM feed. See ",(0,r.jsx)(n.a,{href:"#rss-feed",children:"RSS Widget"})," for full list of available options."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{width:"700",src:"https://pixelflare.cc/alicia/dashy/data-feed"})}),"\n",(0,r.jsx)(n.h4,{id:"example-88",children:"Example"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: rss-feed\n options:\n rssUrl: https://notes.aliciasykes.com/feed\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"usage--customizations",children:"Usage & Customizations"}),"\n",(0,r.jsx)(n.h3,{id:"widget-usage-guide",children:"Widget Usage Guide"}),"\n",(0,r.jsx)(n.p,{children:"Like items, widgets are placed under sections. You may have one or more widgets per section."}),"\n",(0,r.jsx)(n.p,{children:"In your YAML config file, this will look something like:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"sections:\n- name: Today\n icon: far fa-calendar-day\n widgets:\n - type: clock\n options:\n format: en-GB\n - type: weather\n options:\n apiKey: xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\n city: London\n units: metric\n"})}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsx)(n.p,{children:'In this example, there is a single section, named "Today", using a Calendar icon from Font-Awesome. It has 2 widgets, a clock and the current weather.'}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"continuous-updates",children:"Continuous Updates"}),"\n",(0,r.jsxs)(n.p,{children:["By default, a widget which displays dynamic data from an external source, will only fetch results on page load. If you would like to keep data updated at all times, you can enable ",(0,r.jsx)(n.strong,{children:"Continuous Updates"}),". This is done by setting a time value in the ",(0,r.jsx)(n.code,{children:"updateInterval"})," field."]}),"\n",(0,r.jsxs)(n.p,{children:["The value of ",(0,r.jsx)(n.code,{children:"updateInterval"})," is optional, and is specified and seconds. It must be more than ",(0,r.jsx)(n.code,{children:"10"})," and less than ",(0,r.jsx)(n.code,{children:"7200"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example, the following widget displaying stats from Pi-Hole will update ever 20 seconds."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"widgets:\n- type: pi-hole-stats\n updateInterval: 20\n options:\n hostname: http://192.168.130.2\n"})}),"\n",(0,r.jsx)(n.p,{children:"Note that if you have many widgets, and set them to continuously update frequently, you will notice a hit to performance. A widget that relies on data from an external API, will also consume your usage quota faster, if set to keep updating."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"proxying-requests",children:"Proxying Requests"}),"\n",(0,r.jsx)(n.p,{children:"If a widget fails to make a data request, and the console shows a CORS error, this means the server is blocking client-side requests."}),"\n",(0,r.jsxs)(n.p,{children:["Dashy has a built-in CORS proxy (",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/services/cors-proxy.js",children:(0,r.jsx)(n.code,{children:"services/cors-proxy.js"})}),"), which will be used automatically by some widgets, or can be forced to use by other by setting the ",(0,r.jsx)(n.code,{children:"useProxy"})," option."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"widgets:\n- type: pi-hole-stats\n useProxy: true\n options:\n hostname: http://pi-hole.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"Alternatively, and more securely, you can set the auth headers on your service to accept requests from Dashy. For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"handling-secrets",children:"Handling Secrets"}),"\n",(0,r.jsxs)(n.p,{children:["Some widgets require you to pass potentially sensitive info such as API keys. The ",(0,r.jsx)(n.code,{children:"conf.yml"})," is not ideal for this, as it's stored in plaintext. Instead, for secrets you should use environment variables."]}),"\n",(0,r.jsxs)(n.p,{children:["In your widget options, set the value to the name of an environment variable starting with ",(0,r.jsx)(n.code,{children:"DASHY_"})," (or ",(0,r.jsx)(n.code,{children:"VITE_APP_"})," / ",(0,r.jsx)(n.code,{children:"VUE_APP_"})," for backwards compatibility). The Dashy server will substitute it with the matching ",(0,r.jsx)(n.code,{children:"process.env"})," value when proxying the request."]}),"\n",(0,r.jsxs)(n.p,{children:["The widget must be set to route through the Dashy server with ",(0,r.jsx)(n.code,{children:"useProxy: true"}),". Without this, the placeholder is sent directly to the upstream API and will fail. Substitution covers the request URL, headers and body, so any auth pattern (Bearer, Basic, query param, POST body) works."]}),"\n",(0,r.jsxs)(n.p,{children:["For more information about setting and managing your environment variables, see ",(0,r.jsx)(n.a,{href:"/docs/management#passing-in-environmental-variables",children:"Management Docs --\x3e Environmental Variables"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: weather\n useProxy: true\n options:\n apiKey: DASHY_WEATHER_TOKEN\n city: London\n units: metric\n hideDetails: true\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then set ",(0,r.jsx)(n.code,{children:"DASHY_WEATHER_TOKEN='xxx'"})," in your container or local environment, and restart Dashy. To rotate the value, just update the env var and restart. No rebuild required."]}),"\n",(0,r.jsxs)(n.admonition,{type:"note",children:[(0,r.jsxs)(n.p,{children:["Only env vars starting with ",(0,r.jsx)(n.code,{children:"DASHY_"}),", ",(0,r.jsx)(n.code,{children:"VITE_APP_"})," or ",(0,r.jsx)(n.code,{children:"VUE_APP_"})," are eligible for substitution. Other server-side env vars are never exposed."]}),(0,r.jsxs)(n.p,{children:["If you build Dashy from source yourself, ",(0,r.jsx)(n.code,{children:"VITE_APP_*"})," and ",(0,r.jsx)(n.code,{children:"DASHY_*"})," vars set at build time are also baked into the bundle by Vite, which works without ",(0,r.jsx)(n.code,{children:"useProxy: true"}),"."]})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"setting-timeout",children:"Setting Timeout"}),"\n",(0,r.jsxs)(n.p,{children:["If the endpoint you are requesting data from is slow to respond, you may see a timeout error in the console. This can easily be fixed by specifying the ",(0,r.jsx)(n.code,{children:"timeout"})," property on the offending widget. This should be an integer value, in milliseconds. By default timeout is ",(0,r.jsx)(n.code,{children:"2500"})," ms (2\xbd seconds)."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-current-cpu\n timeout: 8000\n options:\n hostname: https://glances.dns-device.local\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"adding-labels",children:"Adding Labels"}),"\n",(0,r.jsxs)(n.p,{children:["If you have multiple widgets of the same type in a single section, it may not be clear what each one is. To overcome this, you can add a custom label to any given widget, using the ",(0,r.jsx)(n.code,{children:"label"})," property."]}),"\n",(0,r.jsx)(n.p,{children:"For example:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- name: CPU Usage\n icon: fas fa-tachometer\n widgets:\n - type: gl-current-cpu\n label: Meida Server\n options:\n hostname: http://media-server.lan:61208\n - type: gl-current-cpu\n label: Firewall\n options:\n hostname: http://firewall.lan:61208\n - type: gl-current-cpu\n label: File Sync Server\n options:\n hostname: http://file-sync.lan:61208\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"ignoring-errors",children:"Ignoring Errors"}),"\n",(0,r.jsx)(n.p,{children:"When there's an error fetching or displaying a widgets data, then it will be highlighted in yellow, and a message displayed on the UI."}),"\n",(0,r.jsxs)(n.p,{children:["In some instances, this is a false positive, and the widget is actually functioning correctly. If this is the case, you can disable the UI error message of a given widget by setting: ",(0,r.jsx)(n.code,{children:"ignoreErrors: true"})]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-disk-io\n ignoreErrors: true\n options:\n hostname: https://glances.dns-device.local\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"widget-styling",children:"Widget Styling"}),"\n",(0,r.jsx)(n.p,{children:"Like elsewhere in Dashy, all colours can be easily modified with CSS variables."}),"\n",(0,r.jsx)(n.p,{children:"Widgets use the following color variables, which can be overridden if desired:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-text-color"})," - Text color, defaults to ",(0,r.jsx)(n.code,{children:"--primary"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-background-color"})," - Background color, defaults to ",(0,r.jsx)(n.code,{children:"--background-darker"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.code,{children:"--widget-accent-color"})," - Accent color, defaults to ",(0,r.jsx)(n.code,{children:"--background"})]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more info on how to apply custom variables, see the ",(0,r.jsx)(n.a,{href:"/docs/theming#setting-custom-css-in-the-ui",children:"Theming Docs"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"customizing-charts",children:"Customizing Charts"}),"\n",(0,r.jsxs)(n.p,{children:["For widgets that contain charts, you can set an array of colors under ",(0,r.jsx)(n.code,{children:"chartColors"}),".\nTo specify the chart height, set ",(0,r.jsx)(n.code,{children:"chartHeight"})," to an integer (in ",(0,r.jsx)(n.code,{children:"px"}),"), defaults to ",(0,r.jsx)(n.code,{children:"300"}),".\nFor example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"- type: gl-load-history\n options:\n hostname: http://192.168.130.2:61208\n chartColors: ['#9b5de5', '#f15bb5', '#00bbf9', '#00f5d4']\n chartHeight: 450\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"language-translations",children:"Language Translations"}),"\n",(0,r.jsx)(n.p,{children:"Since most of the content displayed within widgets is fetched from an external API, unless that API supports multiple languages, translating dynamic content is not possible."}),"\n",(0,r.jsx)(n.p,{children:"However, any hard-coded content is translatable, and all dates and times will display in your local format."}),"\n",(0,r.jsxs)(n.p,{children:["For more info about multi-language support, see the ",(0,r.jsx)(n.a,{href:"/docs/multi-language-support",children:"Internationalization Docs"}),"."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"widget-ui-options",children:"Widget UI Options"}),"\n",(0,r.jsx)(n.p,{children:"Widgets can be opened in full-page view, by clicking the Arrow icon (top-right). The URL in your address bar will also update, and visiting that web address directly will take you straight to that widget."}),"\n",(0,r.jsx)(n.p,{children:"You can reload the data of any widget, by clicking the Refresh Data icon (also in top-right). This will only affect the widget where the action was triggered from."}),"\n",(0,r.jsxs)(n.p,{children:["All ",(0,r.jsx)(n.a,{href:"/docs/configuring#section",children:"config options"})," that can be applied to sections, can also be applied to widget sections. For example, to make a widget section double the width, set ",(0,r.jsx)(n.code,{children:"displayData.cols: 2"})," within the parent section. You can collapse a widget (by clicking the section title), and collapse state will be saved locally."]}),"\n",(0,r.jsxs)(n.p,{children:["Widgets cannot currently be edited through the UI. This feature is in development, and will be released soon. In the meantime, you can either use the JSON config editor, or use ",(0,r.jsx)(n.a,{href:"https://github.com/coder/code-server",children:"VS Code Server"}),", or just SSH into your box and edit the conf.yml file directly."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"build-your-own-widget",children:"Build your own Widget"}),"\n",(0,r.jsx)(n.p,{children:"Widgets are built in a modular fashion, making it easy for anyone to create their own custom components."}),"\n",(0,r.jsxs)(n.p,{children:["For a full tutorial on creating your own widget, you can follow ",(0,r.jsx)(n.a,{href:"/docs/development-guides#building-a-widget",children:"this guide"}),", or take a look at ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/commit/3da76ce2999f57f76a97454c0276301e39957b8e",children:"here"})," for a code example."]}),"\n",(0,r.jsxs)(n.p,{children:["Alternatively, for displaying simple data, you could also just use the either the ",(0,r.jsx)(n.a,{href:"#iframe-widget",children:"iframe"}),", ",(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"embed"}),", ",(0,r.jsx)(n.a,{href:"#data-feed",children:"data feed"})," or ",(0,r.jsx)(n.a,{href:"#api-response",children:"API response"})," widgets."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"requesting-a-widget",children:"Requesting a Widget"}),"\n",(0,r.jsx)(n.p,{children:"Suggestions for widget ideas are welcome. But there is no guarantee that I will build your widget idea."}),"\n",(0,r.jsx)(n.p,{children:"Please only request widgets for services that:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Have a publicly accessible API"}),"\n",(0,r.jsx)(n.li,{children:"Are CORS and HTTPS enabled"}),"\n",(0,r.jsx)(n.li,{children:"Are free to use, or have a free plan"}),"\n",(0,r.jsx)(n.li,{children:"Allow for use in their Terms of Service"}),"\n",(0,r.jsx)(n.li,{children:"Would be useful for other users"}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["You can suggest a widget ",(0,r.jsx)(n.a,{href:"https://git.io/Jygo3",children:"here"}),", please star the repo before submitting a ticket. If you are a monthly GitHub sponsor, I will happily build out a custom widget for any service that meets the above criteria, usually within 2 weeks of initial request."]}),"\n",(0,r.jsxs)(n.p,{children:["For services that are not officially supported, it is likely still possible to display data using either the ",(0,r.jsx)(n.a,{href:"#iframe-widget",children:"iframe"}),", ",(0,r.jsx)(n.a,{href:"#html-embedded-widget",children:"embed"})," or ",(0,r.jsx)(n.a,{href:"#api-response",children:"API response"})," widgets. For more advanced features, like charts and action buttons, you could also build your own widget, using ",(0,r.jsx)(n.a,{href:"/docs/development-guides#building-a-widget",children:"this tutorial"}),", it's fairly straight forward, and you can use an ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/components/Widgets",children:"existing widget"})," (or ",(0,r.jsx)(n.a,{href:"https://git.io/JygKI",children:"this example"}),") as a template."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"troubleshooting-widget-errors",children:"Troubleshooting Widget Errors"}),"\n",(0,r.jsxs)(n.p,{children:["If an error occurs when fetching or rendering results, you will see a short message in the UI. If that message doesn't adequately explain the problem, then you can ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open the browser console"})," to see more details."]}),"\n",(0,r.jsx)(n.p,{children:"Before proceeding, ensure that if the widget requires auth your API is correct, and for custom widgets, double check that the URL and protocol is correct."}),"\n",(0,r.jsx)(n.p,{children:"If you're able to, you can find more information about why the request may be failing in the Dev Tools under the Network tab, and you can ensure your endpoint is correct and working using a tool like Postman."}),"\n",(0,r.jsx)(n.h4,{id:"cors-errors",children:"CORS Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The most common issue is a CORS error. This is a browser security mechanism which prevents the client-side app (Dashy) from from accessing resources on a remote origin, without that server's explicit permission (e.g. with headers like Access-Control-Allow-Origin). See the MDN Docs for more info: ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"There are several ways to fix a CORS error:"}),"\n",(0,r.jsx)(n.h4,{id:"option-1---ensure-correct-protocol",children:"Option 1 - Ensure Correct Protocol"}),"\n",(0,r.jsx)(n.p,{children:"You will get a CORS error if you try and access a http service from a https source. So ensure that the URL you are requesting has the right protocol, and is correctly formatted."}),"\n",(0,r.jsx)(n.h4,{id:"option-2---set-headers",children:"Option 2 - Set Headers"}),"\n",(0,r.jsxs)(n.p,{children:["If you have control over the destination (e.g. for a self-hosted service), then you can simply apply the correct headers.\nAdd the ",(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, with the value of either ",(0,r.jsx)(n.code,{children:"*"})," to allow requests from anywhere, or more securely, the host of where Dashy is served from. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://url-of-dashy.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: *\n"})}),"\n",(0,r.jsx)(n.h4,{id:"option-3---proxying-request",children:"Option 3 - Proxying Request"}),"\n",(0,r.jsxs)(n.p,{children:["You can route requests through Dashy's built-in CORS proxy. Instructions and more details can be found ",(0,r.jsx)(n.a,{href:"#proxying-requests",children:"here"}),". If you don't have control over the target origin, and you are running Dashy either through Docker, with the Node server or on Netlify, then this solution will work for you."]}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.code,{children:"useProxy: true"})," option to the failing widget."]}),"\n",(0,r.jsx)(n.h4,{id:"option-4---use-a-plugin",children:"Option 4 - Use a plugin"}),"\n",(0,r.jsxs)(n.p,{children:["For testing purposes, you can use an addon, which will disable the CORS checks. You can get the Allow-CORS extension for ",(0,r.jsx)(n.a,{href:"https://chrome.google.com/webstore/detail/allow-cors-access-control/lhobafahddgcelffkeicbaginigeejlf?hl=en-US",children:"Chrome"})," or ",(0,r.jsx)(n.a,{href:"https://addons.mozilla.org/en-US/firefox/addon/access-control-allow-origin/",children:"Firefox"}),", more details ",(0,r.jsx)(n.a,{href:"https://mybrowseraddon.com/access-control-allow-origin.html",children:"here"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h3,{id:"raising-an-issue",children:"Raising an Issue"}),"\n",(0,r.jsxs)(n.p,{children:["If you need to submit a bug report for a failing widget, then please include the full console output (see ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"how"}),") as well as the relevant parts of your config file. Before sending the request, ensure you've read the docs. If you're new to GitHub, an haven't previously contributed to the project, then please fist star the repo to avoid your ticket being closed by the anti-spam bot."]})]})}function x(e={}){const{wrapper:n}={...(0,d.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(o,{...e})}):o(e)}},8453(e,n,s){s.d(n,{R:()=>l,x:()=>t});var i=s(6540);const r={},d=i.createContext(r);function l(e){const n=i.useContext(d);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function t(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:l(e.components),i.createElement(d.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/9d9f8394.794684b5.js b/assets/js/9d9f8394.1b16e9cd.js similarity index 99% rename from assets/js/9d9f8394.794684b5.js rename to assets/js/9d9f8394.1b16e9cd.js index 8ea92207..2f7d87ff 100644 --- a/assets/js/9d9f8394.794684b5.js +++ b/assets/js/9d9f8394.1b16e9cd.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9013],{7309(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>a,default:()=>c,frontMatter:()=>t,metadata:()=>i,toc:()=>d});const i=JSON.parse('{"id":"troubleshooting","title":"Troubleshooting","description":"**This document contains common problems and their solutions.**","source":"@site/docs/troubleshooting.md","sourceDirName":".","slug":"/troubleshooting","permalink":"/docs/troubleshooting","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/troubleshooting.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"App Management","permalink":"/docs/management"},"next":{"title":"Icons","permalink":"/docs/icons"}}');var r=s(4848),o=s(8453);const t={},a="Troubleshooting",l={},d=[{value:"Contents",id:"contents",level:2},{value:"Config not saving",id:"config-not-saving",level:2},{value:"Permission denied or read-only filesystem (EACCES, EROFS)",id:"permission-denied-or-read-only-filesystem-eacces-erofs",level:3},{value:"Kubernetes ConfigMap mount is read-only",id:"kubernetes-configmap-mount-is-read-only",level:3},{value:"SELinux or AppArmor blocks the write",id:"selinux-or-apparmor-blocks-the-write",level:3},{value:"Backup step fails so save aborts",id:"backup-step-fails-so-save-aborts",level:3},{value:"Save button is missing or returns 403 Forbidden",id:"save-button-is-missing-or-returns-403-forbidden",level:3},{value:"Save unavailable on Vercel, Netlify or other static hosts",id:"save-unavailable-on-vercel-netlify-or-other-static-hosts",level:3},{value:"/config-manager/save returns 404 or HTML",id:"config-managersave-returns-404-or-html",level:3},{value:""Invalid filename" when saving a sub-page",id:"invalid-filename-when-saving-a-sub-page",level:3},{value:""Cannot save to an external URL"",id:"cannot-save-to-an-external-url",level:3},{value:"Saved successfully but the UI shows the old config",id:"saved-successfully-but-the-ui-shows-the-old-config",level:3},{value:"Container crashes or restart loop after saving (3.1.0 and 3.1.1 only)",id:"container-crashes-or-restart-loop-after-saving-310-and-311-only",level:3},{value:"Intentionally read-only mode",id:"intentionally-read-only-mode",level:3},{value:"Refused to Connect in Modal or Workspace View",id:"refused-to-connect-in-modal-or-workspace-view",level:2},{value:"NGINX",id:"nginx",level:3},{value:"Caddy",id:"caddy",level:3},{value:"Apache",id:"apache",level:3},{value:"LightHttpd",id:"lighthttpd",level:3},{value:"404 / Routing issues",id:"404--routing-issues",level:2},{value:"404 On Static Hosting",id:"404-on-static-hosting",level:3},{value:"404 after Launch from Mobile Home Screen",id:"404-after-launch-from-mobile-home-screen",level:3},{value:"404 On Multi-Page Apps",id:"404-on-multi-page-apps",level:3},{value:"Dashy hosted at a sub-path (e.g. example.com/dashy)",id:"dashy-hosted-at-a-sub-path-eg-examplecomdashy",level:3},{value:"Sub-pages",id:"sub-pages",level:2},{value:"Sub-page shows "Unable to find config for ..."",id:"sub-page-shows-unable-to-find-config-for-",level:3},{value:"Old bookmark from before an upgrade",id:"old-bookmark-from-before-an-upgrade",level:4},{value:"The page was renamed or removed",id:"the-page-was-renamed-or-removed",level:4},{value:"The path points at an unreachable file",id:"the-path-points-at-an-unreachable-file",level:4},{value:"Page name literally "Main"",id:"page-name-literally-main",level:4},{value:"Service worker is serving a stale app",id:"service-worker-is-serving-a-stale-app",level:4},{value:"Sub-page missing from nav, or won't open when clicked",id:"sub-page-missing-from-nav-or-wont-open-when-clicked",level:3},{value:"Sub-page ignores its theme, layout or appConfig",id:"sub-page-ignores-its-theme-layout-or-appconfig",level:3},{value:"Sub-config files return 404",id:"sub-config-files-return-404",level:3},{value:"Remote Config Not Loading",id:"remote-config-not-loading",level:3},{value:"Build & memory errors",id:"build--memory-errors",level:2},{value:"Yarn Error",id:"yarn-error",level:3},{value:"yarn build fails inside the container",id:"yarn-build-fails-inside-the-container",level:3},{value:"High CPU or RAM Usage on Startup",id:"high-cpu-or-ram-usage-on-startup",level:3},{value:"Ineffective mark-compacts near heap limit Allocation failed",id:"ineffective-mark-compacts-near-heap-limit-allocation-failed",level:3},{value:"Command failed with signal "SIGKILL"",id:"command-failed-with-signal-sigkill",level:3},{value:"Node Sass does not yet support your current environment",id:"node-sass-does-not-yet-support-your-current-environment",level:3},{value:"Unreachable Code Error",id:"unreachable-code-error",level:3},{value:"Error: Cannot find module './_baseValues'",id:"error-cannot-find-module-_basevalues",level:3},{value:"Auth & OIDC",id:"auth--oidc",level:2},{value:"Auth Validation Error: "should be object"",id:"auth-validation-error-should-be-object",level:3},{value:"Keycloak Redirect Error",id:"keycloak-redirect-error",level:3},{value:"OIDC or Keycloak failure on numeric client IDs",id:"oidc-or-keycloak-failure-on-numeric-client-ids",level:3},{value:"Redirect loop after login",id:"redirect-loop-after-login",level:3},{value:"invalid_redirect_uri",id:"invalid_redirect_uri",level:3},{value:"Login works in the browser but the dashboard refuses to save anything (403)",id:"login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",level:3},{value:"Logged in but no admin controls",id:"logged-in-but-no-admin-controls",level:3},{value:"Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"",id:"login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",level:3},{value:"Sign-out leaves you stuck on Authentik",id:"sign-out-leaves-you-stuck-on-authentik",level:3},{value:"Untrusted certificate from Authentik",id:"untrusted-certificate-from-authentik",level:3},{value:"Numeric client_id getting truncated",id:"numeric-client_id-getting-truncated",level:3},{value:"Docker & image issues",id:"docker--image-issues",level:2},{value:"App Not Starting After Update to 2.0.4",id:"app-not-starting-after-update-to-204",level:3},{value:"Mount Type Mismatch",id:"mount-type-mismatch",level:3},{value:"DockerHub toomanyrequests",id:"dockerhub-toomanyrequests",level:3},{value:"Solution 1 - Use an alternate container registry",id:"solution-1---use-an-alternate-container-registry",level:4},{value:"Solution 2 - Increase your rate limits",id:"solution-2---increase-your-rate-limits",level:4},{value:"Old image tags fail to pull",id:"old-image-tags-fail-to-pull",level:3},{value:"Healthcheck Failing in Docker",id:"healthcheck-failing-in-docker",level:3},{value:"SSL-enabled Dashy",id:"ssl-enabled-dashy",level:4},{value:"Custom port",id:"custom-port",level:4},{value:"Container is unhealthy past the grace period",id:"container-is-unhealthy-past-the-grace-period",level:4},{value:"Docker Login Fails on Ubuntu",id:"docker-login-fails-on-ubuntu",level:3},{value:"Styles and Assets not Updating",id:"styles-and-assets-not-updating",level:2},{value:"Config Validation Errors",id:"config-validation-errors",level:2},{value:"Invalid Host Header while running through ngrok",id:"invalid-host-header-while-running-through-ngrok",level:2},{value:"Warnings in the Console during deploy",id:"warnings-in-the-console-during-deploy",level:2},{value:"Status Checks Failing",id:"status-checks-failing",level:2},{value:"Widgets",id:"widgets",level:2},{value:"Widget Errors",id:"widget-errors",level:3},{value:"Find Error Message",id:"find-error-message",level:4},{value:"Check Config",id:"check-config",level:4},{value:"Timeout Error",id:"timeout-error",level:4},{value:"CORS error",id:"cors-error",level:4},{value:"More Info",id:"more-info",level:4},{value:"Widget CORS Errors",id:"widget-cors-errors",level:3},{value:"Option 1 - Ensure Correct Protocol",id:"option-1---ensure-correct-protocol",level:4},{value:"Option 2 - Set Headers",id:"option-2---set-headers",level:4},{value:"Option 3 - Proxying Request",id:"option-3---proxying-request",level:4},{value:"Option 4 - Use a plugin",id:"option-4---use-a-plugin",level:4},{value:"CORS Proxy connect ECONNREFUSED ... or getaddrinfo ENOTFOUND ...",id:"cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound-",level:3},{value:"CORS Proxy Target-URL host '...' is blocked / must use http:// or https://",id:"cors-proxy-target-url-host--is-blocked--must-use-http-or-https",level:3},{value:"Widget Shows Error Incorrectly",id:"widget-shows-error-incorrectly",level:3},{value:"Weather Forecast Widget 401",id:"weather-forecast-widget-401",level:3},{value:"Widget Displaying Inaccurate Data",id:"widget-displaying-inaccurate-data",level:3},{value:"Public IP Widget not working for ipinfo or ipquery providers",id:"public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",level:3},{value:"Font Awesome Icons not Displaying",id:"font-awesome-icons-not-displaying",level:2},{value:"Copy to Clipboard not Working",id:"copy-to-clipboard-not-working",level:2},{value:"How-To / Reference",id:"how-to--reference",level:2},{value:"How to Reset Local Settings",id:"how-to-reset-local-settings",level:3},{value:"How to make a bug report",id:"how-to-make-a-bug-report",level:3},{value:"Step 1 - Where to open issues",id:"step-1---where-to-open-issues",level:4},{value:"Step 2 - Checking it's not already covered",id:"step-2---checking-its-not-already-covered",level:4},{value:"Step 3 - Describe the Issue",id:"step-3---describe-the-issue",level:4},{value:"Step 4 - Provide Supporting Info",id:"step-4---provide-supporting-info",level:4},{value:"Step 5 - Fix Released",id:"step-5---fix-released",level:4},{value:"How-To Open Browser Console",id:"how-to-open-browser-console",level:3},{value:"Git Contributions not Displaying",id:"git-contributions-not-displaying",level:3}];function h(e){const n={a:"a",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",kbd:"kbd",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"troubleshooting",children:"Troubleshooting"})}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.em,{children:(0,r.jsx)(n.strong,{children:"This document contains common problems and their solutions."})}),(0,r.jsx)(n.br,{}),"\nPlease ensure your issue isn't listed here, before opening a new ticket."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"Found something not listed here? Consider adding it, to help other users."})}),"\n"]}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#config-not-saving",children:"Config not saving"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#permission-denied-or-read-only-filesystem-eacces-erofs",children:"Permission denied or read-only filesystem"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#kubernetes-configmap-mount-is-read-only",children:"Kubernetes ConfigMap mount is read-only"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#selinux-or-apparmor-blocks-the-write",children:"SELinux or AppArmor blocks the write"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#backup-step-fails-so-save-aborts",children:"Backup step fails so save aborts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#save-button-is-missing-or-returns-403-forbidden",children:"Save button is missing or returns 403"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#save-unavailable-on-vercel-netlify-or-other-static-hosts",children:"Save unavailable on Vercel, Netlify or other static hosts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#config-managersave-returns-404-or-html",children:"/config-manager/save returns 404 or HTML"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid-filename-when-saving-a-sub-page",children:'"Invalid filename" when saving a sub-page'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cannot-save-to-an-external-url",children:'"Cannot save to an external URL"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#saved-successfully-but-the-ui-shows-the-old-config",children:"Saved successfully but the UI shows the old config"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-crashes-or-restart-loop-after-saving-310-and-311-only",children:"Container crashes or restart loop after saving"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#intentionally-read-only-mode",children:"Intentionally read-only mode"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#refused-to-connect-in-modal-or-workspace-view",children:"Refused to Connect in Web Content View"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#404--routing-issues",children:"404 / Routing issues"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-on-static-hosting",children:"404 On Static Hosting"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-after-launch-from-mobile-home-screen",children:"404 from Mobile Home Screen"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-on-multi-page-apps",children:"404 On Multi-Page Apps"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dashy-hosted-at-a-sub-path-eg-examplecomdashy",children:"Dashy hosted at a sub-path"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#sub-pages",children:"Sub-pages"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-shows-unable-to-find-config-for",children:'Sub-page shows "Unable to find config for ..."'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-missing-from-nav-or-wont-open-when-clicked",children:"Sub-page missing from nav, or won't open when clicked"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-ignores-its-theme-layout-or-appconfig",children:"Sub-page ignores its theme, layout or appConfig"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-config-files-return-404",children:"Sub-config files return 404"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#remote-config-not-loading",children:"Remote Config Not Loading"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#build--memory-errors",children:"Build & memory errors"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#yarn-error",children:"Yarn Build or Run Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsxs)(n.a,{href:"#yarn-build-fails-inside-the-container",children:[(0,r.jsx)(n.code,{children:"yarn build"})," fails inside the container"]})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Heap limit Allocation failed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#command-failed-with-signal-sigkill",children:'Command failed with signal "SIGKILL"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#node-sass-does-not-yet-support-your-current-environment",children:"Node Sass unsupported environment"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#unreachable-code-error",children:"Unreachable Code Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#error-cannot-find-module-_basevalues",children:"Cannot find module './_baseValues'"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#auth--oidc",children:"Auth & OIDC"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#auth-validation-error-should-be-object",children:'Auth Validation Error: "should be object"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#keycloak-redirect-error",children:"Keycloak Redirect Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#oidc-or-keycloak-failure-on-numeric-client-ids",children:"OIDC or Keycloak failure on numeric client IDs"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#redirect-loop-after-login",children:"Redirect loop after login"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid_redirect_uri",children:"invalid_redirect_uri"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",children:"Login works in the browser but the dashboard refuses to save anything (403)"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#logged-in-but-no-admin-controls",children:"Logged in but no admin controls"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",children:'Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sign-out-leaves-you-stuck-on-authentik",children:"Sign-out leaves you stuck on Authentik"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#untrusted-certificate-from-authentik",children:"Untrusted certificate from Authentik"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#numeric-client_id-getting-truncated",children:"Numeric client_id getting truncated"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#docker--image-issues",children:"Docker & image issues"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#app-not-starting-after-update-to-204",children:"App Not Starting After Update to 2.0.4"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mount-type-mismatch",children:"Mount Type Mismatch"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dockerhub-toomanyrequests",children:"DockerHub toomanyrequests"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#old-image-tags-fail-to-pull",children:"Old image tags fail to pull"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthcheck-failing-in-docker",children:"Healthcheck Failing in Docker"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#docker-login-fails-on-ubuntu",children:"Docker Login Fails on Ubuntu"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#styles-and-assets-not-updating",children:"Styles and Assets not Updating"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#config-validation-errors",children:"Config Validation Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid-host-header-while-running-through-ngrok",children:"Ngrok Invalid Host Headers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#warnings-in-the-console-during-deploy",children:"Warnings in the Console during deploy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#status-checks-failing",children:"Status Checks Failing"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#widgets",children:"Widgets"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-errors",children:"Diagnosing Widget Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-cors-errors",children:"Fixing Widget CORS Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound",children:"CORS Proxy connect ECONNREFUSED or ENOTFOUND"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cors-proxy-target-url-host--is-blocked--must-use-http-or-https",children:"CORS Proxy Target-URL host blocked or scheme rejected"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-shows-error-incorrectly",children:"Widget Shows Error Incorrectly"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather-forecast-widget-401",children:"Weather Forecast Widget 401"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-displaying-inaccurate-data",children:"Widget Displaying Inaccurate Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",children:"Public IP Widget not working for ipinfo or ipquery providers"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#font-awesome-icons-not-displaying",children:"Font Awesome Icons not Displaying"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#copy-to-clipboard-not-working",children:"Copy to Clipboard not Working"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#how-to--reference",children:"How-To / Reference"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-reset-local-settings",children:"How to Reset Local Settings"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-make-a-bug-report",children:"How to make a bug report"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"How-To Open Browser Console"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#git-contributions-not-displaying",children:"Git Contributions not Displaying"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"config-not-saving",children:"Config not saving"}),"\n",(0,r.jsxs)(n.p,{children:["There should be an error message, explaining the reason the config save failed. First check ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," (",(0,r.jsx)(n.kbd,{children:"F12"})," --\x3e Console), and then your server-side logs in the terminal. Then, see the following sections for solutions to each possible error."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{id:"config-not-saving-on-vercel--netlify--cdn"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-1-unable-to-call-save-endpoint-from-cdnstatic-server"}),"\n",(0,r.jsx)(n.a,{id:"unable-to-write-confyml-eacces-permission-denied"}),"\n",(0,r.jsx)(n.a,{id:"permission-denied-saving-config"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-2-unable-to-save"}),"\n",(0,r.jsx)(n.a,{id:"config-not-updating"}),"\n",(0,r.jsx)(n.a,{id:"config-still-not-updating"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-3-saved-but-not-updating"})]}),"\n",(0,r.jsx)(n.h3,{id:"permission-denied-or-read-only-filesystem-eacces-erofs",children:"Permission denied or read-only filesystem (EACCES, EROFS)"}),"\n",(0,r.jsxs)(n.p,{children:["The container can't write to your ",(0,r.jsx)(n.code,{children:"conf.yml"})," or its directory. Almost always an ownership mismatch: the host directory belongs to a different uid than the one Dashy runs as inside the container. Less commonly a read-only mount or an over-strict file mode."]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy runs as UID=1000 (default non-root node user). You can see this by running ",(0,r.jsx)(n.code,{children:"docker exec -it dashy id"}),". Then, check who owns the user-data directory, with: ",(0,r.jsx)(n.code,{children:"docker exec -it dashy ls -la /app/user-data"})," - if it's not ",(0,r.jsx)(n.code,{children:"1000"})," then that's the issue. And the solution is just to run ",(0,r.jsx)(n.code,{children:"sudo chown -R 1000:1000 /path/to/your/user-data"})," to set the right owner."]}),"\n",(0,r.jsx)(n.p,{children:"Fixes:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Hand the directory to uid 1000"})," (recommended). Keeps the container running as a non-root user, which is how Dashy is built to run ",(0,r.jsx)(n.code,{children:"sudo chown -R 1000:1000 /path/to/your/user-data"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Run the container as your own user"})," if ",(0,r.jsx)(n.code,{children:"chown"})," isn't practical (multi-user hosts, NAS appliances, host directories you don't want relabelled). Add ",(0,r.jsx)(n.code,{children:"--user $(id -u):$(id -g)"})," to ",(0,r.jsx)(n.code,{children:"docker run"}),", or set ",(0,r.jsx)(n.code,{children:'user: "1000:1000"'})," (or your host uid:gid) on the service in ",(0,r.jsx)(n.code,{children:"docker-compose.yml"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Loosen a single-file mount"})," if its mode is ",(0,r.jsx)(n.code,{children:"444"}),". Narrow case, only fixes that one symptom: ",(0,r.jsx)(n.code,{children:"chmod 644 /path/to/conf.yml"})]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Common mistakes"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Using uid/gid 1001."})," A common guess on Synology, Unraid and similar where 1001 is the host's first user. Dashy's container is 1000, not 1001."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsxs)(n.strong,{children:[(0,r.jsx)(n.code,{children:"chmod"})," alone for a UID mismatch."]})," Loosens permissions but doesn't change who owns the file. You need ",(0,r.jsx)(n.code,{children:"chown"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsxs)(n.strong,{children:[(0,r.jsx)(n.code,{children:"chmod -R 777"})," or ",(0,r.jsx)(n.code,{children:"775"}),"."]})," Works as a workaround, masks the real problem, weakens security. Use ",(0,r.jsx)(n.code,{children:"chown"})," to the right uid instead."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Other gotchas:"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Named Docker volumes"})," (created with ",(0,r.jsx)(n.code,{children:"docker volume create"}),") inherit ownership from whatever first writes to them. If an older container set them up as root, the diagnose step will show that. Recreate the volume or ",(0,r.jsx)(n.code,{children:"chown"})," the underlying directory under ",(0,r.jsx)(n.code,{children:"/var/lib/docker/volumes/"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"macOS hosts"})," rarely hit this. Docker Desktop transparently maps host uid to container uid through its VM. If saves are failing on macOS, look elsewhere first."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Storage layers that ignore POSIX permissions"})," (some NAS app-data folders use FUSE, SMB or overlay mounts where ",(0,r.jsx)(n.code,{children:"chmod"})," and ",(0,r.jsx)(n.code,{children:"chgrp"})," are silent no-ops). Bind-mount user-data from a native filesystem path instead."]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"kubernetes-configmap-mount-is-read-only",children:"Kubernetes ConfigMap mount is read-only"}),"\n",(0,r.jsxs)(n.p,{children:["If you've mounted your ",(0,r.jsx)(n.code,{children:"conf.yml"})," from a ConfigMap, writes will always fail with ",(0,r.jsx)(n.code,{children:"EROFS"})," regardless of UID. ConfigMap volumes are read-only by design. Either treat the ConfigMap as the source of truth and edit it directly (saves through the UI won't work), or use a writable volume type like a ",(0,r.jsx)(n.code,{children:"PersistentVolumeClaim"})," for ",(0,r.jsx)(n.code,{children:"user-data/"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"selinux-or-apparmor-blocks-the-write",children:"SELinux or AppArmor blocks the write"}),"\n",(0,r.jsxs)(n.p,{children:["If you're on RHEL/Fedora, or systems with SELinux or AppArmour, and you've confirmed permissions are fine, and container's UID matches the host owner, but you still see ",(0,r.jsx)(n.code,{children:"EACCES"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["For SELinux, add the ",(0,r.jsx)(n.code,{children:":Z"})," flag to your volume mount so Docker relabels it for the container (e.g. ",(0,r.jsx)(n.code,{children:"volumes: [ './user-data:/app/user-data:Z' ]"}),")"]}),"\n",(0,r.jsxs)(n.p,{children:["For AppArmor, check ",(0,r.jsx)(n.code,{children:"dmesg"})," for ",(0,r.jsx)(n.code,{children:'apparmor="DENIED"'})," lines and adjust the profile. Disabling enforcement is a last resort."]}),"\n",(0,r.jsx)(n.h3,{id:"backup-step-fails-so-save-aborts",children:"Backup step fails so save aborts"}),"\n",(0,r.jsxs)(n.p,{children:["Before each save, Dashy backs up the current ",(0,r.jsx)(n.code,{children:"conf.yml"})," to ",(0,r.jsx)(n.code,{children:"user-data/config-backups/"}),". If that folder can't be written, the whole save aborts with ",(0,r.jsx)(n.code,{children:"Unable to backup conf.yml"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Two ways out:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Point ",(0,r.jsx)(n.code,{children:"BACKUP_DIR"})," at a writable path"]}),"\n",(0,r.jsxs)(n.li,{children:["Set ",(0,r.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS=true"})," to skip the backup step entirely"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"save-button-is-missing-or-returns-403-forbidden",children:"Save button is missing or returns 403 Forbidden"}),"\n",(0,r.jsxs)(n.p,{children:["Have you got auth setup? If so, make sure you are logged in as an admin, or set ",(0,r.jsx)(n.code,{children:"type: admin"})," to your user in ",(0,r.jsx)(n.code,{children:"conf.yml"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Beyond that, there's several other config options which prevent saving the config file, so if you didn't mean to add them, just remove from ",(0,r.jsx)(n.code,{children:"conf.yml"})]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.preventWriteToDisk: true"})})," disables disk save and the button"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.preventLocalSave: true"})}),' disables the "Local" save option']}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.disableConfiguration: true"})})," hides the editor entirely. ",(0,r.jsx)(n.code,{children:"disableConfigurationForNonAdmin: true"})," does the same just for non-admins."]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"save-unavailable-on-vercel-netlify-or-other-static-hosts",children:"Save unavailable on Vercel, Netlify or other static hosts"}),"\n",(0,r.jsx)(n.p,{children:'Updating source config file on static hosts is not possible, since they have no Node server, nor have write access to modify any files.\nThe "Local" save mode will still work (changes are just persisted in your browser), but the real solution is to copy/export the updated YAML and replace it in the source config file in your repo.'}),"\n",(0,r.jsxs)(n.p,{children:["Related: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1465",children:"#1465"}),"."]}),"\n",(0,r.jsxs)(n.h3,{id:"config-managersave-returns-404-or-html",children:[(0,r.jsx)(n.code,{children:"/config-manager/save"})," returns 404 or HTML"]}),"\n",(0,r.jsxs)(n.p,{children:["How are you running/serving Dashy?\nIf you've got a reverse proxy which only forwards specific path prefixes then maybe you're missing the ",(0,r.jsx)(n.code,{children:"/config-manager/*"})," API endpoints?\nCheck the failed request in the browser's Network tab. If the response is HTML (a proxy error page) or a plain 404, your proxy isn't routing the path. Add ",(0,r.jsx)(n.code,{children:"/config-manager/"})," to whatever you're forwarding, or simplify the rules so everything reaches Dashy."]}),"\n",(0,r.jsx)(n.p,{children:"Or if you're serving up the compiled Vue app directly, instead of using the Node server, then the endpoint won't be available."}),"\n",(0,r.jsx)(n.h3,{id:"invalid-filename-when-saving-a-sub-page",children:'"Invalid filename" when saving a sub-page'}),"\n",(0,r.jsxs)(n.p,{children:["The save endpoint rejects sub-page filenames with path separators or non-yaml extensions. Check the ",(0,r.jsx)(n.code,{children:"path:"})," value of the page in your ",(0,r.jsx)(n.code,{children:"pages:"})," block. It needs to be a plain basename like ",(0,r.jsx)(n.code,{children:"home.yml"}),", not ",(0,r.jsx)(n.code,{children:"pages/home.yml"})," or ",(0,r.jsx)(n.code,{children:"home.txt"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"cannot-save-to-an-external-url",children:'"Cannot save to an external URL"'}),"\n",(0,r.jsxs)(n.p,{children:["The sub-page you are editing is loaded from a remote URL. Dashy can't write back to that URL.\nYou will need to edit the file at it's origin yourself instead (click the Export to view the YAML).\nOr you could download the config to ",(0,r.jsx)(n.code,{children:"user-data/something.yml"}),", and update ",(0,r.jsx)(n.code,{children:"path:"})," to point to the local version."]}),"\n",(0,r.jsx)(n.h3,{id:"saved-successfully-but-the-ui-shows-the-old-config",children:"Saved successfully but the UI shows the old config"}),"\n",(0,r.jsx)(n.p,{children:"Two unrelated causes share this symptom:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Local storage overrides the file."})," Dashy lets users save settings locally in browser storage, which take priority over ",(0,r.jsx)(n.code,{children:"conf.yml"}),'. Open Dashy in incognito to confirm. If the changes appear there, clear local settings via Config menu > "Clear Local Settings".']}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Docker isn't picking up file changes."})," Some text editors save by replacing the inode, which breaks single-file bind mounts. Edit the file in place, or mount the parent directory rather than the single file. ",(0,r.jsx)(n.a,{href:"https://medium.com/@jonsbun/why-need-to-be-careful-when-mounting-single-files-into-a-docker-container-4f929340834",children:"More background"}),"."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.a,{id:"container-crashes-or-restart-loop-after-saving-config"})}),"\n",(0,r.jsx)(n.h3,{id:"container-crashes-or-restart-loop-after-saving-310-and-311-only",children:"Container crashes or restart loop after saving (3.1.0 and 3.1.1 only)"}),"\n",(0,r.jsxs)(n.p,{children:["If your container crashes or restart-loops right after clicking save, with logs like ",(0,r.jsx)(n.code,{children:"ERR_HTTP_HEADERS_SENT"})," or ",(0,r.jsx)(n.code,{children:"ERR_STREAM_WRITE_AFTER_END"}),", this was a known double-",(0,r.jsx)(n.code,{children:"res.end()"})," bug in 3.1.0 and 3.1.1. Fixed in v3.2.13 and later."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker pull lissy93/dashy:latest\ndocker compose up -d --force-recreate\n"})}),"\n",(0,r.jsx)(n.h3,{id:"intentionally-read-only-mode",children:"Intentionally read-only mode"}),"\n",(0,r.jsxs)(n.p,{children:['To hide the "Save to disk" UI for everyone, set ',(0,r.jsx)(n.code,{children:"appConfig.preventWriteToDisk: true"})," in ",(0,r.jsx)(n.code,{children:"conf.yml"}),". This is a UI-only flag \u2014 the ",(0,r.jsx)(n.code,{children:"/config-manager/save"})," server endpoint itself is gated by the configured auth method (",(0,r.jsx)(n.code,{children:"auth.users"})," with ",(0,r.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"}),", OIDC/Keycloak admin role, header-auth, etc.), so anyone unauthenticated or non-admin already can't save regardless of this flag. For Docker users, you can harden things further by mounting ",(0,r.jsx)(n.code,{children:"user-data"})," (or just ",(0,r.jsx)(n.code,{children:"conf.yml"}),") as read-only \u2014 the kernel will refuse the write even before the server tries."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsxs)(n.h2,{id:"refused-to-connect-in-modal-or-workspace-view",children:[(0,r.jsx)(n.code,{children:"Refused to Connect"})," in Modal or Workspace View"]}),"\n",(0,r.jsx)(n.p,{children:"This is not an issue with Dashy, but instead caused by the target app preventing direct access through embedded elements."}),"\n",(0,r.jsxs)(n.p,{children:["As defined in ",(0,r.jsx)(n.a,{href:"https://datatracker.ietf.org/doc/html/rfc7034",children:"RFC-7034"}),", for any web content to be accessed through an embedded element, it must have the ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Frame-Options",children:(0,r.jsx)(n.code,{children:"X-Frame-Options"})})," HTTP header set to ",(0,r.jsx)(n.code,{children:"ALLOW"}),". If you are getting a ",(0,r.jsx)(n.code,{children:"Refused to Connect"})," error then this header is set to ",(0,r.jsx)(n.code,{children:"DENY"})," (or ",(0,r.jsx)(n.code,{children:"SAMEORIGIN"})," and it's on a different host). Thankfully, for self-hosted services, it is easy to set these headers."]}),"\n",(0,r.jsx)(n.p,{children:"These settings are usually set in the config file for the web server that's hosting the target application, here are some examples of how to enable cross-origin access with common web servers:"}),"\n",(0,r.jsx)(n.h3,{id:"nginx",children:"NGINX"}),"\n",(0,r.jsxs)(n.p,{children:["In NGINX, you can use the ",(0,r.jsx)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_headers_module.html",children:(0,r.jsx)(n.code,{children:"add_header"})})," module within the app block."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"server {\n ...\n add_header X-Frame-Options SAMEORIGIN always;\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then reload with ",(0,r.jsx)(n.code,{children:"service nginx reload"})]}),"\n",(0,r.jsx)(n.h3,{id:"caddy",children:"Caddy"}),"\n",(0,r.jsxs)(n.p,{children:["In Caddy, you can use the ",(0,r.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/header",children:(0,r.jsx)(n.code,{children:"header"})})," directive."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"header {\n X-Frame-Options SAMEORIGIN\n}\n"})}),"\n",(0,r.jsx)(n.h3,{id:"apache",children:"Apache"}),"\n",(0,r.jsxs)(n.p,{children:["In Apache, you can use the ",(0,r.jsx)(n.a,{href:"https://httpd.apache.org/docs/current/mod/mod_headers.html",children:(0,r.jsx)(n.code,{children:"mod_headers"})})," module to set the ",(0,r.jsx)(n.code,{children:"X-Frame-Options"})," in your config file. This file is usually located somewhere like `/etc/apache2/httpd.conf"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'Header set X-Frame-Options: "ALLOW-FROM http://[dashy-location]/"\n'})}),"\n",(0,r.jsx)(n.h3,{id:"lighthttpd",children:"LightHttpd"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Content-Security-Policy: frame-ancestors 'self' https://[dashy-location]/\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"404--routing-issues",children:"404 / Routing issues"}),"\n",(0,r.jsx)(n.h3,{id:"404-on-static-hosting",children:"404 On Static Hosting"}),"\n",(0,r.jsx)(n.p,{children:"If you're seeing Dashy's 404 page on initial load/ refresh, and then the main app when you go back to Home, then this is likely caused by the Vue router, and if so can be fixed in one of two ways."}),"\n",(0,r.jsxs)(n.p,{children:["The first solution is to switch the routing mode, from HTML5 ",(0,r.jsx)(n.code,{children:"history"})," mode to ",(0,r.jsx)(n.code,{children:"hash"})," mode, by rebuilding Dashy with the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set."]}),"\n",(0,r.jsxs)(n.p,{children:["If this works, but you wish to continue using HTML5 history mode, then a bit of extra ",(0,r.jsx)(n.a,{href:"/docs/management#web-server-configuration",children:"server configuration"})," is required. This is explained in more detaail in the ",(0,r.jsx)(n.a,{href:"https://router.vuejs.org/guide/essentials/history-mode.html",children:"Vue Docs"}),". Once completed, you can then use ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=history"})," (the default) again, for neater URLs."]}),"\n",(0,r.jsx)(n.h3,{id:"404-after-launch-from-mobile-home-screen",children:"404 after Launch from Mobile Home Screen"}),"\n",(0,r.jsxs)(n.p,{children:['Similar to the above issue, if you get a 404 after using iOS and Android\'s "Add to Home Screen" feature, then this is caused by Vue router.\nIt can be fixed by rebuilding Dashy with the ',(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/628",children:"#628"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/762",children:"#762"})]}),"\n",(0,r.jsx)(n.h3,{id:"404-on-multi-page-apps",children:"404 On Multi-Page Apps"}),"\n",(0,r.jsxs)(n.p,{children:["Similar to above, if you get a 404 error when visiting a page directly on multi-page apps, then this can be fixed by rebuilding Dashy with the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set, then refreshing the page."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/670",children:"#670"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/763",children:"#763"})]}),"\n",(0,r.jsxs)(n.h3,{id:"dashy-hosted-at-a-sub-path-eg-examplecomdashy",children:["Dashy hosted at a sub-path (e.g. ",(0,r.jsx)(n.code,{children:"example.com/dashy"}),")"]}),"\n",(0,r.jsxs)(n.p,{children:["If the homepage works but sub-page links 404, or assets fail to load, it's almost always the base path.\nRebuild with ",(0,r.jsx)(n.code,{children:"BASE_URL"})," set to the sub-path - leading slash, no trailing slash:"]}),"\n",(0,r.jsxs)(n.p,{children:["Vue Router uses this to prefix every route. Without it, links resolve to ",(0,r.jsx)(n.code,{children:"/home/..."})," instead of ",(0,r.jsx)(n.code,{children:"/dashy/home/..."})," and skip your reverse proxy altogether. More detail in ",(0,r.jsx)(n.a,{href:"/docs/management#web-server-configuration",children:"web-server configuration"}),"."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"sub-pages",children:"Sub-pages"}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-shows-unable-to-find-config-for-",children:'Sub-page shows "Unable to find config for ..."'}),"\n",(0,r.jsxs)(n.p,{children:["This means Dashy couldn't match the URL segment to any entry in your ",(0,r.jsx)(n.code,{children:"pages:"})," list. A few causes:"]}),"\n",(0,r.jsx)(n.h4,{id:"old-bookmark-from-before-an-upgrade",children:"Old bookmark from before an upgrade"}),"\n",(0,r.jsxs)(n.p,{children:["Slugs are now trimmed more aggressively (e.g. ",(0,r.jsx)(n.code,{children:"\ud83c\udf10 Command Center"})," used to give ",(0,r.jsx)(n.code,{children:"-command-center"}),", now gives ",(0,r.jsx)(n.code,{children:"command-center"}),"). Re-bookmark from the nav, or update the URL by hand."]}),"\n",(0,r.jsx)(n.h4,{id:"the-page-was-renamed-or-removed",children:"The page was renamed or removed"}),"\n",(0,r.jsxs)(n.p,{children:["The URL no longer resolves to anything. Check the ",(0,r.jsx)(n.code,{children:"pages:"})," array in ",(0,r.jsx)(n.code,{children:"conf.yml"})," and confirm the sub-page still exists."]}),"\n",(0,r.jsx)(n.h4,{id:"the-path-points-at-an-unreachable-file",children:"The path points at an unreachable file"}),"\n",(0,r.jsxs)(n.p,{children:["If the sub-config YAML can't be fetched (404, CORS, auth), you'll see \"Unable to load config from ...\" instead. Verify the ",(0,r.jsx)(n.code,{children:"path:"})," is correct, reachable from the browser, and CORS-open if remote."]}),"\n",(0,r.jsx)(n.h4,{id:"page-name-literally-main",children:'Page name literally "Main"'}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.code,{children:"main"}),' is reserved in the URL scheme to mean "the root config". A page named "Main" becomes reachable at ',(0,r.jsx)(n.code,{children:"/home/main-page"})," (not ",(0,r.jsx)(n.code,{children:"/home/main"}),"). Rename the page if that's confusing."]}),"\n",(0,r.jsx)(n.h4,{id:"service-worker-is-serving-a-stale-app",children:"Service worker is serving a stale app"}),"\n",(0,r.jsxs)(n.p,{children:["Hard-refresh (",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"F5"}),") after a major upgrade. The PWA cache may still be pointing at old routes. Also see ",(0,r.jsx)(n.a,{href:"#styles-and-assets-not-updating",children:"Styles and Assets not Updating"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-missing-from-nav-or-wont-open-when-clicked",children:"Sub-page missing from nav, or won't open when clicked"}),"\n",(0,r.jsxs)(n.p,{children:["If page defined in ",(0,r.jsx)(n.code,{children:"pages:"})," is nowhere in the nav bar, or its link goes to a different page, then there's probably something wrong with the name you chose. Note that Dashy strips out any non-alphanumeric characters."]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Ensure each page does have a valid ",(0,r.jsx)(n.code,{children:"name"})," and ",(0,r.jsx)(n.code,{children:"path"})," field"]}),"\n",(0,r.jsx)(n.li,{children:"Check two pages don't have the same/similar name"}),"\n",(0,r.jsx)(n.li,{children:"Check each page has a name which has at least some alpha-numeric characters"}),"\n",(0,r.jsx)(n.li,{children:"Very long names could be being stripped/truncated"}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-ignores-its-theme-layout-or-appconfig",children:"Sub-page ignores its theme, layout or appConfig"}),"\n",(0,r.jsxs)(n.p,{children:["This is by design. Only the ",(0,r.jsx)(n.code,{children:"appConfig"})," from your root ",(0,r.jsx)(n.code,{children:"conf.yml"})," is used - theme, layout, iconSize, statusCheck, etc. are inherited globally so behaviour stays consistent across pages."]}),"\n",(0,r.jsxs)(n.p,{children:["If you put ",(0,r.jsx)(n.code,{children:"appConfig"})," inside a sub-page YAML, it's silently dropped on load. Move the values to the root config. See ",(0,r.jsx)(n.a,{href:"/docs/pages-and-sections#restrictions",children:"Restrictions"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"sub-config-files-return-404",children:"Sub-config files return 404"}),"\n",(0,r.jsxs)(n.p,{children:["If your ",(0,r.jsx)(n.code,{children:"conf.yml"})," references additional pages via ",(0,r.jsx)(n.code,{children:"pages:"})," and the browser shows ",(0,r.jsx)(n.code,{children:"Sub-config load failed: /something.yml"}),", the cause is almost always a Docker mount that only exposes ",(0,r.jsx)(n.code,{children:"conf.yml"})," and not the rest of ",(0,r.jsx)(n.code,{children:"user-data/"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you've done this:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n - ./my-conf.yml:/app/user-data/conf.yml\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Only ",(0,r.jsx)(n.code,{children:"conf.yml"})," exists inside the container. Anything it references (sub-configs, custom icons, fonts, CSS) isn't there."]}),"\n",(0,r.jsx)(n.p,{children:"Mount the directory instead:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n - ./user-data:/app/user-data\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Now everything in your ",(0,r.jsx)(n.code,{children:"user-data"})," folder is reachable at the web root. Same applies to ",(0,r.jsx)(n.code,{children:"docker run -v"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"remote-config-not-loading",children:"Remote Config Not Loading"}),"\n",(0,r.jsx)(n.p,{children:"If you've got a multi-page dashboard, and are hosting the additional config files yourself, then CORS rules will apply. A CORS error will look something like:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access to XMLHttpRequest at 'https://example.com/raw/my-config.yml' from origin 'http://dashy.local' has been blocked by CORS policy:\nNo 'Access-Control-Allow-Origin' header is present on the requested resource.\n"})}),"\n",(0,r.jsx)(n.p,{children:"The solution is to add the appropriate headers onto the target server, to allow it to accept requests from the origin where you're running Dashy."}),"\n",(0,r.jsxs)(n.p,{children:["If it is a remote service, that you do not have admin access to, then another option is to proxy the request. Either host your own, or use a publicly accessible service, like ",(0,r.jsx)(n.a,{href:"https://allorigins.win",children:"allorigins.win"}),", e.g: ",(0,r.jsx)(n.code,{children:"https://api.allorigins.win/raw?url=https://pastebin.com/raw/4tZpaJV5"}),". For git-based services specifically, there's ",(0,r.jsx)(n.a,{href:"https://raw.githack.com/",children:"raw.githack.com"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"build--memory-errors",children:"Build & memory errors"}),"\n",(0,r.jsx)(n.h3,{id:"yarn-error",children:"Yarn Error"}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1",children:"Issue #1"})]}),"\n",(0,r.jsxs)(n.p,{children:["First of all, check that you've got yarn installed correctly - see the ",(0,r.jsx)(n.a,{href:"https://classic.yarnpkg.com/en/docs/install",children:"yarn installation docs"})," for more info."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're getting an error about scenarios, then you've likely installed the wrong yarn... (you're ",(0,r.jsx)(n.a,{href:"https://github.com/yarnpkg/yarn/issues/2821",children:"not"})," the only one!). You can fix it by uninstalling, adding the correct repo, and reinstalling, for example, in Debian:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"sudo apt remove yarn"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add -"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:'echo "deb https://dl.yarnpkg.com/debian/ stable main" | sudo tee /etc/apt/sources.list.d/yarn.list'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"sudo apt update && sudo apt install yarn"})}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Alternatively, as a workaround, you have several options:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Try using ",(0,r.jsx)(n.a,{href:"https://www.npmjs.com/get-npm",children:"NPM"})," instead: So clone, cd, then run ",(0,r.jsx)(n.code,{children:"npm install"}),", ",(0,r.jsx)(n.code,{children:"npm run build"})," and ",(0,r.jsx)(n.code,{children:"npm start"})]}),"\n",(0,r.jsxs)(n.li,{children:["Try using ",(0,r.jsx)(n.a,{href:"https://www.docker.com/get-started",children:"Docker"})," instead, and all of the system setup and dependencies will already be taken care of. So from within the directory, just run ",(0,r.jsx)(n.code,{children:"docker build -t lissy93/dashy ."})," to build, and then use docker start to run the project, e.g: ",(0,r.jsx)(n.code,{children:"docker run -it -p 8080:8080 lissy93/dashy"})," (see the ",(0,r.jsx)(n.a,{href:"/docs/deployment#deploy-with-docker",children:"deploying docs"})," for more info)"]}),"\n"]}),"\n",(0,r.jsxs)(n.h3,{id:"yarn-build-fails-inside-the-container",children:[(0,r.jsx)(n.code,{children:"yarn build"})," fails inside the container"]}),"\n",(0,r.jsxs)(n.p,{children:["If you run ",(0,r.jsx)(n.code,{children:"docker exec yarn build"})," and get ",(0,r.jsx)(n.code,{children:"vite: not found"})," (or similar), it's because the published image ships only production dependencies. The build toolchain (vite, vue-tsc, sass, etc.) lives in ",(0,r.jsx)(n.code,{children:"devDependencies"})," and isn't installed in the runtime image."]}),"\n",(0,r.jsxs)(n.p,{children:["You almost certainly don't need to rebuild. Dashy's Express server reads ",(0,r.jsx)(n.code,{children:"user-data/conf.yml"})," on every request, so config changes show up on a page refresh, no rebuild required."]}),"\n",(0,r.jsxs)(n.p,{children:["If you genuinely need a fresh build (you've patched something in ",(0,r.jsx)(n.code,{children:"src/"}),"), do it on the host with ",(0,r.jsx)(n.code,{children:"yarn install && yarn build"}),", or build a custom image from a checkout of the repo."]}),"\n",(0,r.jsx)(n.h3,{id:"high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"}),"\n",(0,r.jsxs)(n.p,{children:["When the Dashy container first starts, it runs a Vue production build in parallel with the server. This is ",(0,r.jsx)(n.strong,{children:"a one-time cost per container start"}),", but it briefly uses around ",(0,r.jsx)(n.strong,{children:"1\u20131.5 GB of RAM and 100% of one CPU core"})," for anywhere from 30 seconds to several minutes (depending on host speed). On Pi-class hardware or VMs with less than 1 GB of RAM, this spike can be enough to lock up the host."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"To work around it:"})}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Allocate at least 1 GB of RAM to the container"})," - 2 GB is recommended on Raspberry Pi or low-powered VMs. Anything below 512 MB is unlikely to complete the first build."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Set explicit Docker resource limits"})," so the build can't starve other services on the same host:\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy:latest\n deploy:\n resources:\n limits:\n memory: 2g\n cpus: '1.5'\n"})}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Wait it out"})," - once the build completes, idle CPU drops to near zero and idle RAM is typically under 100 MB. If you watch ",(0,r.jsx)(n.code,{children:"docker stats"}),", you'll see the spike taper off."]}),"\n",(0,r.jsxs)(n.li,{children:["If the spike never tapers (i.e., Dashy stays at 100% CPU forever and never serves the page), see ",(0,r.jsx)(n.a,{href:"#ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Heap limit Allocation failed"})," below - that usually means the build was killed mid-way and is being retried."]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1585",children:"#1585"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/969",children:"#969"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1500",children:"#1500"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/877",children:"#877"})]}),"\n",(0,r.jsx)(n.h3,{id:"ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Ineffective mark-compacts near heap limit Allocation failed"}),"\n",(0,r.jsx)(n.p,{children:"If you see an error message, similar to:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"<--- Last few GCs ---\x3e\n\n[61:0x74533040] 229060 ms: Mark-sweep (reduce) 127.1 (236.9) -> 127.1 (137.4) MB, 5560.7 / 0.3 ms (average mu = 0.286, current mu = 0.011) allocation failure scavenge might not succeed\n\n<--- JS stacktrace ---\x3e\n\nFATAL ERROR: Reached heap limit Allocation failed - JavaScript heap out of memory\n"})}),"\n",(0,r.jsxs)(n.p,{children:["This is likely caused by insufficient memory allocation to the container. When the container first starts up, or has to rebuild, the memory usage spikes, and if there isn't enough memory, it may terminate. This can be specified with, for example: ",(0,r.jsx)(n.code,{children:"--memory=1024m"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Docker: Runtime options with Memory, CPUs, and GPUs"}),". For more context on what the spike is, see ",(0,r.jsx)(n.a,{href:"#high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"})," above."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/380",children:"#380"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/350",children:"#350"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/297",children:"#297"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/349",children:"#349"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/510",children:"#510"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/511",children:"#511"})," and ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/834",children:"#834"})]}),"\n",(0,r.jsx)(n.h3,{id:"command-failed-with-signal-sigkill",children:'Command failed with signal "SIGKILL"'}),"\n",(0,r.jsxs)(n.p,{children:["In Docker, this can be caused by not enough memory. When the container first starts up, or has to rebuild, the memory usage spikes, and so a larger allocation may be required. This can be specified with, for example: ",(0,r.jsx)(n.code,{children:"--memory=1024m"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Docker: Runtime options with Memory, CPUs, and GPUs"})]}),"\n",(0,r.jsxs)(n.p,{children:["See also ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/624",children:"#624"})]}),"\n",(0,r.jsx)(n.h3,{id:"node-sass-does-not-yet-support-your-current-environment",children:"Node Sass does not yet support your current environment"}),"\n",(0,r.jsxs)(n.p,{children:["Caused by node-sass's binaries being built for a for a different architecture\nTo fix this, just run: ",(0,r.jsx)(n.code,{children:"yarn rebuild node-sass"})]}),"\n",(0,r.jsx)(n.h3,{id:"unreachable-code-error",children:"Unreachable Code Error"}),"\n",(0,r.jsxs)(n.p,{children:["An error similar to: ",(0,r.jsx)(n.code,{children:"Fatal error in , line 0. Unreachable code, FailureMessage Object: 0xffe6c8ac. Illegal instruction (core dumped)"}),"\nIs related to a bug in a downstream package, see ",(0,r.jsx)(n.a,{href:"https://github.com/nodejs/docker-node/issues/1477",children:"nodejs/docker-node#1477"}),".\nUsually, updating your system and packages will resolve the issue."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/776",children:"#776"})]}),"\n",(0,r.jsx)(n.h3,{id:"error-cannot-find-module-_basevalues",children:"Error: Cannot find module './_baseValues'"}),"\n",(0,r.jsxs)(n.p,{children:["Clearing the cache should fix this: ",(0,r.jsx)(n.code,{children:"yarn cache clean"}),"\nIf the issue persists, remove (",(0,r.jsx)(n.code,{children:"rm -rf node_modules\\ yarn.lock"}),") and reinstall (",(0,r.jsx)(n.code,{children:"yarn"}),") node_modules"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"auth--oidc",children:"Auth & OIDC"}),"\n",(0,r.jsx)(n.h3,{id:"auth-validation-error-should-be-object",children:'Auth Validation Error: "should be object"'}),"\n",(0,r.jsxs)(n.p,{children:["In V 1.6.5 an update was made that in the future will become a breaking change. You will need to update you config to reflect this before V 2.0.0 is released. In the meantime, your previous config will continue to function normally, but you will see a validation warning. The change means that the structure of the ",(0,r.jsx)(n.code,{children:"appConfig.auth"})," object is now an object, which has a ",(0,r.jsx)(n.code,{children:"users"})," property."]}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/177",children:"this announcement"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"You can fix this by replacing:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"auth:\n- user: xxx\n hash: xxx\n"})}),"\n",(0,r.jsx)(n.p,{children:"with"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"auth:\n users:\n - user: xxx\n hash: xxx\n"})}),"\n",(0,r.jsx)(n.h3,{id:"keycloak-redirect-error",children:"Keycloak Redirect Error"}),"\n",(0,r.jsxs)(n.p,{children:["Check the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser's console output"}),", if you've not set any headers, you will likely see a CORS error here, which would be the source of the issue."]}),"\n",(0,r.jsxs)(n.p,{children:["You need to allow Dashy to make requests to Keycloak, and Keycloak to redirect to Dashy. The way you do this depends on how you're hosting these applications / which proxy you are using, and examples can be found in the ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Management Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example, add the access control header to Keycloak, like:"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin [URL-of Dashy]"})}),"\n",(0,r.jsxs)(n.p,{children:["Note that for requests that transport sensitive info like credentials, setting the accept header to a wildcard (",(0,r.jsx)(n.code,{children:"*"}),") is not allowed - see ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS#requests_with_credentials",children:"MDN Docs"}),", so you will need to specify the actual URL."]}),"\n",(0,r.jsxs)(n.p,{children:["You should also ensure that Keycloak is correctly configured, with a user, realm and application, and be sure that you have set a valid redirect URL in Keycloak (",(0,r.jsx)(n.a,{href:"https://user-images.githubusercontent.com/1862727/148599768-db4ee4f8-72c5-402d-8f00-051d999e6267.png",children:"screenshot"}),")."]}),"\n",(0,r.jsxs)(n.p,{children:["For more details on how to set headers, see the ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Example Headers"})," in the management docs, or reference the documentation for your proxy."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're running in Kubernetes, you will need to enable CORS ingress rules, see ",(0,r.jsx)(n.a,{href:"https://kubernetes.github.io/ingress-nginx/user-guide/nginx-configuration/annotations/#enable-cors",children:"docs"}),", e.g:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'nginx.ingress.kubernetes.io/cors-allow-origin: "https://dashy.example.com"\nnginx.ingress.kubernetes.io/enable-cors: "true"\n'})}),"\n",(0,r.jsx)(n.p,{children:"See also: #479, #409, #507, #491, #341, #520"}),"\n",(0,r.jsx)(n.h3,{id:"oidc-or-keycloak-failure-on-numeric-client-ids",children:"OIDC or Keycloak failure on numeric client IDs"}),"\n",(0,r.jsxs)(n.p,{children:["If your IdP rejects the login with an ",(0,r.jsx)(n.em,{children:'"invalid client"'})," / ",(0,r.jsx)(n.em,{children:'"client not found"'})," error, and your ",(0,r.jsx)(n.code,{children:"clientId"})," is a long numeric value, the cause is almost certainly YAML number parsing."]}),"\n",(0,r.jsxs)(n.p,{children:["YAML parses unquoted numeric tokens as Numbers, and JavaScript can't represent integers larger than 2^53 (~16 digits) without losing precision. So an unquoted numeric ",(0,r.jsx)(n.code,{children:"clientId"})," will be silently truncated (e.g. ",(0,r.jsx)(n.code,{children:"918756876419824312"})," \u2192 ",(0,r.jsx)(n.code,{children:"918756876419824300"}),"), or - for very large values - converted to scientific notation (e.g. ",(0,r.jsx)(n.code,{children:"9.187568764198242e+37"}),"), and the IdP will reject it."]}),"\n",(0,r.jsxs)(n.p,{children:["The fix is to wrap the ",(0,r.jsx)(n.code,{children:"clientId"})," in quotes in your ",(0,r.jsx)(n.code,{children:"conf.yml"})," so it gets parsed as a string:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'appConfig:\n auth:\n enableOidc: true\n oidc:\n clientId: "918756876419824312"\n endpoint: https://idp.example.com/\n'})}),"\n",(0,r.jsxs)(n.p,{children:["The same applies to ",(0,r.jsx)(n.code,{children:"auth.keycloak.clientId"}),". Dashy will print a warning in the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," when it detects a numeric ",(0,r.jsx)(n.code,{children:"clientId"}),", to help diagnose this."]}),"\n",(0,r.jsx)(n.p,{children:"See also: #1941"}),"\n",(0,r.jsx)(n.h3,{id:"redirect-loop-after-login",children:"Redirect loop after login"}),"\n",(0,r.jsxs)(n.p,{children:["Your ",(0,r.jsx)(n.code,{children:"endpoint"})," probably includes ",(0,r.jsx)(n.code,{children:".well-known/openid-configuration"}),". Drop everything from ",(0,r.jsx)(n.code,{children:".well-known"})," onwards"]}),"\n",(0,r.jsx)(n.h3,{id:"invalid_redirect_uri",children:"invalid_redirect_uri"}),"\n",(0,r.jsxs)(n.p,{children:["The redirect URI Authentik has registered for the provider doesn't exactly match the URL Dashy is being served from. Register both the bare URL and the trailing-slash version, and make sure the scheme matches (",(0,r.jsx)(n.code,{children:"http"})," vs ",(0,r.jsx)(n.code,{children:"https"}),")."]}),"\n",(0,r.jsx)(n.h3,{id:"login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",children:"Login works in the browser but the dashboard refuses to save anything (403)"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy's server is rejecting the id_token. Check Dashy's container logs for ",(0,r.jsx)(n.code,{children:"[auth-oidc] token verification failed"}),". Common causes:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Issuer mismatch"}),". Authentik is behind a reverse proxy that isn't sending ",(0,r.jsx)(n.code,{children:"X-Forwarded-Proto: https"}),", so its discovery document advertises ",(0,r.jsx)(n.code,{children:"http://"})," while you configured ",(0,r.jsx)(n.code,{children:"https://"})," in Dashy. Fix the proxy or set ",(0,r.jsx)(n.code,{children:"AUTHENTIK_HOST"}),"/",(0,r.jsx)(n.code,{children:"AUTHENTIK_LISTEN__TRUSTED_PROXY_CIDRS"})," on the Authentik containers"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Audience mismatch"}),". The ",(0,r.jsx)(n.code,{children:"aud"})," claim in the id_token is not ",(0,r.jsx)(n.code,{children:"dashy"}),". Confirm the provider's Client ID is exactly ",(0,r.jsx)(n.code,{children:"dashy"})," (no leading or trailing whitespace)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Dashy server can't reach Authentik"}),". The Dashy container fails to fetch the discovery document. Exec into the container and try ",(0,r.jsx)(n.code,{children:"wget -qO- https://auth.example.com/application/o/dashy/.well-known/openid-configuration"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Clock skew"}),". The middleware allows 30 seconds of drift. If a container's clock is further off than that, ",(0,r.jsx)(n.code,{children:"exp"}),"/",(0,r.jsx)(n.code,{children:"iat"})," checks fail"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"logged-in-but-no-admin-controls",children:"Logged in but no admin controls"}),"\n",(0,r.jsxs)(n.p,{children:["The id_token doesn't include the groups claim. Open browser devtools after logging in, find the call to ",(0,r.jsx)(n.code,{children:"/application/o/dashy/userinfo/"}),", and check the response. You should see a ",(0,r.jsx)(n.code,{children:"groups"})," array containing ",(0,r.jsx)(n.code,{children:"dashy-admins"}),". If not:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["The ",(0,r.jsx)(n.code,{children:"groups"})," scope mapping doesn't exist, or is not attached to the provider's property mappings"]}),"\n",(0,r.jsxs)(n.li,{children:["The user is not in the ",(0,r.jsx)(n.code,{children:"dashy-admins"})," group"]}),"\n",(0,r.jsxs)(n.li,{children:["The conf.yml is missing ",(0,r.jsx)(n.code,{children:"groups"})," from ",(0,r.jsx)(n.code,{children:"scope:"})," and Authentik is therefore not sending it"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",children:'Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"'}),"\n",(0,r.jsx)(n.p,{children:'The id_token came back without a username claim. Confirm the provider has "profile" and "email" in its scopes and the "Include claims in id_token" is on'}),"\n",(0,r.jsx)(n.h3,{id:"sign-out-leaves-you-stuck-on-authentik",children:"Sign-out leaves you stuck on Authentik"}),"\n",(0,r.jsx)(n.p,{children:"Dashy redirects to Authentik's end-session endpoint on logout. If Authentik's invalidation flow prompts for confirmation (the default), that's expected - click through it. To skip the prompt entirely, change the provider's invalidation flow to one without a consent stage."}),"\n",(0,r.jsx)(n.h3,{id:"untrusted-certificate-from-authentik",children:"Untrusted certificate from Authentik"}),"\n",(0,r.jsx)(n.p,{children:"Self-signed certs make Dashy's server-side fetch of the discovery document fail. Use a real cert (Let's Encrypt, or your homelab CA installed into the Dashy image) for the Authentik hostname."}),"\n",(0,r.jsx)(n.h3,{id:"numeric-client_id-getting-truncated",children:"Numeric client_id getting truncated"}),"\n",(0,r.jsx)(n.p,{children:"Don't use numeric-only client IDs. If you must, wrap the value in quotes in conf.yml so YAML treats it as a string"}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"docker--image-issues",children:"Docker & image issues"}),"\n",(0,r.jsx)(n.h3,{id:"app-not-starting-after-update-to-204",children:"App Not Starting After Update to 2.0.4"}),"\n",(0,r.jsxs)(n.p,{children:["Version 2.0.4 introduced changes to how the config is read, and the app is build. If you were previously mounting ",(0,r.jsx)(n.code,{children:"/public"})," as a volume, then this will over-write the build app, preventing it from starting. The solution is to just pass in the file(s) / sub-directories that you need. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n- /srv/dashy/conf.yml:/app/user-data/conf.yml\n- /srv/dashy/item-icons:/app/public/item-icons\n"})}),"\n",(0,r.jsx)(n.h3,{id:"mount-type-mismatch",children:"Mount Type Mismatch"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Error response from daemon: ... mount through procfd: not a directory:\nAre you trying to mount a directory onto a file (or vice-versa)?\n"})}),"\n",(0,r.jsx)(n.p,{children:"This means the host side and container side of your volume don't agree on whether the target is a file or a directory."}),"\n",(0,r.jsxs)(n.p,{children:["Recommended pattern: mount a host directory onto ",(0,r.jsx)(n.code,{children:"/app/user-data"}),". The directory must exist on the host and contain at least a ",(0,r.jsx)(n.code,{children:"conf.yml"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"mkdir -p ~/dashy-data\ncp /path/to/your/conf.yml ~/dashy-data/conf.yml\ndocker run -d -p 8080:8080 -v ~/dashy-data:/app/user-data lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If you'd rather mount a single file (",(0,r.jsx)(n.code,{children:"-v ~/conf.yml:/app/user-data/conf.yml"}),"), the host path must be a file that already exists, otherwise Docker creates a directory in its place and you'll see this error."]}),"\n",(0,r.jsxs)(n.h3,{id:"dockerhub-toomanyrequests",children:["DockerHub ",(0,r.jsx)(n.code,{children:"toomanyrequests"})]}),"\n",(0,r.jsx)(n.p,{children:"This situation relates to error messages similar to one of the following, returned when pulling, updating or running the Docker container from Docker Hub."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Continuing execution. Pulling image lissy93/dashy:release-1.6.0\nerror pulling image configuration: toomanyrequests\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"You have reached your pull rate limit. You may increase the limit by authenticating and upgrading: https://www.docker.com/increase-rate-limit\n"})}),"\n",(0,r.jsxs)(n.p,{children:["When DockerHub returns one of these errors, or a ",(0,r.jsx)(n.code,{children:"429"})," status, that means you've hit your rate limit. This was ",(0,r.jsx)(n.a,{href:"https://www.docker.com/blog/scaling-docker-to-serve-millions-more-developers-network-egress/",children:"introduced"})," last year, and prevents unauthenticated or free users from running docker pull more than 100 times per 6 hours.\nYou can ",(0,r.jsx)(n.a,{href:"https://www.docker.com/blog/checking-your-current-docker-pull-rate-limits-and-status/",children:"check your rate limit status"})," by looking for the ",(0,r.jsx)(n.code,{children:"ratelimit-remaining"})," header in any DockerHub responses."]}),"\n",(0,r.jsx)(n.h4,{id:"solution-1---use-an-alternate-container-registry",children:"Solution 1 - Use an alternate container registry"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Dashy is also available through GHCR, which at present does not have any hard limits. Just use ",(0,r.jsx)(n.code,{children:"docker pull ghcr.io/lissy93/dashy:latest"})," to fetch the image"]}),"\n",(0,r.jsxs)(n.li,{children:["You can also build the image from source, by cloning the repo, and running ",(0,r.jsx)(n.code,{children:"docker build -t dashy ."})," or use the pre-made docker compose"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"solution-2---increase-your-rate-limits",children:"Solution 2 - Increase your rate limits"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Logging in to DockerHub will increase your rate limit from 100 requests to 200 requests per 6 hour period"}),"\n",(0,r.jsx)(n.li,{children:"Upgrading to a Pro for $5/month will increase your image requests to 5,000 per day, and any plans above have no rate limits"}),"\n",(0,r.jsx)(n.li,{children:"Since rate limits are counted based on your IP address, proxying your requests, or using a VPN may work"}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"old-image-tags-fail-to-pull",children:"Old image tags fail to pull"}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.code,{children:"docker pull"})," returns ",(0,r.jsx)(n.code,{children:"manifest unknown"})," or ",(0,r.jsx)(n.code,{children:"manifest for lissy93/dashy:arm32v7 not found"}),", the cause is a stale architecture-specific tag in your compose file or run command. Tags like ",(0,r.jsx)(n.code,{children:":arm32v7"}),", ",(0,r.jsx)(n.code,{children:":arm64v8"}),", and ",(0,r.jsx)(n.code,{children:":multi-arch"})," are no longer published."]}),"\n",(0,r.jsxs)(n.p,{children:["The ",(0,r.jsx)(n.code,{children:":latest"})," tag is now multi-arch and works on amd64, arm64, and arm/v7 (Raspberry Pi 2 and up) without you having to pick a variant. Just use:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"image: lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Docker fetches the right architecture for your host automatically. To pin a version, use a semver tag, e.g. ",(0,r.jsx)(n.code,{children:"lissy93/dashy:3.2.14"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"healthcheck-failing-in-docker",children:"Healthcheck Failing in Docker"}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.code,{children:"docker ps"})," shows the Dashy container as ",(0,r.jsx)(n.code,{children:"unhealthy"}),", the periodic healthcheck (",(0,r.jsx)(n.code,{children:"node services/healthcheck.js"}),", run every 5 minutes by default) couldn't reach the local server."]}),"\n",(0,r.jsx)(n.h4,{id:"ssl-enabled-dashy",children:"SSL-enabled Dashy"}),"\n",(0,r.jsxs)(n.p,{children:["The healthcheck reads the same cert paths as the main server (",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-pub.pem"})," and ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-priv.key"}),") to detect whether to probe HTTPS or HTTP. If you've mounted certs at non-default paths via ",(0,r.jsx)(n.code,{children:"SSL_PUB_KEY_PATH"})," / ",(0,r.jsx)(n.code,{children:"SSL_PRIV_KEY_PATH"}),", the healthcheck will pick those up automatically as long as those env vars are set in the container environment (not just at run time)."]}),"\n",(0,r.jsx)(n.h4,{id:"custom-port",children:"Custom port"}),"\n",(0,r.jsxs)(n.p,{children:["If you've set ",(0,r.jsx)(n.code,{children:"PORT"})," to override the default 8080, the healthcheck honors the same env var, so it should work without changes. Make sure ",(0,r.jsx)(n.code,{children:"PORT"})," is set in the container environment, not just in the host-side Docker port mapping."]}),"\n",(0,r.jsx)(n.h4,{id:"container-is-unhealthy-past-the-grace-period",children:"Container is unhealthy past the grace period"}),"\n",(0,r.jsxs)(n.p,{children:["The healthcheck has a 20s ",(0,r.jsx)(n.code,{children:"start-period"})," after which failures start counting. The image is prebuilt, so startup is just ",(0,r.jsx)(n.code,{children:"node server.js"})," binding to a port - fast even on a Pi. If the container is still ",(0,r.jsx)(n.code,{children:"unhealthy"})," past the grace period, the server has likely crashed. Check ",(0,r.jsx)(n.code,{children:"docker logs "})," for the real error (usually a malformed ",(0,r.jsx)(n.code,{children:"conf.yml"})," or a missing ",(0,r.jsx)(n.code,{children:"user-data"})," mount)."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1410",children:"#1410"})]}),"\n",(0,r.jsx)(n.h3,{id:"docker-login-fails-on-ubuntu",children:"Docker Login Fails on Ubuntu"}),"\n",(0,r.jsxs)(n.p,{children:["Run ",(0,r.jsx)(n.code,{children:"sudo apt install gnupg2 pass && gpg2 -k"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"styles-and-assets-not-updating",children:"Styles and Assets not Updating"}),"\n",(0,r.jsxs)(n.p,{children:["If you find that your styles and other visual assets work when visiting ",(0,r.jsx)(n.code,{children:"ip:port"})," by not ",(0,r.jsx)(n.code,{children:"dashy.domain.com"}),", then this is usually caused by caching. In your browser, do a hard-refresh (",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"F5"}),"). If you use Cloudflare, then you can clear the cache through the management console, or set the cache level to Bypass for certain files, under the Rules tab."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"config-validation-errors",children:"Config Validation Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The configuration file is validated against ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"Dashy's Schema"})," using AJV."]}),"\n",(0,r.jsxs)(n.p,{children:["First, check that your syntax is valid, using ",(0,r.jsx)(n.a,{href:"https://codebeautify.org/yaml-validator/",children:"YAML Validator"})," or ",(0,r.jsx)(n.a,{href:"https://codebeautify.org/jsonvalidator",children:"JSON Validator"}),". If the issue persists, then take a look at the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"schema"}),", and verify that the field you are trying to add/ modify matches the required format. You can also use ",(0,r.jsx)(n.a,{href:"https://www.jsonschemavalidator.net/s/JFUj7X9J",children:"this tool"})," to validate your JSON config against the schema, or run ",(0,r.jsx)(n.code,{children:"yarn validate-config"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you're trying to use a recently released feature, and are getting a warning, this is likely because you've not yet updated the the current latest version of Dashy."}),"\n",(0,r.jsx)(n.p,{children:"If the issue still persists, you should raise an issue."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"invalid-host-header-while-running-through-ngrok",children:"Invalid Host Header while running through ngrok"}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-host-header",children:"-host-header"})," flag, e.g. ",(0,r.jsx)(n.code,{children:'ngrok http 8080 -host-header="localhost:8080"'})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"warnings-in-the-console-during-deploy",children:"Warnings in the Console during deploy"}),"\n",(0,r.jsx)(n.p,{children:"Please acknowledge the difference between errors and warnings before raising an issue about messages in the console. It's not unusual to see warnings about a new version of a certain package being available, an asset bundle bing oversized or a service worker not yet having a cache. These shouldn't have any impact on the running application, so please don't raise issues about these unless it directly relates to a bug or issue you're experiencing. Errors on the other hand should not appear in the console, and they are worth looking into further."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"status-checks-failing",children:"Status Checks Failing"}),"\n",(0,r.jsx)(n.p,{children:"If you're using status checks, and despite a given service being online, the check is displaying an error, there are a couple of things you can look at:"}),"\n",(0,r.jsxs)(n.p,{children:["If your service requires requests to include any authorization in the headers, then use the ",(0,r.jsx)(n.code,{children:"statusCheckHeaders"})," property, as described in the ",(0,r.jsx)(n.a,{href:"/docs/status-indicators#setting-custom-headers",children:"docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are still having issues, it may be because your target application is blocking requests from Dashy's IP. This is a ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"CORS error"}),", and can be fixed by setting the headers on your target app, to include:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If the URL you are checking has an unsigned certificate, or is not using HTTPS, then you may need to disable the rejection of insecure requests. This can be done by setting ",(0,r.jsx)(n.code,{children:"statusCheckAllowInsecure"})," to true for a given item."]}),"\n",(0,r.jsxs)(n.p,{children:["If your service is online, but responds with a status code that is not in the 2xx range, then you can use ",(0,r.jsx)(n.code,{children:"statusCheckAcceptCodes"})," to set an accepted status code."]}),"\n",(0,r.jsxs)(n.p,{children:["If you get an error, like ",(0,r.jsx)(n.code,{children:"Service Unavailable: Server resulted in a fatal error"}),", even when it's definitely online, this is most likely caused by missing the protocol. Don't forget to include ",(0,r.jsx)(n.code,{children:"https://"})," (or whatever protocol) before the URL, and ensure that if needed, you've specified the port."]}),"\n",(0,r.jsxs)(n.p,{children:["Running Dashy in HOST network mode, instead of BRIDGE will allow status check access to other services in HOST mode. For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/445",children:"#445"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you have firewall rules configured, then ensure that they don't prevent Dashy from making requests to the other services you are trying to access."}),"\n",(0,r.jsxs)(n.p,{children:["Currently, the status check needs a page to be rendered, so if this URL in your browser does not return anything, then status checks will not work. This may be modified in the future, but in the meantime, a fix would be to make your own status service, which just checks if your app responds with whatever code you'd like, and then return a 200 plus renders an arbitrary message. Then just point ",(0,r.jsx)(n.code,{children:"statusCheckUrl"})," to your custom page."]}),"\n",(0,r.jsxs)(n.p,{children:["For further troubleshooting, use an application like ",(0,r.jsx)(n.a,{href:"https://postman.com",children:"Postman"})," to diagnose the issue. Set the parameter to ",(0,r.jsx)(n.code,{children:"GET"}),", and then make a call to: ",(0,r.jsx)(n.code,{children:"https://[url-of-dashy]/status-check/?&url=[service-url]"}),". Where the service URL must have first been encoded (e.g. with ",(0,r.jsx)(n.code,{children:"encodeURIComponent()"})," or ",(0,r.jsx)(n.a,{href:"https://www.urlencoder.io/",children:"urlencoder.io"}),")"]}),"\n",(0,r.jsx)(n.p,{children:"If you're serving Dashy though a CDN, instead of using the Node server or Docker image, then the Node endpoint that makes requests will not be available to you, and all requests will fail. A workaround for this may be implemented in the future, but in the meantime, your only option is to use the Docker or Node deployment method."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"widgets",children:"Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"widget-errors",children:"Widget Errors"}),"\n",(0,r.jsx)(n.h4,{id:"find-error-message",children:"Find Error Message"}),"\n",(0,r.jsxs)(n.p,{children:["If an error occurs when fetching or rendering results, you will see a short message in the UI. If that message doesn't adequately explain the problem, then you can ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open the browser console"})," to see more details."]}),"\n",(0,r.jsx)(n.h4,{id:"check-config",children:"Check Config"}),"\n",(0,r.jsx)(n.p,{children:"Before proceeding, ensure that if the widget requires auth your API is correct, and for custom widgets, double check that the URL and protocol is correct."}),"\n",(0,r.jsx)(n.h4,{id:"timeout-error",children:"Timeout Error"}),"\n",(0,r.jsxs)(n.p,{children:["If the error message in the console includes: ",(0,r.jsx)(n.code,{children:"Error: timeout of 500ms exceeded"}),", then your Glances endpoint is slower to respond than expected. You can fix this by ",(0,r.jsx)(n.a,{href:"/docs/widgets#setting-timeout",children:"setting timeout"})," to a larger value. This is done on each widget, with the ",(0,r.jsx)(n.code,{children:"timeout"})," attribute, and is specified in ms. E.g. ",(0,r.jsx)(n.code,{children:"timeout: 5000"})," would only fail if no response is returned within 5 seconds."]}),"\n",(0,r.jsx)(n.h4,{id:"cors-error",children:"CORS error"}),"\n",(0,r.jsxs)(n.p,{children:["If the console message mentions to corss-origin blocking, then this is a CORS error, see: ",(0,r.jsx)(n.a,{href:"#widget-cors-errors",children:"Fixing Widget CORS Errors"})]}),"\n",(0,r.jsx)(n.h4,{id:"more-info",children:"More Info"}),"\n",(0,r.jsx)(n.p,{children:"If you're able to, you can find more information about why the request may be failing in the Dev Tools under the Network tab, and you can ensure your endpoint is correct and working using a tool like Postman."}),"\n",(0,r.jsx)(n.h3,{id:"widget-cors-errors",children:"Widget CORS Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The most common widget issue is a CORS error. This is a browser security mechanism which prevents the client-side app (Dashy) from from accessing resources on a remote origin, without that server's explicit permission (e.g. with headers like Access-Control-Allow-Origin). See the MDN Docs for more info: ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"There are several ways to fix a CORS error:"}),"\n",(0,r.jsx)(n.h4,{id:"option-1---ensure-correct-protocol",children:"Option 1 - Ensure Correct Protocol"}),"\n",(0,r.jsx)(n.p,{children:"You will get a CORS error if you try and access a http service from a https source. So ensure that the URL you are requesting has the right protocol, and is correctly formatted."}),"\n",(0,r.jsx)(n.h4,{id:"option-2---set-headers",children:"Option 2 - Set Headers"}),"\n",(0,r.jsxs)(n.p,{children:["If you have control over the destination (e.g. for a self-hosted service), then you can simply apply the correct headers.\nAdd the ",(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, with the value of either ",(0,r.jsx)(n.code,{children:"*"})," to allow requests from anywhere, or more securely, the host of where Dashy is served from. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://url-of-dashy.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: *\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more info on how to set headers, see: ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Setting Headers"})," in the management docs"]}),"\n",(0,r.jsx)(n.h4,{id:"option-3---proxying-request",children:"Option 3 - Proxying Request"}),"\n",(0,r.jsxs)(n.p,{children:["You can route requests through Dashy's built-in CORS proxy. Instructions and more details can be found ",(0,r.jsx)(n.a,{href:"/docs/widgets#proxying-requests",children:"here"}),". If you don't have control over the target origin, and you are running Dashy either through Docker, with the Node server or on Netlify, then this solution will work for you."]}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.code,{children:"useProxy: true"})," option to the failing widget."]}),"\n",(0,r.jsx)(n.h4,{id:"option-4---use-a-plugin",children:"Option 4 - Use a plugin"}),"\n",(0,r.jsxs)(n.p,{children:["For testing purposes, you can use an addon, which will disable the CORS checks. You can get the Allow-CORS extension for ",(0,r.jsx)(n.a,{href:"https://chrome.google.com/webstore/detail/allow-cors-access-control/lhobafahddgcelffkeicbaginigeejlf?hl=en-US",children:"Chrome"})," or ",(0,r.jsx)(n.a,{href:"https://addons.mozilla.org/en-US/firefox/addon/access-control-allow-origin/",children:"Firefox"}),", more details ",(0,r.jsx)(n.a,{href:"https://mybrowseraddon.com/access-control-allow-origin.html",children:"here"})]}),"\n",(0,r.jsxs)(n.h3,{id:"cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound-",children:["CORS Proxy ",(0,r.jsx)(n.code,{children:"connect ECONNREFUSED ..."})," or ",(0,r.jsx)(n.code,{children:"getaddrinfo ENOTFOUND ..."})]}),"\n",(0,r.jsxs)(n.p,{children:["The target host is unreachable from the Dashy container. If the target is on the same host as Dashy, ",(0,r.jsxs)(n.strong,{children:["don't use ",(0,r.jsx)(n.code,{children:"localhost"})]})," - inside a Docker container that resolves to the container itself, not the host. Use the host's LAN IP, the Docker bridge gateway, or ",(0,r.jsx)(n.code,{children:"host.docker.internal"})," (on Docker Desktop). If the target is on a different Docker network, attach Dashy to that network too."]}),"\n",(0,r.jsxs)(n.h3,{id:"cors-proxy-target-url-host--is-blocked--must-use-http-or-https",children:["CORS Proxy ",(0,r.jsx)(n.code,{children:"Target-URL host '...' is blocked"})," / ",(0,r.jsx)(n.code,{children:"must use http:// or https://"})]}),"\n",(0,r.jsx)(n.p,{children:"To prevent the CORS proxy from being abused as a Server-Side Request Forgery vector, Dashy refuses to proxy a small number of host/scheme combinations by default:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Cloud instance metadata services"})," - ",(0,r.jsx)(n.code,{children:"169.254.169.254"}),", ",(0,r.jsx)(n.code,{children:"metadata.google.internal"}),", and the matching IPv6 forms. These are reserved magic addresses on AWS, Azure, GCP, DigitalOcean, Hetzner, Oracle Cloud and most other providers. A widget that successfully fetches them on a cloud-hosted Dashy can leak the host's IAM credentials, so they're blocked unconditionally."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Non-HTTP(S) schemes"})," - ",(0,r.jsx)(n.code,{children:"file://"}),", ",(0,r.jsx)(n.code,{children:"ftp://"}),", ",(0,r.jsx)(n.code,{children:"gopher://"}),", ",(0,r.jsx)(n.code,{children:"javascript:"}),", ",(0,r.jsx)(n.code,{children:"data:"}),", and similar. The proxy is meant for HTTP APIs only."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"If you're running Dashy in a fully isolated/private environment and you've deliberately decided you want to allow these (for example, you genuinely need to query your cloud provider's metadata API from a widget), you can opt out of all proxy restrictions by setting the environment variable:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"DANGEROUSLY_DISABLE_PROXY_RESTRICTIONS=true\n"})}),"\n",(0,r.jsxs)(n.p,{children:["The variable is named so loudly because flipping it on a Dashy instance that's exposed to anything other than fully trusted users re-opens the SSRF surface - anyone who can hit ",(0,r.jsx)(n.code,{children:"/cors-proxy"})," can then use Dashy as a relay to reach internal services. ",(0,r.jsx)(n.strong,{children:"Don't set it on cloud-hosted or internet-exposed deployments."})]}),"\n",(0,r.jsx)(n.p,{children:"Note that this is an all-or-nothing escape hatch, not a per-host allowlist. If you only need to reach one specific host that's currently blocked, please open a feature request describing the use case."}),"\n",(0,r.jsx)(n.h3,{id:"widget-shows-error-incorrectly",children:"Widget Shows Error Incorrectly"}),"\n",(0,r.jsx)(n.p,{children:"When there's an error fetching or displaying a widgets data, then it will be highlighted in yellow, and a message displayed on the UI."}),"\n",(0,r.jsxs)(n.p,{children:["In some instances, this is a false positive, and the widget is actually functioning correctly.\nIf this is the case, you can disable the UI error message of a given widget by setting: ",(0,r.jsx)(n.code,{children:"ignoreErrors: true"})]}),"\n",(0,r.jsx)(n.h3,{id:"weather-forecast-widget-401",children:"Weather Forecast Widget 401"}),"\n",(0,r.jsx)(n.p,{children:"A 401 error means your API key is invalid, it is not an issue with Dashy."}),"\n",(0,r.jsxs)(n.p,{children:["Usually this happens due to an error in your config. If you're unsure, copy and paste the ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather",children:"example"})," config, replacing the API key with your own."]}),"\n",(0,r.jsxs)(n.p,{children:["Check that ",(0,r.jsx)(n.code,{children:"apiKey"})," is correctly specified, and nested within ",(0,r.jsx)(n.code,{children:"options"}),". Ensure your input city is valid."]}),"\n",(0,r.jsxs)(n.p,{children:["To test your API key, try making a request to ",(0,r.jsx)(n.code,{children:"https://api.openweathermap.org/data/2.5/weather?q=London&appid=[your-api-key]"})]}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather-forecast",children:"Weather widget"})," is working fine, but you are getting a ",(0,r.jsx)(n.code,{children:"401"})," for the ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather-forecast",children:"Weather Forecast widget"}),", then this is also an OWM API key issue.\nSince the forecasting API requires an upgraded plan. ULPT: You can get a free, premium API key by filling in ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"this form"}),". It's a student plan, but there's no verification to check that you are still a student."]}),"\n",(0,r.jsx)(n.p,{children:"A future update will be pushed out, to use a free weather forecasting API."}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/803",children:"#803"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/789",children:"#789"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/577",children:"#577"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/621",children:"#621"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/578",children:"#578"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/806",children:"#806"})]}),"\n",(0,r.jsx)(n.h3,{id:"widget-displaying-inaccurate-data",children:"Widget Displaying Inaccurate Data"}),"\n",(0,r.jsx)(n.p,{children:"If any widget is not displaying the data you expect, first confirm that your config is correct, then try manually calling the API endpoint."}),"\n",(0,r.jsxs)(n.p,{children:["If the raw API output is correct, yet the widget is rendering incorrect results, then it is likely a bug, and a ticket should be raised. You can start to debug the issue, by looking at the widget's code (",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/components/Widgets",children:"here"}),"), and the browser console + networking tab."]}),"\n",(0,r.jsxs)(n.p,{children:["If the API itself is returning incorrect, incomplete or inaccurate data then an issue needs to be raised ",(0,r.jsx)(n.strong,{children:"with the API provider"})," (not Dashy!). You can find the API provider included within the widget docs, or for a full list see the ",(0,r.jsx)(n.a,{href:"/docs/privacy#widgets",children:"Privacy Docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/807",children:"#807"})," (re, domain monitor)"]}),"\n",(0,r.jsxs)(n.h3,{id:"public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",children:["Public IP Widget not working for ",(0,r.jsx)(n.code,{children:"ipinfo"})," or ",(0,r.jsx)(n.code,{children:"ipquery"})," providers"]}),"\n",(0,r.jsxs)(n.p,{children:["If you've set ",(0,r.jsx)(n.code,{children:"options.provider"})," to either ",(0,r.jsx)(n.code,{children:"ipinfo"})," and ",(0,r.jsx)(n.code,{children:"ipquery"})," and the requests are failing, it's likley that they're being blocked. Check your adblocker (uBlock, PrivacyBadger, etc), DNS block lists (PiHole, AdGuard, etc). Or, try proxying the request (with ",(0,r.jsx)(n.code,{children:"useProxy: true"}),") or just try a different provider."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"font-awesome-icons-not-displaying",children:"Font Awesome Icons not Displaying"}),"\n",(0,r.jsxs)(n.p,{children:["Usually, Font Awesome will be automatically enabled if one or more of your icons are using Font-Awesome. If this is not happening, then you can always manually enable (or disable) Font Awesome by setting: ",(0,r.jsx)(n.a,{href:"/docs/configuring#appconfig-optional",children:(0,r.jsx)(n.code,{children:"appConfig"})}),".",(0,r.jsx)(n.code,{children:"enableFontAwesome"})," to ",(0,r.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are trying to use a premium icon, then you must have a ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/plans",children:"Pro License"}),". You'll then need to specify your Pro plan API key under ",(0,r.jsx)(n.code,{children:"appConfig.fontAwesomeKey"}),". You can find this key, by logging into your FA account, navigate to Account \u2192 ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/kits",children:"Kits"})," \u2192 New Kit \u2192 Copy Kit Code. The code is a 10-digit alpha-numeric code, and is also visible within the new kit's URL, for example: ",(0,r.jsx)(n.code,{children:"81e48ce079"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{src:"https://i.ibb.co/hZ0D9vs/where-do-i-find-my-font-awesome-key.png",width:"600"})}),"\n",(0,r.jsxs)(n.p,{children:["Be sure that you're specifying the icon category and name correctly. You're icon should look be ",(0,r.jsx)(n.code,{children:"[category] fa-[icon-name]"}),". The following categories are supported: ",(0,r.jsx)(n.code,{children:"far"})," ",(0,r.jsx)(n.em,{children:"(regular)"}),", ",(0,r.jsx)(n.code,{children:"fas"})," ",(0,r.jsx)(n.em,{children:"(solid)"}),", ",(0,r.jsx)(n.code,{children:"fal"}),(0,r.jsx)(n.em,{children:"(light)"}),", ",(0,r.jsx)(n.code,{children:"fad"})," ",(0,r.jsx)(n.em,{children:"(duo-tone)"})," and ",(0,r.jsx)(n.code,{children:"fab"}),(0,r.jsx)(n.em,{children:"(brands)"}),". With the exception of brands, you'll usually want all your icons to be in from same category, so they look uniform."]}),"\n",(0,r.jsxs)(n.p,{children:["Ensure the icon you are trying to use, is available within ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/v5/search",children:"FontAwesome Version 5"})," (we've not yet upgraded to V6, as it works a little differently)."]}),"\n",(0,r.jsxs)(n.p,{children:["Examples: ",(0,r.jsx)(n.code,{children:"fab fa-raspberry-pi"}),", ",(0,r.jsx)(n.code,{children:"fas fa-database"}),", ",(0,r.jsx)(n.code,{children:"fas fa-server"}),", ",(0,r.jsx)(n.code,{children:"fas fa-ethernet"})]}),"\n",(0,r.jsxs)(n.p,{children:["Finally, check the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," for any error messages, and raise a ticket if the issue persists."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"copy-to-clipboard-not-working",children:"Copy to Clipboard not Working"}),"\n",(0,r.jsxs)(n.p,{children:["If the copy to clipboard feature (either under Config --\x3e Export, or Item --\x3e Copy URL) isn't functioning as expected, first check the browser console. If you see ",(0,r.jsx)(n.code,{children:"TypeError: Cannot read properties of undefined (reading 'writeText')"})," then this feature is not supported by your browser.\nThe most common reason for this, is if you not running the app over HTTPS. Copying to the clipboard requires the app to be running in a secure origin / aka have valid HTTPS cert. You can read more about this ",(0,r.jsx)(n.a,{href:"https://stackoverflow.com/a/71876238/979052",children:"here"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"As a workaround, you could either:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Highlight the text and copy / ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"C"})]}),"\n",(0,r.jsxs)(n.li,{children:["Or setup SSL - ",(0,r.jsx)(n.a,{href:"/docs/management#ssl-certificates",children:"here's a guide"})," on doing so"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"how-to--reference",children:"How-To / Reference"}),"\n",(0,r.jsx)(n.h3,{id:"how-to-reset-local-settings",children:"How to Reset Local Settings"}),"\n",(0,r.jsx)(n.p,{children:"Some settings are stored locally, in the browser's storage."}),"\n",(0,r.jsx)(n.p,{children:"In some instances cached assets can prevent your settings from being updated, in which case you may wish to reset local data."}),"\n",(0,r.jsx)(n.p,{children:'To clear all local data from the UI, head to the Config Menu, then click "Reset Local Settings", and Confirm when prompted.\nThis will not affect your config file. But be sure that you keep a backup of your config, if you\'ve not written changes it to disk.'}),"\n",(0,r.jsxs)(n.p,{children:["You can also view any and all data that Dashy is storing, using the developer tools. Open your browser's dev tools (usually ",(0,r.jsx)(n.kbd,{children:"F12"}),"), in Chromium head to the Application tab, or in Firefox go to the Storage tab. Select Local Storage, then scroll down the the URL Dashy is running on. You should now see all data being stored, and you can select and delete any fields you wish."]}),"\n",(0,r.jsxs)(n.p,{children:["For a full list of all data that may be cached, see the ",(0,r.jsx)(n.a,{href:"/docs/privacy#browser-storage",children:"Privacy Docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"how-to-make-a-bug-report",children:"How to make a bug report"}),"\n",(0,r.jsx)(n.h4,{id:"step-1---where-to-open-issues",children:"Step 1 - Where to open issues"}),"\n",(0,r.jsxs)(n.p,{children:["You will need a GitHub account in order to raise a ticket. You can then ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=lissy93&labels=%F0%9F%90%9B+Bug&template=bug.yml&title=%5BBUG%5D+%3Ctitle%3E",children:"click here"})," to open a new bug report."]}),"\n",(0,r.jsx)(n.h4,{id:"step-2---checking-its-not-already-covered",children:"Step 2 - Checking it's not already covered"}),"\n",(0,r.jsx)(n.p,{children:"Before submitting, please check that:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"A similar ticket has not previously been opened"}),"\n",(0,r.jsxs)(n.li,{children:["The issue is not covered in the ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting",children:"troubleshooting guide"})," or ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/docs#readme",children:"docs"})]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"step-3---describe-the-issue",children:"Step 3 - Describe the Issue"}),"\n",(0,r.jsx)(n.p,{children:"Your ticket will likely be dealt with more effectively if you can explain the issue clearly, and provide all relevant supporting material."}),"\n",(0,r.jsx)(n.p,{children:"Complete the fields, asking for your environment info and version of Dashy.\nThen describe the issue, briefly explaining the steps to reproduce, expected outcome and actual outcome."}),"\n",(0,r.jsx)(n.h4,{id:"step-4---provide-supporting-info",children:"Step 4 - Provide Supporting Info"}),"\n",(0,r.jsx)(n.p,{children:"Where relevant please also include:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"A screenshot of the issue"}),"\n",(0,r.jsx)(n.li,{children:"The relevant parts of your config file"}),"\n",(0,r.jsxs)(n.li,{children:["Logs\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["If client-side issue, then include the browser logs (",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"see how"}),")"]}),"\n",(0,r.jsx)(n.li,{children:"If server-side / during deployment, include the terminal output"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"Take care to redact any personal info, (like IP addresses, auth hashes or API keys)."})}),"\n",(0,r.jsx)(n.h4,{id:"step-5---fix-released",children:"Step 5 - Fix Released"}),"\n",(0,r.jsx)(n.p,{children:"A maintainer will aim to respond within 48 hours.\nThe timeframe for resolving your issue, will vary depending on severity of the bug and the complexity of the fix.\nYou will be notified on your ticket, when a fix has been released."}),"\n",(0,r.jsxs)(n.p,{children:["Finally, be sure to remain respectful to other users and project maintainers, in line with the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/CODE_OF_CONDUCT#contributor-covenant-code-of-conduct",children:"Contributor Covenant Code of Conduct"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"how-to-open-browser-console",children:"How-To Open Browser Console"}),"\n",(0,r.jsx)(n.p,{children:"When raising a bug, one crucial piece of info needed is the browser's console output. This will help the developer diagnose and fix the issue."}),"\n",(0,r.jsxs)(n.p,{children:['If you\'ve been asked for this info, but are unsure where to find it, then it is under the "Console" tab, in the browsers developer tools, which can be opened with ',(0,r.jsx)(n.kbd,{children:"F12"}),". You can right-click the console, and select Save As to download the log."]}),"\n",(0,r.jsx)(n.p,{children:"To open dev tools, and jump straight to the console:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Win / Linux: ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"Shift"})," + ",(0,r.jsx)(n.kbd,{children:"J"})]}),"\n",(0,r.jsxs)(n.li,{children:["MacOS: ",(0,r.jsx)(n.kbd,{children:"Cmd"})," + ",(0,r.jsx)(n.kbd,{children:"Option"})," + ",(0,r.jsx)(n.kbd,{children:"J"})]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more detailed walk through, see ",(0,r.jsx)(n.a,{href:"https://support.shortpoint.com/support/solutions/articles/1000222881-save-browser-console-file",children:"this article"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"git-contributions-not-displaying",children:"Git Contributions not Displaying"}),"\n",(0,r.jsxs)(n.p,{children:["If you've contributed to Dashy (or any other project), but your contributions are not showing up on your GH profile, or in Dashy's ",(0,r.jsx)(n.a,{href:"/docs/credits",children:"Credits Page"}),", then this is likely a git config issue."]}),"\n",(0,r.jsxs)(n.p,{children:["These statistics are generated using the username / email associated with commits. This info needs to be setup on your local machine using ",(0,r.jsx)(n.a,{href:"https://git-scm.com/docs/git-config",children:(0,r.jsx)(n.code,{children:"git config"})}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Run the following commands (replacing name + email with your info):"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:'git config --global user.name "John Doe"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"git config --global user.email johndoe@example.com"})}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://git-scm.com/book/en/v2/Getting-Started-First-Time-Git-Setup",children:"Git First Time Setup Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Note that only contributions to the master / main branch or a project are counted"})]})}function c(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>a});var i=s(6540);const r={},o=i.createContext(r);function t(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9013],{7309(e,n,s){s.r(n),s.d(n,{assets:()=>l,contentTitle:()=>a,default:()=>c,frontMatter:()=>t,metadata:()=>i,toc:()=>d});const i=JSON.parse('{"id":"troubleshooting","title":"Troubleshooting","description":"**This document contains common problems and their solutions.**","source":"@site/docs/troubleshooting.md","sourceDirName":".","slug":"/troubleshooting","permalink":"/docs/troubleshooting","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/troubleshooting.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"App Management","permalink":"/docs/management"},"next":{"title":"Icons","permalink":"/docs/icons"}}');var r=s(4848),o=s(8453);const t={},a="Troubleshooting",l={},d=[{value:"Contents",id:"contents",level:2},{value:"Config not saving",id:"config-not-saving",level:2},{value:"Permission denied or read-only filesystem (EACCES, EROFS)",id:"permission-denied-or-read-only-filesystem-eacces-erofs",level:3},{value:"Kubernetes ConfigMap mount is read-only",id:"kubernetes-configmap-mount-is-read-only",level:3},{value:"SELinux or AppArmor blocks the write",id:"selinux-or-apparmor-blocks-the-write",level:3},{value:"Backup step fails so save aborts",id:"backup-step-fails-so-save-aborts",level:3},{value:"Save button is missing or returns 403 Forbidden",id:"save-button-is-missing-or-returns-403-forbidden",level:3},{value:"Save unavailable on Vercel, Netlify or other static hosts",id:"save-unavailable-on-vercel-netlify-or-other-static-hosts",level:3},{value:"/config-manager/save returns 404 or HTML",id:"config-managersave-returns-404-or-html",level:3},{value:""Invalid filename" when saving a sub-page",id:"invalid-filename-when-saving-a-sub-page",level:3},{value:""Cannot save to an external URL"",id:"cannot-save-to-an-external-url",level:3},{value:"Saved successfully but the UI shows the old config",id:"saved-successfully-but-the-ui-shows-the-old-config",level:3},{value:"Container crashes or restart loop after saving (3.1.0 and 3.1.1 only)",id:"container-crashes-or-restart-loop-after-saving-310-and-311-only",level:3},{value:"Intentionally read-only mode",id:"intentionally-read-only-mode",level:3},{value:"Refused to Connect in Modal or Workspace View",id:"refused-to-connect-in-modal-or-workspace-view",level:2},{value:"NGINX",id:"nginx",level:3},{value:"Caddy",id:"caddy",level:3},{value:"Apache",id:"apache",level:3},{value:"LightHttpd",id:"lighthttpd",level:3},{value:"404 / Routing issues",id:"404--routing-issues",level:2},{value:"404 On Static Hosting",id:"404-on-static-hosting",level:3},{value:"404 after Launch from Mobile Home Screen",id:"404-after-launch-from-mobile-home-screen",level:3},{value:"404 On Multi-Page Apps",id:"404-on-multi-page-apps",level:3},{value:"Dashy hosted at a sub-path (e.g. example.com/dashy)",id:"dashy-hosted-at-a-sub-path-eg-examplecomdashy",level:3},{value:"Sub-pages",id:"sub-pages",level:2},{value:"Sub-page shows "Unable to find config for ..."",id:"sub-page-shows-unable-to-find-config-for-",level:3},{value:"Old bookmark from before an upgrade",id:"old-bookmark-from-before-an-upgrade",level:4},{value:"The page was renamed or removed",id:"the-page-was-renamed-or-removed",level:4},{value:"The path points at an unreachable file",id:"the-path-points-at-an-unreachable-file",level:4},{value:"Page name literally "Main"",id:"page-name-literally-main",level:4},{value:"Service worker is serving a stale app",id:"service-worker-is-serving-a-stale-app",level:4},{value:"Sub-page missing from nav, or won't open when clicked",id:"sub-page-missing-from-nav-or-wont-open-when-clicked",level:3},{value:"Sub-page ignores its theme, layout or appConfig",id:"sub-page-ignores-its-theme-layout-or-appconfig",level:3},{value:"Sub-config files return 404",id:"sub-config-files-return-404",level:3},{value:"Remote Config Not Loading",id:"remote-config-not-loading",level:3},{value:"Build & memory errors",id:"build--memory-errors",level:2},{value:"Yarn Error",id:"yarn-error",level:3},{value:"yarn build fails inside the container",id:"yarn-build-fails-inside-the-container",level:3},{value:"High CPU or RAM Usage on Startup",id:"high-cpu-or-ram-usage-on-startup",level:3},{value:"Ineffective mark-compacts near heap limit Allocation failed",id:"ineffective-mark-compacts-near-heap-limit-allocation-failed",level:3},{value:"Command failed with signal "SIGKILL"",id:"command-failed-with-signal-sigkill",level:3},{value:"Node Sass does not yet support your current environment",id:"node-sass-does-not-yet-support-your-current-environment",level:3},{value:"Unreachable Code Error",id:"unreachable-code-error",level:3},{value:"Error: Cannot find module './_baseValues'",id:"error-cannot-find-module-_basevalues",level:3},{value:"Auth & OIDC",id:"auth--oidc",level:2},{value:"Auth Validation Error: "should be object"",id:"auth-validation-error-should-be-object",level:3},{value:"Keycloak Redirect Error",id:"keycloak-redirect-error",level:3},{value:"OIDC or Keycloak failure on numeric client IDs",id:"oidc-or-keycloak-failure-on-numeric-client-ids",level:3},{value:"Redirect loop after login",id:"redirect-loop-after-login",level:3},{value:"invalid_redirect_uri",id:"invalid_redirect_uri",level:3},{value:"Login works in the browser but the dashboard refuses to save anything (403)",id:"login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",level:3},{value:"Logged in but no admin controls",id:"logged-in-but-no-admin-controls",level:3},{value:"Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"",id:"login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",level:3},{value:"Sign-out leaves you stuck on Authentik",id:"sign-out-leaves-you-stuck-on-authentik",level:3},{value:"Untrusted certificate from Authentik",id:"untrusted-certificate-from-authentik",level:3},{value:"Numeric client_id getting truncated",id:"numeric-client_id-getting-truncated",level:3},{value:"Docker & image issues",id:"docker--image-issues",level:2},{value:"App Not Starting After Update to 2.0.4",id:"app-not-starting-after-update-to-204",level:3},{value:"Mount Type Mismatch",id:"mount-type-mismatch",level:3},{value:"DockerHub toomanyrequests",id:"dockerhub-toomanyrequests",level:3},{value:"Solution 1 - Use an alternate container registry",id:"solution-1---use-an-alternate-container-registry",level:4},{value:"Solution 2 - Increase your rate limits",id:"solution-2---increase-your-rate-limits",level:4},{value:"Old image tags fail to pull",id:"old-image-tags-fail-to-pull",level:3},{value:"Healthcheck Failing in Docker",id:"healthcheck-failing-in-docker",level:3},{value:"SSL-enabled Dashy",id:"ssl-enabled-dashy",level:4},{value:"Custom port",id:"custom-port",level:4},{value:"Container is unhealthy past the grace period",id:"container-is-unhealthy-past-the-grace-period",level:4},{value:"Docker Login Fails on Ubuntu",id:"docker-login-fails-on-ubuntu",level:3},{value:"Styles and Assets not Updating",id:"styles-and-assets-not-updating",level:2},{value:"Config Validation Errors",id:"config-validation-errors",level:2},{value:"Invalid Host Header while running through ngrok",id:"invalid-host-header-while-running-through-ngrok",level:2},{value:"Warnings in the Console during deploy",id:"warnings-in-the-console-during-deploy",level:2},{value:"Status Checks Failing",id:"status-checks-failing",level:2},{value:"Widgets",id:"widgets",level:2},{value:"Widget Errors",id:"widget-errors",level:3},{value:"Find Error Message",id:"find-error-message",level:4},{value:"Check Config",id:"check-config",level:4},{value:"Timeout Error",id:"timeout-error",level:4},{value:"CORS error",id:"cors-error",level:4},{value:"More Info",id:"more-info",level:4},{value:"Widget CORS Errors",id:"widget-cors-errors",level:3},{value:"Option 1 - Ensure Correct Protocol",id:"option-1---ensure-correct-protocol",level:4},{value:"Option 2 - Set Headers",id:"option-2---set-headers",level:4},{value:"Option 3 - Proxying Request",id:"option-3---proxying-request",level:4},{value:"Option 4 - Use a plugin",id:"option-4---use-a-plugin",level:4},{value:"CORS Proxy connect ECONNREFUSED ... or getaddrinfo ENOTFOUND ...",id:"cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound-",level:3},{value:"CORS Proxy Target-URL host '...' is blocked / must use http:// or https://",id:"cors-proxy-target-url-host--is-blocked--must-use-http-or-https",level:3},{value:"Widget Shows Error Incorrectly",id:"widget-shows-error-incorrectly",level:3},{value:"Weather Forecast Widget 401",id:"weather-forecast-widget-401",level:3},{value:"Widget Displaying Inaccurate Data",id:"widget-displaying-inaccurate-data",level:3},{value:"Public IP Widget not working for ipinfo or ipquery providers",id:"public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",level:3},{value:"Font Awesome Icons not Displaying",id:"font-awesome-icons-not-displaying",level:2},{value:"Copy to Clipboard not Working",id:"copy-to-clipboard-not-working",level:2},{value:"How-To / Reference",id:"how-to--reference",level:2},{value:"How to Reset Local Settings",id:"how-to-reset-local-settings",level:3},{value:"How to make a bug report",id:"how-to-make-a-bug-report",level:3},{value:"Step 1 - Where to open issues",id:"step-1---where-to-open-issues",level:4},{value:"Step 2 - Checking it's not already covered",id:"step-2---checking-its-not-already-covered",level:4},{value:"Step 3 - Describe the Issue",id:"step-3---describe-the-issue",level:4},{value:"Step 4 - Provide Supporting Info",id:"step-4---provide-supporting-info",level:4},{value:"Step 5 - Fix Released",id:"step-5---fix-released",level:4},{value:"How-To Open Browser Console",id:"how-to-open-browser-console",level:3},{value:"Git Contributions not Displaying",id:"git-contributions-not-displaying",level:3}];function h(e){const n={a:"a",blockquote:"blockquote",br:"br",code:"code",em:"em",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",img:"img",kbd:"kbd",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,o.R)(),...e.components};return(0,r.jsxs)(r.Fragment,{children:[(0,r.jsx)(n.header,{children:(0,r.jsx)(n.h1,{id:"troubleshooting",children:"Troubleshooting"})}),"\n",(0,r.jsxs)(n.blockquote,{children:["\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.em,{children:(0,r.jsx)(n.strong,{children:"This document contains common problems and their solutions."})}),(0,r.jsx)(n.br,{}),"\nPlease ensure your issue isn't listed here, before opening a new ticket."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"Found something not listed here? Consider adding it, to help other users."})}),"\n"]}),"\n",(0,r.jsx)(n.h2,{id:"contents",children:"Contents"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#config-not-saving",children:"Config not saving"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#permission-denied-or-read-only-filesystem-eacces-erofs",children:"Permission denied or read-only filesystem"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#kubernetes-configmap-mount-is-read-only",children:"Kubernetes ConfigMap mount is read-only"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#selinux-or-apparmor-blocks-the-write",children:"SELinux or AppArmor blocks the write"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#backup-step-fails-so-save-aborts",children:"Backup step fails so save aborts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#save-button-is-missing-or-returns-403-forbidden",children:"Save button is missing or returns 403"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#save-unavailable-on-vercel-netlify-or-other-static-hosts",children:"Save unavailable on Vercel, Netlify or other static hosts"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#config-managersave-returns-404-or-html",children:"/config-manager/save returns 404 or HTML"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid-filename-when-saving-a-sub-page",children:'"Invalid filename" when saving a sub-page'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cannot-save-to-an-external-url",children:'"Cannot save to an external URL"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#saved-successfully-but-the-ui-shows-the-old-config",children:"Saved successfully but the UI shows the old config"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#container-crashes-or-restart-loop-after-saving-310-and-311-only",children:"Container crashes or restart loop after saving"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#intentionally-read-only-mode",children:"Intentionally read-only mode"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#refused-to-connect-in-modal-or-workspace-view",children:"Refused to Connect in Web Content View"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#404--routing-issues",children:"404 / Routing issues"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-on-static-hosting",children:"404 On Static Hosting"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-after-launch-from-mobile-home-screen",children:"404 from Mobile Home Screen"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#404-on-multi-page-apps",children:"404 On Multi-Page Apps"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dashy-hosted-at-a-sub-path-eg-examplecomdashy",children:"Dashy hosted at a sub-path"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#sub-pages",children:"Sub-pages"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-shows-unable-to-find-config-for",children:'Sub-page shows "Unable to find config for ..."'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-missing-from-nav-or-wont-open-when-clicked",children:"Sub-page missing from nav, or won't open when clicked"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-page-ignores-its-theme-layout-or-appconfig",children:"Sub-page ignores its theme, layout or appConfig"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sub-config-files-return-404",children:"Sub-config files return 404"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#remote-config-not-loading",children:"Remote Config Not Loading"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#build--memory-errors",children:"Build & memory errors"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#yarn-error",children:"Yarn Build or Run Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsxs)(n.a,{href:"#yarn-build-fails-inside-the-container",children:[(0,r.jsx)(n.code,{children:"yarn build"})," fails inside the container"]})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Heap limit Allocation failed"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#command-failed-with-signal-sigkill",children:'Command failed with signal "SIGKILL"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#node-sass-does-not-yet-support-your-current-environment",children:"Node Sass unsupported environment"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#unreachable-code-error",children:"Unreachable Code Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#error-cannot-find-module-_basevalues",children:"Cannot find module './_baseValues'"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#auth--oidc",children:"Auth & OIDC"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#auth-validation-error-should-be-object",children:'Auth Validation Error: "should be object"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#keycloak-redirect-error",children:"Keycloak Redirect Error"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#oidc-or-keycloak-failure-on-numeric-client-ids",children:"OIDC or Keycloak failure on numeric client IDs"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#redirect-loop-after-login",children:"Redirect loop after login"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid_redirect_uri",children:"invalid_redirect_uri"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",children:"Login works in the browser but the dashboard refuses to save anything (403)"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#logged-in-but-no-admin-controls",children:"Logged in but no admin controls"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",children:'Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#sign-out-leaves-you-stuck-on-authentik",children:"Sign-out leaves you stuck on Authentik"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#untrusted-certificate-from-authentik",children:"Untrusted certificate from Authentik"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#numeric-client_id-getting-truncated",children:"Numeric client_id getting truncated"})}),"\n"]}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#docker--image-issues",children:"Docker & image issues"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#app-not-starting-after-update-to-204",children:"App Not Starting After Update to 2.0.4"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#mount-type-mismatch",children:"Mount Type Mismatch"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#dockerhub-toomanyrequests",children:"DockerHub toomanyrequests"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#old-image-tags-fail-to-pull",children:"Old image tags fail to pull"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#healthcheck-failing-in-docker",children:"Healthcheck Failing in Docker"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#docker-login-fails-on-ubuntu",children:"Docker Login Fails on Ubuntu"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#styles-and-assets-not-updating",children:"Styles and Assets not Updating"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#config-validation-errors",children:"Config Validation Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#invalid-host-header-while-running-through-ngrok",children:"Ngrok Invalid Host Headers"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#warnings-in-the-console-during-deploy",children:"Warnings in the Console during deploy"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#status-checks-failing",children:"Status Checks Failing"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#widgets",children:"Widgets"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-errors",children:"Diagnosing Widget Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-cors-errors",children:"Fixing Widget CORS Errors"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound",children:"CORS Proxy connect ECONNREFUSED or ENOTFOUND"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#cors-proxy-target-url-host--is-blocked--must-use-http-or-https",children:"CORS Proxy Target-URL host blocked or scheme rejected"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-shows-error-incorrectly",children:"Widget Shows Error Incorrectly"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#weather-forecast-widget-401",children:"Weather Forecast Widget 401"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#widget-displaying-inaccurate-data",children:"Widget Displaying Inaccurate Data"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",children:"Public IP Widget not working for ipinfo or ipquery providers"})}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#font-awesome-icons-not-displaying",children:"Font Awesome Icons not Displaying"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#copy-to-clipboard-not-working",children:"Copy to Clipboard not Working"})}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.a,{href:"#how-to--reference",children:"How-To / Reference"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-reset-local-settings",children:"How to Reset Local Settings"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-make-a-bug-report",children:"How to make a bug report"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"How-To Open Browser Console"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.a,{href:"#git-contributions-not-displaying",children:"Git Contributions not Displaying"})}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"config-not-saving",children:"Config not saving"}),"\n",(0,r.jsxs)(n.p,{children:["There should be an error message, explaining the reason the config save failed. First check ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," (",(0,r.jsx)(n.kbd,{children:"F12"})," --\x3e Console), and then your server-side logs in the terminal. Then, see the following sections for solutions to each possible error."]}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.a,{id:"config-not-saving-on-vercel--netlify--cdn"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-1-unable-to-call-save-endpoint-from-cdnstatic-server"}),"\n",(0,r.jsx)(n.a,{id:"unable-to-write-confyml-eacces-permission-denied"}),"\n",(0,r.jsx)(n.a,{id:"permission-denied-saving-config"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-2-unable-to-save"}),"\n",(0,r.jsx)(n.a,{id:"config-not-updating"}),"\n",(0,r.jsx)(n.a,{id:"config-still-not-updating"}),"\n",(0,r.jsx)(n.a,{id:"possible-issue-3-saved-but-not-updating"})]}),"\n",(0,r.jsx)(n.h3,{id:"permission-denied-or-read-only-filesystem-eacces-erofs",children:"Permission denied or read-only filesystem (EACCES, EROFS)"}),"\n",(0,r.jsxs)(n.p,{children:["The container can't write to your ",(0,r.jsx)(n.code,{children:"conf.yml"})," or its directory. Almost always an ownership mismatch: the host directory belongs to a different uid than the one Dashy runs as inside the container. Less commonly a read-only mount or an over-strict file mode."]}),"\n",(0,r.jsxs)(n.p,{children:["Dashy runs as UID=1000 (default non-root node user). You can see this by running ",(0,r.jsx)(n.code,{children:"docker exec -it dashy id"}),". Then, check who owns the user-data directory, with: ",(0,r.jsx)(n.code,{children:"docker exec -it dashy ls -la /app/user-data"})," - if it's not ",(0,r.jsx)(n.code,{children:"1000"})," then that's the issue. And the solution is just to run ",(0,r.jsx)(n.code,{children:"sudo chown -R 1000:1000 /path/to/your/user-data"})," to set the right owner."]}),"\n",(0,r.jsx)(n.p,{children:"Fixes:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Hand the directory to uid 1000"})," (recommended). Keeps the container running as a non-root user, which is how Dashy is built to run ",(0,r.jsx)(n.code,{children:"sudo chown -R 1000:1000 /path/to/your/user-data"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Run the container as your own user"})," if ",(0,r.jsx)(n.code,{children:"chown"})," isn't practical (multi-user hosts, NAS appliances, host directories you don't want relabelled). Add ",(0,r.jsx)(n.code,{children:"--user $(id -u):$(id -g)"})," to ",(0,r.jsx)(n.code,{children:"docker run"}),", or set ",(0,r.jsx)(n.code,{children:'user: "1000:1000"'})," (or your host uid:gid) on the service in ",(0,r.jsx)(n.code,{children:"docker-compose.yml"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Loosen a single-file mount"})," if its mode is ",(0,r.jsx)(n.code,{children:"444"}),". Narrow case, only fixes that one symptom: ",(0,r.jsx)(n.code,{children:"chmod 644 /path/to/conf.yml"})]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Common mistakes"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Using uid/gid 1001."})," A common guess on Synology, Unraid and similar where 1001 is the host's first user. Dashy's container is 1000, not 1001."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsxs)(n.strong,{children:[(0,r.jsx)(n.code,{children:"chmod"})," alone for a UID mismatch."]})," Loosens permissions but doesn't change who owns the file. You need ",(0,r.jsx)(n.code,{children:"chown"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsxs)(n.strong,{children:[(0,r.jsx)(n.code,{children:"chmod -R 777"})," or ",(0,r.jsx)(n.code,{children:"775"}),"."]})," Works as a workaround, masks the real problem, weakens security. Use ",(0,r.jsx)(n.code,{children:"chown"})," to the right uid instead."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"Other gotchas:"})}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Named Docker volumes"})," (created with ",(0,r.jsx)(n.code,{children:"docker volume create"}),") inherit ownership from whatever first writes to them. If an older container set them up as root, the diagnose step will show that. Recreate the volume or ",(0,r.jsx)(n.code,{children:"chown"})," the underlying directory under ",(0,r.jsx)(n.code,{children:"/var/lib/docker/volumes/"}),"."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"macOS hosts"})," rarely hit this. Docker Desktop transparently maps host uid to container uid through its VM. If saves are failing on macOS, look elsewhere first."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Storage layers that ignore POSIX permissions"})," (some NAS app-data folders use FUSE, SMB or overlay mounts where ",(0,r.jsx)(n.code,{children:"chmod"})," and ",(0,r.jsx)(n.code,{children:"chgrp"})," are silent no-ops). Bind-mount user-data from a native filesystem path instead."]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"kubernetes-configmap-mount-is-read-only",children:"Kubernetes ConfigMap mount is read-only"}),"\n",(0,r.jsxs)(n.p,{children:["If you've mounted your ",(0,r.jsx)(n.code,{children:"conf.yml"})," from a ConfigMap, writes will always fail with ",(0,r.jsx)(n.code,{children:"EROFS"})," regardless of UID. ConfigMap volumes are read-only by design. Either treat the ConfigMap as the source of truth and edit it directly (saves through the UI won't work), or use a writable volume type like a ",(0,r.jsx)(n.code,{children:"PersistentVolumeClaim"})," for ",(0,r.jsx)(n.code,{children:"user-data/"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"selinux-or-apparmor-blocks-the-write",children:"SELinux or AppArmor blocks the write"}),"\n",(0,r.jsxs)(n.p,{children:["If you're on RHEL/Fedora, or systems with SELinux or AppArmour, and you've confirmed permissions are fine, and container's UID matches the host owner, but you still see ",(0,r.jsx)(n.code,{children:"EACCES"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["For SELinux, add the ",(0,r.jsx)(n.code,{children:":Z"})," flag to your volume mount so Docker relabels it for the container (e.g. ",(0,r.jsx)(n.code,{children:"volumes: [ './user-data:/app/user-data:Z' ]"}),")"]}),"\n",(0,r.jsxs)(n.p,{children:["For AppArmor, check ",(0,r.jsx)(n.code,{children:"dmesg"})," for ",(0,r.jsx)(n.code,{children:'apparmor="DENIED"'})," lines and adjust the profile. Disabling enforcement is a last resort."]}),"\n",(0,r.jsx)(n.h3,{id:"backup-step-fails-so-save-aborts",children:"Backup step fails so save aborts"}),"\n",(0,r.jsxs)(n.p,{children:["Before each save, Dashy backs up the current ",(0,r.jsx)(n.code,{children:"conf.yml"})," to ",(0,r.jsx)(n.code,{children:"user-data/config-backups/"}),". If that folder can't be written, the whole save aborts with ",(0,r.jsx)(n.code,{children:"Unable to backup conf.yml"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Two ways out:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:["Point ",(0,r.jsx)(n.code,{children:"BACKUP_DIR"})," at a writable path"]}),"\n",(0,r.jsxs)(n.li,{children:["Set ",(0,r.jsx)(n.code,{children:"DISABLE_CONFIG_BACKUPS=true"})," to skip the backup step entirely"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"save-button-is-missing-or-returns-403-forbidden",children:"Save button is missing or returns 403 Forbidden"}),"\n",(0,r.jsxs)(n.p,{children:["Have you got auth setup? If so, make sure you are logged in as an admin, or set ",(0,r.jsx)(n.code,{children:"type: admin"})," to your user in ",(0,r.jsx)(n.code,{children:"conf.yml"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["Beyond that, there's several other config options which prevent saving the config file, so if you didn't mean to add them, just remove from ",(0,r.jsx)(n.code,{children:"conf.yml"})]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.preventWriteToDisk: true"})})," disables disk save and the button"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.preventLocalSave: true"})}),' disables the "Local" save option']}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:(0,r.jsx)(n.code,{children:"appConfig.disableConfiguration: true"})})," hides the editor entirely. ",(0,r.jsx)(n.code,{children:"disableConfigurationForNonAdmin: true"})," does the same just for non-admins."]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"save-unavailable-on-vercel-netlify-or-other-static-hosts",children:"Save unavailable on Vercel, Netlify or other static hosts"}),"\n",(0,r.jsx)(n.p,{children:'Updating source config file on static hosts is not possible, since they have no Node server, nor have write access to modify any files.\nThe "Local" save mode will still work (changes are just persisted in your browser), but the real solution is to copy/export the updated YAML and replace it in the source config file in your repo.'}),"\n",(0,r.jsxs)(n.p,{children:["Related: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1465",children:"#1465"}),"."]}),"\n",(0,r.jsxs)(n.h3,{id:"config-managersave-returns-404-or-html",children:[(0,r.jsx)(n.code,{children:"/config-manager/save"})," returns 404 or HTML"]}),"\n",(0,r.jsxs)(n.p,{children:["How are you running/serving Dashy?\nIf you've got a reverse proxy which only forwards specific path prefixes then maybe you're missing the ",(0,r.jsx)(n.code,{children:"/config-manager/*"})," API endpoints?\nCheck the failed request in the browser's Network tab. If the response is HTML (a proxy error page) or a plain 404, your proxy isn't routing the path. Add ",(0,r.jsx)(n.code,{children:"/config-manager/"})," to whatever you're forwarding, or simplify the rules so everything reaches Dashy."]}),"\n",(0,r.jsx)(n.p,{children:"Or if you're serving up the compiled Vue app directly, instead of using the Node server, then the endpoint won't be available."}),"\n",(0,r.jsx)(n.h3,{id:"invalid-filename-when-saving-a-sub-page",children:'"Invalid filename" when saving a sub-page'}),"\n",(0,r.jsxs)(n.p,{children:["The save endpoint rejects sub-page filenames with path separators or non-yaml extensions. Check the ",(0,r.jsx)(n.code,{children:"path:"})," value of the page in your ",(0,r.jsx)(n.code,{children:"pages:"})," block. It needs to be a plain basename like ",(0,r.jsx)(n.code,{children:"home.yml"}),", not ",(0,r.jsx)(n.code,{children:"pages/home.yml"})," or ",(0,r.jsx)(n.code,{children:"home.txt"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"cannot-save-to-an-external-url",children:'"Cannot save to an external URL"'}),"\n",(0,r.jsxs)(n.p,{children:["The sub-page you are editing is loaded from a remote URL. Dashy can't write back to that URL.\nYou will need to edit the file at it's origin yourself instead (click the Export to view the YAML).\nOr you could download the config to ",(0,r.jsx)(n.code,{children:"user-data/something.yml"}),", and update ",(0,r.jsx)(n.code,{children:"path:"})," to point to the local version."]}),"\n",(0,r.jsx)(n.h3,{id:"saved-successfully-but-the-ui-shows-the-old-config",children:"Saved successfully but the UI shows the old config"}),"\n",(0,r.jsx)(n.p,{children:"Two unrelated causes share this symptom:"}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Local storage overrides the file."})," Dashy lets users save settings locally in browser storage, which take priority over ",(0,r.jsx)(n.code,{children:"conf.yml"}),'. Open Dashy in incognito to confirm. If the changes appear there, clear local settings via Config menu > "Clear Local Settings".']}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Docker isn't picking up file changes."})," Some text editors save by replacing the inode, which breaks single-file bind mounts. Edit the file in place, or mount the parent directory rather than the single file. ",(0,r.jsx)(n.a,{href:"https://medium.com/@jonsbun/why-need-to-be-careful-when-mounting-single-files-into-a-docker-container-4f929340834",children:"More background"}),"."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.a,{id:"container-crashes-or-restart-loop-after-saving-config"})}),"\n",(0,r.jsx)(n.h3,{id:"container-crashes-or-restart-loop-after-saving-310-and-311-only",children:"Container crashes or restart loop after saving (3.1.0 and 3.1.1 only)"}),"\n",(0,r.jsxs)(n.p,{children:["If your container crashes or restart-loops right after clicking save, with logs like ",(0,r.jsx)(n.code,{children:"ERR_HTTP_HEADERS_SENT"})," or ",(0,r.jsx)(n.code,{children:"ERR_STREAM_WRITE_AFTER_END"}),", this was a known double-",(0,r.jsx)(n.code,{children:"res.end()"})," bug in 3.1.0 and 3.1.1. Fixed in v3.2.13 and later."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"docker pull lissy93/dashy:latest\ndocker compose up -d --force-recreate\n"})}),"\n",(0,r.jsx)(n.h3,{id:"intentionally-read-only-mode",children:"Intentionally read-only mode"}),"\n",(0,r.jsxs)(n.p,{children:['To hide the "Save to disk" UI for everyone, set ',(0,r.jsx)(n.code,{children:"appConfig.preventWriteToDisk: true"})," in ",(0,r.jsx)(n.code,{children:"conf.yml"}),". This is a UI-only flag \u2014 the ",(0,r.jsx)(n.code,{children:"/config-manager/save"})," server endpoint itself is gated by the configured auth method (",(0,r.jsx)(n.code,{children:"auth.users"})," with ",(0,r.jsx)(n.code,{children:"ENABLE_HTTP_AUTH"}),", OIDC/Keycloak admin role, header-auth, etc.), so anyone unauthenticated or non-admin already can't save regardless of this flag. For Docker users, you can harden things further by mounting ",(0,r.jsx)(n.code,{children:"user-data"})," (or just ",(0,r.jsx)(n.code,{children:"conf.yml"}),") as read-only \u2014 the kernel will refuse the write even before the server tries."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsxs)(n.h2,{id:"refused-to-connect-in-modal-or-workspace-view",children:[(0,r.jsx)(n.code,{children:"Refused to Connect"})," in Modal or Workspace View"]}),"\n",(0,r.jsx)(n.p,{children:"This is not an issue with Dashy, but instead caused by the target app preventing direct access through embedded elements."}),"\n",(0,r.jsxs)(n.p,{children:["As defined in ",(0,r.jsx)(n.a,{href:"https://datatracker.ietf.org/doc/html/rfc7034",children:"RFC-7034"}),", for any web content to be accessed through an embedded element, it must have the ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Frame-Options",children:(0,r.jsx)(n.code,{children:"X-Frame-Options"})})," HTTP header set to ",(0,r.jsx)(n.code,{children:"ALLOW"}),". If you are getting a ",(0,r.jsx)(n.code,{children:"Refused to Connect"})," error then this header is set to ",(0,r.jsx)(n.code,{children:"DENY"})," (or ",(0,r.jsx)(n.code,{children:"SAMEORIGIN"})," and it's on a different host). Thankfully, for self-hosted services, it is easy to set these headers."]}),"\n",(0,r.jsx)(n.p,{children:"These settings are usually set in the config file for the web server that's hosting the target application, here are some examples of how to enable cross-origin access with common web servers:"}),"\n",(0,r.jsx)(n.h3,{id:"nginx",children:"NGINX"}),"\n",(0,r.jsxs)(n.p,{children:["In NGINX, you can use the ",(0,r.jsx)(n.a,{href:"https://nginx.org/en/docs/http/ngx_http_headers_module.html",children:(0,r.jsx)(n.code,{children:"add_header"})})," module within the app block."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"server {\n ...\n add_header X-Frame-Options SAMEORIGIN always;\n}\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Then reload with ",(0,r.jsx)(n.code,{children:"service nginx reload"})]}),"\n",(0,r.jsx)(n.h3,{id:"caddy",children:"Caddy"}),"\n",(0,r.jsxs)(n.p,{children:["In Caddy, you can use the ",(0,r.jsx)(n.a,{href:"https://caddyserver.com/docs/caddyfile/directives/header",children:(0,r.jsx)(n.code,{children:"header"})})," directive."]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"header {\n X-Frame-Options SAMEORIGIN\n}\n"})}),"\n",(0,r.jsx)(n.h3,{id:"apache",children:"Apache"}),"\n",(0,r.jsxs)(n.p,{children:["In Apache, you can use the ",(0,r.jsx)(n.a,{href:"https://httpd.apache.org/docs/current/mod/mod_headers.html",children:(0,r.jsx)(n.code,{children:"mod_headers"})})," module to set the ",(0,r.jsx)(n.code,{children:"X-Frame-Options"})," in your config file. This file is usually located somewhere like `/etc/apache2/httpd.conf"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'Header set X-Frame-Options: "ALLOW-FROM http://[dashy-location]/"\n'})}),"\n",(0,r.jsx)(n.h3,{id:"lighthttpd",children:"LightHttpd"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Content-Security-Policy: frame-ancestors 'self' https://[dashy-location]/\n"})}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"404--routing-issues",children:"404 / Routing issues"}),"\n",(0,r.jsx)(n.h3,{id:"404-on-static-hosting",children:"404 On Static Hosting"}),"\n",(0,r.jsx)(n.p,{children:"If you're seeing Dashy's 404 page on initial load/ refresh, and then the main app when you go back to Home, then this is likely caused by the Vue router, and if so can be fixed in one of two ways."}),"\n",(0,r.jsxs)(n.p,{children:["The first solution is to switch the routing mode, from HTML5 ",(0,r.jsx)(n.code,{children:"history"})," mode to ",(0,r.jsx)(n.code,{children:"hash"})," mode, by rebuilding Dashy with the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set."]}),"\n",(0,r.jsxs)(n.p,{children:["If this works, but you wish to continue using HTML5 history mode, then a bit of extra ",(0,r.jsx)(n.a,{href:"/docs/management#web-server-configuration",children:"server configuration"})," is required. This is explained in more detaail in the ",(0,r.jsx)(n.a,{href:"https://router.vuejs.org/guide/essentials/history-mode.html",children:"Vue Docs"}),". Once completed, you can then use ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=history"})," (the default) again, for neater URLs."]}),"\n",(0,r.jsx)(n.h3,{id:"404-after-launch-from-mobile-home-screen",children:"404 after Launch from Mobile Home Screen"}),"\n",(0,r.jsxs)(n.p,{children:['Similar to the above issue, if you get a 404 after using iOS and Android\'s "Add to Home Screen" feature, then this is caused by Vue router.\nIt can be fixed by rebuilding Dashy with the ',(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/628",children:"#628"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/762",children:"#762"})]}),"\n",(0,r.jsx)(n.h3,{id:"404-on-multi-page-apps",children:"404 On Multi-Page Apps"}),"\n",(0,r.jsxs)(n.p,{children:["Similar to above, if you get a 404 error when visiting a page directly on multi-page apps, then this can be fixed by rebuilding Dashy with the ",(0,r.jsx)(n.code,{children:"VITE_APP_ROUTING_MODE=hash"})," build-time environment variable set, then refreshing the page."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/670",children:"#670"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/763",children:"#763"})]}),"\n",(0,r.jsxs)(n.h3,{id:"dashy-hosted-at-a-sub-path-eg-examplecomdashy",children:["Dashy hosted at a sub-path (e.g. ",(0,r.jsx)(n.code,{children:"example.com/dashy"}),")"]}),"\n",(0,r.jsxs)(n.p,{children:["If the homepage works but sub-page links 404, or assets fail to load, it's almost always the base path.\nRebuild with ",(0,r.jsx)(n.code,{children:"BASE_URL"})," set to the sub-path - leading slash, no trailing slash:"]}),"\n",(0,r.jsxs)(n.p,{children:["Vue Router uses this to prefix every route. Without it, links resolve to ",(0,r.jsx)(n.code,{children:"/home/..."})," instead of ",(0,r.jsx)(n.code,{children:"/dashy/home/..."})," and skip your reverse proxy altogether. More detail in ",(0,r.jsx)(n.a,{href:"/docs/management#web-server-configuration",children:"web-server configuration"}),"."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"sub-pages",children:"Sub-pages"}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-shows-unable-to-find-config-for-",children:'Sub-page shows "Unable to find config for ..."'}),"\n",(0,r.jsxs)(n.p,{children:["This means Dashy couldn't match the URL segment to any entry in your ",(0,r.jsx)(n.code,{children:"pages:"})," list. A few causes:"]}),"\n",(0,r.jsx)(n.h4,{id:"old-bookmark-from-before-an-upgrade",children:"Old bookmark from before an upgrade"}),"\n",(0,r.jsxs)(n.p,{children:["Slugs are now trimmed more aggressively (e.g. ",(0,r.jsx)(n.code,{children:"\ud83c\udf10 Command Center"})," used to give ",(0,r.jsx)(n.code,{children:"-command-center"}),", now gives ",(0,r.jsx)(n.code,{children:"command-center"}),"). Re-bookmark from the nav, or update the URL by hand."]}),"\n",(0,r.jsx)(n.h4,{id:"the-page-was-renamed-or-removed",children:"The page was renamed or removed"}),"\n",(0,r.jsxs)(n.p,{children:["The URL no longer resolves to anything. Check the ",(0,r.jsx)(n.code,{children:"pages:"})," array in ",(0,r.jsx)(n.code,{children:"conf.yml"})," and confirm the sub-page still exists."]}),"\n",(0,r.jsx)(n.h4,{id:"the-path-points-at-an-unreachable-file",children:"The path points at an unreachable file"}),"\n",(0,r.jsxs)(n.p,{children:["If the sub-config YAML can't be fetched (404, CORS, auth), you'll see \"Unable to load config from ...\" instead. Verify the ",(0,r.jsx)(n.code,{children:"path:"})," is correct, reachable from the browser, and CORS-open if remote."]}),"\n",(0,r.jsx)(n.h4,{id:"page-name-literally-main",children:'Page name literally "Main"'}),"\n",(0,r.jsxs)(n.p,{children:[(0,r.jsx)(n.code,{children:"main"}),' is reserved in the URL scheme to mean "the root config". A page named "Main" becomes reachable at ',(0,r.jsx)(n.code,{children:"/home/main-page"})," (not ",(0,r.jsx)(n.code,{children:"/home/main"}),"). Rename the page if that's confusing."]}),"\n",(0,r.jsx)(n.h4,{id:"service-worker-is-serving-a-stale-app",children:"Service worker is serving a stale app"}),"\n",(0,r.jsxs)(n.p,{children:["Hard-refresh (",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"F5"}),") after a major upgrade. The PWA cache may still be pointing at old routes. Also see ",(0,r.jsx)(n.a,{href:"#styles-and-assets-not-updating",children:"Styles and Assets not Updating"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-missing-from-nav-or-wont-open-when-clicked",children:"Sub-page missing from nav, or won't open when clicked"}),"\n",(0,r.jsxs)(n.p,{children:["If page defined in ",(0,r.jsx)(n.code,{children:"pages:"})," is nowhere in the nav bar, or its link goes to a different page, then there's probably something wrong with the name you chose. Note that Dashy strips out any non-alphanumeric characters."]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Ensure each page does have a valid ",(0,r.jsx)(n.code,{children:"name"})," and ",(0,r.jsx)(n.code,{children:"path"})," field"]}),"\n",(0,r.jsx)(n.li,{children:"Check two pages don't have the same/similar name"}),"\n",(0,r.jsx)(n.li,{children:"Check each page has a name which has at least some alpha-numeric characters"}),"\n",(0,r.jsx)(n.li,{children:"Very long names could be being stripped/truncated"}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"sub-page-ignores-its-theme-layout-or-appconfig",children:"Sub-page ignores its theme, layout or appConfig"}),"\n",(0,r.jsxs)(n.p,{children:["This is by design. Only the ",(0,r.jsx)(n.code,{children:"appConfig"})," from your root ",(0,r.jsx)(n.code,{children:"conf.yml"})," is used - theme, layout, iconSize, statusCheck, etc. are inherited globally so behaviour stays consistent across pages."]}),"\n",(0,r.jsxs)(n.p,{children:["If you put ",(0,r.jsx)(n.code,{children:"appConfig"})," inside a sub-page YAML, it's silently dropped on load. Move the values to the root config. See ",(0,r.jsx)(n.a,{href:"/docs/pages-and-sections#restrictions",children:"Restrictions"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"sub-config-files-return-404",children:"Sub-config files return 404"}),"\n",(0,r.jsxs)(n.p,{children:["If your ",(0,r.jsx)(n.code,{children:"conf.yml"})," references additional pages via ",(0,r.jsx)(n.code,{children:"pages:"})," and the browser shows ",(0,r.jsx)(n.code,{children:"Sub-config load failed: /something.yml"}),", the cause is almost always a Docker mount that only exposes ",(0,r.jsx)(n.code,{children:"conf.yml"})," and not the rest of ",(0,r.jsx)(n.code,{children:"user-data/"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you've done this:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n - ./my-conf.yml:/app/user-data/conf.yml\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Only ",(0,r.jsx)(n.code,{children:"conf.yml"})," exists inside the container. Anything it references (sub-configs, custom icons, fonts, CSS) isn't there."]}),"\n",(0,r.jsx)(n.p,{children:"Mount the directory instead:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n - ./user-data:/app/user-data\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Now everything in your ",(0,r.jsx)(n.code,{children:"user-data"})," folder is reachable at the web root. Same applies to ",(0,r.jsx)(n.code,{children:"docker run -v"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"remote-config-not-loading",children:"Remote Config Not Loading"}),"\n",(0,r.jsx)(n.p,{children:"If you've got a multi-page dashboard, and are hosting the additional config files yourself, then CORS rules will apply. A CORS error will look something like:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access to XMLHttpRequest at 'https://example.com/raw/my-config.yml' from origin 'http://dashy.local' has been blocked by CORS policy:\nNo 'Access-Control-Allow-Origin' header is present on the requested resource.\n"})}),"\n",(0,r.jsx)(n.p,{children:"The solution is to add the appropriate headers onto the target server, to allow it to accept requests from the origin where you're running Dashy."}),"\n",(0,r.jsxs)(n.p,{children:["If it is a remote service, that you do not have admin access to, then another option is to proxy the request. Either host your own, or use a publicly accessible service, like ",(0,r.jsx)(n.a,{href:"https://allorigins.win",children:"allorigins.win"}),", e.g: ",(0,r.jsx)(n.code,{children:"https://api.allorigins.win/raw?url=https://pastebin.com/raw/4tZpaJV5"}),". For git-based services specifically, there's ",(0,r.jsx)(n.a,{href:"https://raw.githack.com/",children:"raw.githack.com"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"build--memory-errors",children:"Build & memory errors"}),"\n",(0,r.jsx)(n.h3,{id:"yarn-error",children:"Yarn Error"}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1",children:"Issue #1"})]}),"\n",(0,r.jsxs)(n.p,{children:["First of all, check that you've got yarn installed correctly - see the ",(0,r.jsx)(n.a,{href:"https://classic.yarnpkg.com/en/docs/install",children:"yarn installation docs"})," for more info."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're getting an error about scenarios, then you've likely installed the wrong yarn... (you're ",(0,r.jsx)(n.a,{href:"https://github.com/yarnpkg/yarn/issues/2821",children:"not"})," the only one!). You can fix it by uninstalling, adding the correct repo, and reinstalling, for example, in Debian:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"sudo apt remove yarn"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | sudo apt-key add -"})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:'echo "deb https://dl.yarnpkg.com/debian/ stable main" | sudo tee /etc/apt/sources.list.d/yarn.list'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"sudo apt update && sudo apt install yarn"})}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"Alternatively, as a workaround, you have several options:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Try using ",(0,r.jsx)(n.a,{href:"https://www.npmjs.com/get-npm",children:"NPM"})," instead: So clone, cd, then run ",(0,r.jsx)(n.code,{children:"npm install"}),", ",(0,r.jsx)(n.code,{children:"npm run build"})," and ",(0,r.jsx)(n.code,{children:"npm start"})]}),"\n",(0,r.jsxs)(n.li,{children:["Try using ",(0,r.jsx)(n.a,{href:"https://www.docker.com/get-started",children:"Docker"})," instead, and all of the system setup and dependencies will already be taken care of. So from within the directory, just run ",(0,r.jsx)(n.code,{children:"docker build -t lissy93/dashy ."})," to build, and then use docker start to run the project, e.g: ",(0,r.jsx)(n.code,{children:"docker run -it -p 8080:8080 lissy93/dashy"})," (see the ",(0,r.jsx)(n.a,{href:"/docs/deployment#deploy-with-docker",children:"deploying docs"})," for more info)"]}),"\n"]}),"\n",(0,r.jsxs)(n.h3,{id:"yarn-build-fails-inside-the-container",children:[(0,r.jsx)(n.code,{children:"yarn build"})," fails inside the container"]}),"\n",(0,r.jsxs)(n.p,{children:["If you run ",(0,r.jsx)(n.code,{children:"docker exec yarn build"})," and get ",(0,r.jsx)(n.code,{children:"vite: not found"})," (or similar), it's because the published image ships only production dependencies. The build toolchain (vite, vue-tsc, sass, etc.) lives in ",(0,r.jsx)(n.code,{children:"devDependencies"})," and isn't installed in the runtime image."]}),"\n",(0,r.jsxs)(n.p,{children:["You almost certainly don't need to rebuild. Dashy's Express server reads ",(0,r.jsx)(n.code,{children:"user-data/conf.yml"})," on every request, so config changes show up on a page refresh, no rebuild required."]}),"\n",(0,r.jsxs)(n.p,{children:["If you genuinely need a fresh build (you've patched something in ",(0,r.jsx)(n.code,{children:"src/"}),"), do it on the host with ",(0,r.jsx)(n.code,{children:"yarn install && yarn build"}),", or build a custom image from a checkout of the repo."]}),"\n",(0,r.jsx)(n.h3,{id:"high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"}),"\n",(0,r.jsxs)(n.p,{children:["When the Dashy container first starts, it runs a Vue production build in parallel with the server. This is ",(0,r.jsx)(n.strong,{children:"a one-time cost per container start"}),", but it briefly uses around ",(0,r.jsx)(n.strong,{children:"1\u20131.5 GB of RAM and 100% of one CPU core"})," for anywhere from 30 seconds to several minutes (depending on host speed). On Pi-class hardware or VMs with less than 1 GB of RAM, this spike can be enough to lock up the host."]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.strong,{children:"To work around it:"})}),"\n",(0,r.jsxs)(n.ol,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Allocate at least 1 GB of RAM to the container"})," - 2 GB is recommended on Raspberry Pi or low-powered VMs. Anything below 512 MB is unlikely to complete the first build."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Set explicit Docker resource limits"})," so the build can't starve other services on the same host:\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"services:\n dashy:\n image: lissy93/dashy:latest\n deploy:\n resources:\n limits:\n memory: 2g\n cpus: '1.5'\n"})}),"\n"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Wait it out"})," - once the build completes, idle CPU drops to near zero and idle RAM is typically under 100 MB. If you watch ",(0,r.jsx)(n.code,{children:"docker stats"}),", you'll see the spike taper off."]}),"\n",(0,r.jsxs)(n.li,{children:["If the spike never tapers (i.e., Dashy stays at 100% CPU forever and never serves the page), see ",(0,r.jsx)(n.a,{href:"#ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Heap limit Allocation failed"})," below - that usually means the build was killed mid-way and is being retried."]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1585",children:"#1585"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/969",children:"#969"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1500",children:"#1500"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/877",children:"#877"})]}),"\n",(0,r.jsx)(n.h3,{id:"ineffective-mark-compacts-near-heap-limit-allocation-failed",children:"Ineffective mark-compacts near heap limit Allocation failed"}),"\n",(0,r.jsx)(n.p,{children:"If you see an error message, similar to:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"<--- Last few GCs ---\x3e\n\n[61:0x74533040] 229060 ms: Mark-sweep (reduce) 127.1 (236.9) -> 127.1 (137.4) MB, 5560.7 / 0.3 ms (average mu = 0.286, current mu = 0.011) allocation failure scavenge might not succeed\n\n<--- JS stacktrace ---\x3e\n\nFATAL ERROR: Reached heap limit Allocation failed - JavaScript heap out of memory\n"})}),"\n",(0,r.jsxs)(n.p,{children:["This is likely caused by insufficient memory allocation to the container. When the container first starts up, or has to rebuild, the memory usage spikes, and if there isn't enough memory, it may terminate. This can be specified with, for example: ",(0,r.jsx)(n.code,{children:"--memory=1024m"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Docker: Runtime options with Memory, CPUs, and GPUs"}),". For more context on what the spike is, see ",(0,r.jsx)(n.a,{href:"#high-cpu-or-ram-usage-on-startup",children:"High CPU or RAM Usage on Startup"})," above."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/380",children:"#380"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/350",children:"#350"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/297",children:"#297"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/349",children:"#349"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/510",children:"#510"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/511",children:"#511"})," and ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/834",children:"#834"})]}),"\n",(0,r.jsx)(n.h3,{id:"command-failed-with-signal-sigkill",children:'Command failed with signal "SIGKILL"'}),"\n",(0,r.jsxs)(n.p,{children:["In Docker, this can be caused by not enough memory. When the container first starts up, or has to rebuild, the memory usage spikes, and so a larger allocation may be required. This can be specified with, for example: ",(0,r.jsx)(n.code,{children:"--memory=1024m"}),". For more info, see ",(0,r.jsx)(n.a,{href:"https://docs.docker.com/config/containers/resource_constraints/",children:"Docker: Runtime options with Memory, CPUs, and GPUs"})]}),"\n",(0,r.jsxs)(n.p,{children:["See also ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/624",children:"#624"})]}),"\n",(0,r.jsx)(n.h3,{id:"node-sass-does-not-yet-support-your-current-environment",children:"Node Sass does not yet support your current environment"}),"\n",(0,r.jsxs)(n.p,{children:["Caused by node-sass's binaries being built for a for a different architecture\nTo fix this, just run: ",(0,r.jsx)(n.code,{children:"yarn rebuild node-sass"})]}),"\n",(0,r.jsx)(n.h3,{id:"unreachable-code-error",children:"Unreachable Code Error"}),"\n",(0,r.jsxs)(n.p,{children:["An error similar to: ",(0,r.jsx)(n.code,{children:"Fatal error in , line 0. Unreachable code, FailureMessage Object: 0xffe6c8ac. Illegal instruction (core dumped)"}),"\nIs related to a bug in a downstream package, see ",(0,r.jsx)(n.a,{href:"https://github.com/nodejs/docker-node/issues/1477",children:"nodejs/docker-node#1477"}),".\nUsually, updating your system and packages will resolve the issue."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/776",children:"#776"})]}),"\n",(0,r.jsx)(n.h3,{id:"error-cannot-find-module-_basevalues",children:"Error: Cannot find module './_baseValues'"}),"\n",(0,r.jsxs)(n.p,{children:["Clearing the cache should fix this: ",(0,r.jsx)(n.code,{children:"yarn cache clean"}),"\nIf the issue persists, remove (",(0,r.jsx)(n.code,{children:"rm -rf node_modules\\ yarn.lock"}),") and reinstall (",(0,r.jsx)(n.code,{children:"yarn"}),") node_modules"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"auth--oidc",children:"Auth & OIDC"}),"\n",(0,r.jsx)(n.h3,{id:"auth-validation-error-should-be-object",children:'Auth Validation Error: "should be object"'}),"\n",(0,r.jsxs)(n.p,{children:["In V 1.6.5 an update was made that in the future will become a breaking change. You will need to update you config to reflect this before V 2.0.0 is released. In the meantime, your previous config will continue to function normally, but you will see a validation warning. The change means that the structure of the ",(0,r.jsx)(n.code,{children:"appConfig.auth"})," object is now an object, which has a ",(0,r.jsx)(n.code,{children:"users"})," property."]}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/177",children:"this announcement"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"You can fix this by replacing:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"auth:\n- user: xxx\n hash: xxx\n"})}),"\n",(0,r.jsx)(n.p,{children:"with"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"auth:\n users:\n - user: xxx\n hash: xxx\n"})}),"\n",(0,r.jsx)(n.h3,{id:"keycloak-redirect-error",children:"Keycloak Redirect Error"}),"\n",(0,r.jsxs)(n.p,{children:["Check the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser's console output"}),", if you've not set any headers, you will likely see a CORS error here, which would be the source of the issue."]}),"\n",(0,r.jsxs)(n.p,{children:["You need to allow Dashy to make requests to Keycloak, and Keycloak to redirect to Dashy. The way you do this depends on how you're hosting these applications / which proxy you are using, and examples can be found in the ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Management Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"For example, add the access control header to Keycloak, like:"}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin [URL-of Dashy]"})}),"\n",(0,r.jsxs)(n.p,{children:["Note that for requests that transport sensitive info like credentials, setting the accept header to a wildcard (",(0,r.jsx)(n.code,{children:"*"}),") is not allowed - see ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS#requests_with_credentials",children:"MDN Docs"}),", so you will need to specify the actual URL."]}),"\n",(0,r.jsxs)(n.p,{children:["You should also ensure that Keycloak is correctly configured, with a user, realm and application, and be sure that you have set a valid redirect URL in Keycloak (",(0,r.jsx)(n.a,{href:"https://user-images.githubusercontent.com/1862727/148599768-db4ee4f8-72c5-402d-8f00-051d999e6267.png",children:"screenshot"}),")."]}),"\n",(0,r.jsxs)(n.p,{children:["For more details on how to set headers, see the ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Example Headers"})," in the management docs, or reference the documentation for your proxy."]}),"\n",(0,r.jsxs)(n.p,{children:["If you're running in Kubernetes, you will need to enable CORS ingress rules, see ",(0,r.jsx)(n.a,{href:"https://kubernetes.github.io/ingress-nginx/user-guide/nginx-configuration/annotations/#enable-cors",children:"docs"}),", e.g:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:'nginx.ingress.kubernetes.io/cors-allow-origin: "https://dashy.example.com"\nnginx.ingress.kubernetes.io/enable-cors: "true"\n'})}),"\n",(0,r.jsx)(n.p,{children:"See also: #479, #409, #507, #491, #341, #520"}),"\n",(0,r.jsx)(n.h3,{id:"oidc-or-keycloak-failure-on-numeric-client-ids",children:"OIDC or Keycloak failure on numeric client IDs"}),"\n",(0,r.jsxs)(n.p,{children:["If your IdP rejects the login with an ",(0,r.jsx)(n.em,{children:'"invalid client"'})," / ",(0,r.jsx)(n.em,{children:'"client not found"'})," error, and your ",(0,r.jsx)(n.code,{children:"clientId"})," is a long numeric value, the cause is almost certainly YAML number parsing."]}),"\n",(0,r.jsxs)(n.p,{children:["YAML parses unquoted numeric tokens as Numbers, and JavaScript can't represent integers larger than 2^53 (~16 digits) without losing precision. So an unquoted numeric ",(0,r.jsx)(n.code,{children:"clientId"})," will be silently truncated (e.g. ",(0,r.jsx)(n.code,{children:"918756876419824312"})," \u2192 ",(0,r.jsx)(n.code,{children:"918756876419824300"}),"), or - for very large values - converted to scientific notation (e.g. ",(0,r.jsx)(n.code,{children:"9.187568764198242e+37"}),"), and the IdP will reject it."]}),"\n",(0,r.jsxs)(n.p,{children:["The fix is to wrap the ",(0,r.jsx)(n.code,{children:"clientId"})," in quotes in your ",(0,r.jsx)(n.code,{children:"conf.yml"})," so it gets parsed as a string:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:'appConfig:\n auth:\n enableOidc: true\n oidc:\n clientId: "918756876419824312"\n endpoint: https://idp.example.com/\n'})}),"\n",(0,r.jsxs)(n.p,{children:["The same applies to ",(0,r.jsx)(n.code,{children:"auth.keycloak.clientId"}),". Dashy will print a warning in the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," when it detects a numeric ",(0,r.jsx)(n.code,{children:"clientId"}),", to help diagnose this."]}),"\n",(0,r.jsx)(n.p,{children:"See also: #1941"}),"\n",(0,r.jsx)(n.h3,{id:"redirect-loop-after-login",children:"Redirect loop after login"}),"\n",(0,r.jsxs)(n.p,{children:["Your ",(0,r.jsx)(n.code,{children:"endpoint"})," probably includes ",(0,r.jsx)(n.code,{children:".well-known/openid-configuration"}),". Drop everything from ",(0,r.jsx)(n.code,{children:".well-known"})," onwards"]}),"\n",(0,r.jsx)(n.h3,{id:"invalid_redirect_uri",children:"invalid_redirect_uri"}),"\n",(0,r.jsxs)(n.p,{children:["The redirect URI Authentik has registered for the provider doesn't exactly match the URL Dashy is being served from. Register both the bare URL and the trailing-slash version, and make sure the scheme matches (",(0,r.jsx)(n.code,{children:"http"})," vs ",(0,r.jsx)(n.code,{children:"https"}),")."]}),"\n",(0,r.jsx)(n.h3,{id:"login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403",children:"Login works in the browser but the dashboard refuses to save anything (403)"}),"\n",(0,r.jsxs)(n.p,{children:["Dashy's server is rejecting the id_token. Check Dashy's container logs for ",(0,r.jsx)(n.code,{children:"[auth-oidc] token verification failed"}),". Common causes:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Issuer mismatch"}),". Authentik is behind a reverse proxy that isn't sending ",(0,r.jsx)(n.code,{children:"X-Forwarded-Proto: https"}),", so its discovery document advertises ",(0,r.jsx)(n.code,{children:"http://"})," while you configured ",(0,r.jsx)(n.code,{children:"https://"})," in Dashy. Fix the proxy or set ",(0,r.jsx)(n.code,{children:"AUTHENTIK_HOST"}),"/",(0,r.jsx)(n.code,{children:"AUTHENTIK_LISTEN__TRUSTED_PROXY_CIDRS"})," on the Authentik containers"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Audience mismatch"}),". The ",(0,r.jsx)(n.code,{children:"aud"})," claim in the id_token is not ",(0,r.jsx)(n.code,{children:"dashy"}),". Confirm the provider's Client ID is exactly ",(0,r.jsx)(n.code,{children:"dashy"})," (no leading or trailing whitespace)"]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Dashy server can't reach Authentik"}),". The Dashy container fails to fetch the discovery document. Exec into the container and try ",(0,r.jsx)(n.code,{children:"wget -qO- https://auth.example.com/application/o/dashy/.well-known/openid-configuration"})]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Clock skew"}),". The middleware allows 30 seconds of drift. If a container's clock is further off than that, ",(0,r.jsx)(n.code,{children:"exp"}),"/",(0,r.jsx)(n.code,{children:"iat"})," checks fail"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"logged-in-but-no-admin-controls",children:"Logged in but no admin controls"}),"\n",(0,r.jsxs)(n.p,{children:["The id_token doesn't include the groups claim. Open browser devtools after logging in, find the call to ",(0,r.jsx)(n.code,{children:"/application/o/dashy/userinfo/"}),", and check the response. You should see a ",(0,r.jsx)(n.code,{children:"groups"})," array containing ",(0,r.jsx)(n.code,{children:"dashy-admins"}),". If not:"]}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["The ",(0,r.jsx)(n.code,{children:"groups"})," scope mapping doesn't exist, or is not attached to the provider's property mappings"]}),"\n",(0,r.jsxs)(n.li,{children:["The user is not in the ",(0,r.jsx)(n.code,{children:"dashy-admins"})," group"]}),"\n",(0,r.jsxs)(n.li,{children:["The conf.yml is missing ",(0,r.jsx)(n.code,{children:"groups"})," from ",(0,r.jsx)(n.code,{children:"scope:"})," and Authentik is therefore not sending it"]}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user",children:'Login works but Dashy errors on the callback with "OIDC signinCallback returned no user"'}),"\n",(0,r.jsx)(n.p,{children:'The id_token came back without a username claim. Confirm the provider has "profile" and "email" in its scopes and the "Include claims in id_token" is on'}),"\n",(0,r.jsx)(n.h3,{id:"sign-out-leaves-you-stuck-on-authentik",children:"Sign-out leaves you stuck on Authentik"}),"\n",(0,r.jsx)(n.p,{children:"Dashy redirects to Authentik's end-session endpoint on logout. If Authentik's invalidation flow prompts for confirmation (the default), that's expected - click through it. To skip the prompt entirely, change the provider's invalidation flow to one without a consent stage."}),"\n",(0,r.jsx)(n.h3,{id:"untrusted-certificate-from-authentik",children:"Untrusted certificate from Authentik"}),"\n",(0,r.jsx)(n.p,{children:"Self-signed certs make Dashy's server-side fetch of the discovery document fail. Use a real cert (Let's Encrypt, or your homelab CA installed into the Dashy image) for the Authentik hostname."}),"\n",(0,r.jsx)(n.h3,{id:"numeric-client_id-getting-truncated",children:"Numeric client_id getting truncated"}),"\n",(0,r.jsx)(n.p,{children:"Don't use numeric-only client IDs. If you must, wrap the value in quotes in conf.yml so YAML treats it as a string"}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"docker--image-issues",children:"Docker & image issues"}),"\n",(0,r.jsx)(n.h3,{id:"app-not-starting-after-update-to-204",children:"App Not Starting After Update to 2.0.4"}),"\n",(0,r.jsxs)(n.p,{children:["Version 2.0.4 introduced changes to how the config is read, and the app is build. If you were previously mounting ",(0,r.jsx)(n.code,{children:"/public"})," as a volume, then this will over-write the build app, preventing it from starting. The solution is to just pass in the file(s) / sub-directories that you need. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"volumes:\n- /srv/dashy/conf.yml:/app/user-data/conf.yml\n- /srv/dashy/item-icons:/app/public/item-icons\n"})}),"\n",(0,r.jsx)(n.h3,{id:"mount-type-mismatch",children:"Mount Type Mismatch"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Error response from daemon: ... mount through procfd: not a directory:\nAre you trying to mount a directory onto a file (or vice-versa)?\n"})}),"\n",(0,r.jsx)(n.p,{children:"This means the host side and container side of your volume don't agree on whether the target is a file or a directory."}),"\n",(0,r.jsxs)(n.p,{children:["Recommended pattern: mount a host directory onto ",(0,r.jsx)(n.code,{children:"/app/user-data"}),". The directory must exist on the host and contain at least a ",(0,r.jsx)(n.code,{children:"conf.yml"}),":"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"mkdir -p ~/dashy-data\ncp /path/to/your/conf.yml ~/dashy-data/conf.yml\ndocker run -d -p 8080:8080 -v ~/dashy-data:/app/user-data lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If you'd rather mount a single file (",(0,r.jsx)(n.code,{children:"-v ~/conf.yml:/app/user-data/conf.yml"}),"), the host path must be a file that already exists, otherwise Docker creates a directory in its place and you'll see this error."]}),"\n",(0,r.jsxs)(n.h3,{id:"dockerhub-toomanyrequests",children:["DockerHub ",(0,r.jsx)(n.code,{children:"toomanyrequests"})]}),"\n",(0,r.jsx)(n.p,{children:"This situation relates to error messages similar to one of the following, returned when pulling, updating or running the Docker container from Docker Hub."}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Continuing execution. Pulling image lissy93/dashy:release-1.6.0\nerror pulling image configuration: toomanyrequests\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"You have reached your pull rate limit. You may increase the limit by authenticating and upgrading: https://www.docker.com/increase-rate-limit\n"})}),"\n",(0,r.jsxs)(n.p,{children:["When DockerHub returns one of these errors, or a ",(0,r.jsx)(n.code,{children:"429"})," status, that means you've hit your rate limit. This was ",(0,r.jsx)(n.a,{href:"https://www.docker.com/blog/scaling-docker-to-serve-millions-more-developers-network-egress/",children:"introduced"})," last year, and prevents unauthenticated or free users from running docker pull more than 100 times per 6 hours.\nYou can ",(0,r.jsx)(n.a,{href:"https://www.docker.com/blog/checking-your-current-docker-pull-rate-limits-and-status/",children:"check your rate limit status"})," by looking for the ",(0,r.jsx)(n.code,{children:"ratelimit-remaining"})," header in any DockerHub responses."]}),"\n",(0,r.jsx)(n.h4,{id:"solution-1---use-an-alternate-container-registry",children:"Solution 1 - Use an alternate container registry"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Dashy is also available through GHCR, which at present does not have any hard limits. Just use ",(0,r.jsx)(n.code,{children:"docker pull ghcr.io/lissy93/dashy:latest"})," to fetch the image"]}),"\n",(0,r.jsxs)(n.li,{children:["You can also build the image from source, by cloning the repo, and running ",(0,r.jsx)(n.code,{children:"docker build -t dashy ."})," or use the pre-made docker compose"]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"solution-2---increase-your-rate-limits",children:"Solution 2 - Increase your rate limits"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"Logging in to DockerHub will increase your rate limit from 100 requests to 200 requests per 6 hour period"}),"\n",(0,r.jsx)(n.li,{children:"Upgrading to a Pro for $5/month will increase your image requests to 5,000 per day, and any plans above have no rate limits"}),"\n",(0,r.jsx)(n.li,{children:"Since rate limits are counted based on your IP address, proxying your requests, or using a VPN may work"}),"\n"]}),"\n",(0,r.jsx)(n.h3,{id:"old-image-tags-fail-to-pull",children:"Old image tags fail to pull"}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.code,{children:"docker pull"})," returns ",(0,r.jsx)(n.code,{children:"manifest unknown"})," or ",(0,r.jsx)(n.code,{children:"manifest for lissy93/dashy:arm32v7 not found"}),", the cause is a stale architecture-specific tag in your compose file or run command. Tags like ",(0,r.jsx)(n.code,{children:":arm32v7"}),", ",(0,r.jsx)(n.code,{children:":arm64v8"}),", and ",(0,r.jsx)(n.code,{children:":multi-arch"})," are no longer published."]}),"\n",(0,r.jsxs)(n.p,{children:["The ",(0,r.jsx)(n.code,{children:":latest"})," tag is now multi-arch and works on amd64, arm64, and arm/v7 (Raspberry Pi 2 and up) without you having to pick a variant. Just use:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-yaml",children:"image: lissy93/dashy:latest\n"})}),"\n",(0,r.jsxs)(n.p,{children:["Docker fetches the right architecture for your host automatically. To pin a version, use a semver tag, e.g. ",(0,r.jsx)(n.code,{children:"lissy93/dashy:3.2.14"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"healthcheck-failing-in-docker",children:"Healthcheck Failing in Docker"}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.code,{children:"docker ps"})," shows the Dashy container as ",(0,r.jsx)(n.code,{children:"unhealthy"}),", the periodic healthcheck (",(0,r.jsx)(n.code,{children:"node services/healthcheck.js"}),", run every 5 minutes by default) couldn't reach the local server."]}),"\n",(0,r.jsx)(n.h4,{id:"ssl-enabled-dashy",children:"SSL-enabled Dashy"}),"\n",(0,r.jsxs)(n.p,{children:["The healthcheck reads the same cert paths as the main server (",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-pub.pem"})," and ",(0,r.jsx)(n.code,{children:"/etc/ssl/certs/dashy-priv.key"}),") to detect whether to probe HTTPS or HTTP. If you've mounted certs at non-default paths via ",(0,r.jsx)(n.code,{children:"SSL_PUB_KEY_PATH"})," / ",(0,r.jsx)(n.code,{children:"SSL_PRIV_KEY_PATH"}),", the healthcheck will pick those up automatically as long as those env vars are set in the container environment (not just at run time)."]}),"\n",(0,r.jsx)(n.h4,{id:"custom-port",children:"Custom port"}),"\n",(0,r.jsxs)(n.p,{children:["If you've set ",(0,r.jsx)(n.code,{children:"PORT"})," to override the default 8080, the healthcheck honors the same env var, so it should work without changes. Make sure ",(0,r.jsx)(n.code,{children:"PORT"})," is set in the container environment, not just in the host-side Docker port mapping."]}),"\n",(0,r.jsx)(n.h4,{id:"container-is-unhealthy-past-the-grace-period",children:"Container is unhealthy past the grace period"}),"\n",(0,r.jsxs)(n.p,{children:["The healthcheck has a 20s ",(0,r.jsx)(n.code,{children:"start-period"})," after which failures start counting. The image is prebuilt, so startup is just ",(0,r.jsx)(n.code,{children:"node server.js"})," binding to a port - fast even on a Pi. If the container is still ",(0,r.jsx)(n.code,{children:"unhealthy"})," past the grace period, the server has likely crashed. Check ",(0,r.jsx)(n.code,{children:"docker logs "})," for the real error (usually a malformed ",(0,r.jsx)(n.code,{children:"conf.yml"})," or a missing ",(0,r.jsx)(n.code,{children:"user-data"})," mount)."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/1410",children:"#1410"})]}),"\n",(0,r.jsx)(n.h3,{id:"docker-login-fails-on-ubuntu",children:"Docker Login Fails on Ubuntu"}),"\n",(0,r.jsxs)(n.p,{children:["Run ",(0,r.jsx)(n.code,{children:"sudo apt install gnupg2 pass && gpg2 -k"})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"styles-and-assets-not-updating",children:"Styles and Assets not Updating"}),"\n",(0,r.jsxs)(n.p,{children:["If you find that your styles and other visual assets work when visiting ",(0,r.jsx)(n.code,{children:"ip:port"})," by not ",(0,r.jsx)(n.code,{children:"dashy.domain.com"}),", then this is usually caused by caching. In your browser, do a hard-refresh (",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"F5"}),"). If you use Cloudflare, then you can clear the cache through the management console, or set the cache level to Bypass for certain files, under the Rules tab."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"config-validation-errors",children:"Config Validation Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The configuration file is validated against ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"Dashy's Schema"})," using AJV."]}),"\n",(0,r.jsxs)(n.p,{children:["First, check that your syntax is valid, using ",(0,r.jsx)(n.a,{href:"https://codebeautify.org/yaml-validator/",children:"YAML Validator"})," or ",(0,r.jsx)(n.a,{href:"https://codebeautify.org/jsonvalidator",children:"JSON Validator"}),". If the issue persists, then take a look at the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"schema"}),", and verify that the field you are trying to add/ modify matches the required format. You can also use ",(0,r.jsx)(n.a,{href:"https://www.jsonschemavalidator.net/s/JFUj7X9J",children:"this tool"})," to validate your JSON config against the schema, or run ",(0,r.jsx)(n.code,{children:"yarn validate-config"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you're trying to use a recently released feature, and are getting a warning, this is likely because you've not yet updated the the current latest version of Dashy."}),"\n",(0,r.jsx)(n.p,{children:"If the issue still persists, you should raise an issue."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"invalid-host-header-while-running-through-ngrok",children:"Invalid Host Header while running through ngrok"}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.a,{href:"https://ngrok.com/docs#http-host-header",children:"-host-header"})," flag, e.g. ",(0,r.jsx)(n.code,{children:'ngrok http 8080 -host-header="localhost:8080"'})]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"warnings-in-the-console-during-deploy",children:"Warnings in the Console during deploy"}),"\n",(0,r.jsx)(n.p,{children:"Please acknowledge the difference between errors and warnings before raising an issue about messages in the console. It's not unusual to see warnings about a new version of a certain package being available, an asset bundle bing oversized or a service worker not yet having a cache. These shouldn't have any impact on the running application, so please don't raise issues about these unless it directly relates to a bug or issue you're experiencing. Errors on the other hand should not appear in the console, and they are worth looking into further."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"status-checks-failing",children:"Status Checks Failing"}),"\n",(0,r.jsx)(n.p,{children:"If you're using status checks, and despite a given service being online, the check is displaying an error, there are a couple of things you can look at:"}),"\n",(0,r.jsxs)(n.p,{children:["If your service requires requests to include any authorization in the headers, then use the ",(0,r.jsx)(n.code,{children:"statusCheckHeaders"})," property, as described in the ",(0,r.jsx)(n.a,{href:"/docs/status-indicators#setting-custom-headers",children:"docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are still having issues, it may be because your target application is blocking requests from Dashy's IP. This is a ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"CORS error"}),", and can be fixed by setting the headers on your target app, to include:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://location-of-dashy/\nVary: Origin\n"})}),"\n",(0,r.jsxs)(n.p,{children:["If the URL you are checking has an unsigned certificate, or is not using HTTPS, then you may need to disable the rejection of insecure requests. This can be done by setting ",(0,r.jsx)(n.code,{children:"statusCheckAllowInsecure"})," to true for a given item."]}),"\n",(0,r.jsxs)(n.p,{children:["If your service is online, but responds with a status code that is not in the 2xx range, then you can use ",(0,r.jsx)(n.code,{children:"statusCheckAcceptCodes"})," to set an accepted status code."]}),"\n",(0,r.jsxs)(n.p,{children:["If you get an error, like ",(0,r.jsx)(n.code,{children:"Service Unavailable: Server resulted in a fatal error"}),", even when it's definitely online, this is most likely caused by missing the protocol. Don't forget to include ",(0,r.jsx)(n.code,{children:"https://"})," (or whatever protocol) before the URL, and ensure that if needed, you've specified the port."]}),"\n",(0,r.jsxs)(n.p,{children:["Running Dashy in HOST network mode, instead of BRIDGE will allow status check access to other services in HOST mode. For more info, see ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/445",children:"#445"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"If you have firewall rules configured, then ensure that they don't prevent Dashy from making requests to the other services you are trying to access."}),"\n",(0,r.jsxs)(n.p,{children:["Currently, the status check needs a page to be rendered, so if this URL in your browser does not return anything, then status checks will not work. This may be modified in the future, but in the meantime, a fix would be to make your own status service, which just checks if your app responds with whatever code you'd like, and then return a 200 plus renders an arbitrary message. Then just point ",(0,r.jsx)(n.code,{children:"statusCheckUrl"})," to your custom page."]}),"\n",(0,r.jsxs)(n.p,{children:["For further troubleshooting, use an application like ",(0,r.jsx)(n.a,{href:"https://postman.com",children:"Postman"})," to diagnose the issue. Set the parameter to ",(0,r.jsx)(n.code,{children:"GET"}),", and then make a call to: ",(0,r.jsx)(n.code,{children:"https://[url-of-dashy]/status-check/?&url=[service-url]"}),". Where the service URL must have first been encoded (e.g. with ",(0,r.jsx)(n.code,{children:"encodeURIComponent()"})," or ",(0,r.jsx)(n.a,{href:"https://www.urlencoder.io/",children:"urlencoder.io"}),")"]}),"\n",(0,r.jsx)(n.p,{children:"If you're serving Dashy though a CDN, instead of using the Node server or Docker image, then the Node endpoint that makes requests will not be available to you, and all requests will fail. A workaround for this may be implemented in the future, but in the meantime, your only option is to use the Docker or Node deployment method."}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"widgets",children:"Widgets"}),"\n",(0,r.jsx)(n.h3,{id:"widget-errors",children:"Widget Errors"}),"\n",(0,r.jsx)(n.h4,{id:"find-error-message",children:"Find Error Message"}),"\n",(0,r.jsxs)(n.p,{children:["If an error occurs when fetching or rendering results, you will see a short message in the UI. If that message doesn't adequately explain the problem, then you can ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting#how-to-open-browser-console",children:"open the browser console"})," to see more details."]}),"\n",(0,r.jsx)(n.h4,{id:"check-config",children:"Check Config"}),"\n",(0,r.jsx)(n.p,{children:"Before proceeding, ensure that if the widget requires auth your API is correct, and for custom widgets, double check that the URL and protocol is correct."}),"\n",(0,r.jsx)(n.h4,{id:"timeout-error",children:"Timeout Error"}),"\n",(0,r.jsxs)(n.p,{children:["If the error message in the console includes: ",(0,r.jsx)(n.code,{children:"Error: timeout of 500ms exceeded"}),", then your Glances endpoint is slower to respond than expected. You can fix this by ",(0,r.jsx)(n.a,{href:"/docs/widgets#setting-timeout",children:"setting timeout"})," to a larger value. This is done on each widget, with the ",(0,r.jsx)(n.code,{children:"timeout"})," attribute, and is specified in ms. E.g. ",(0,r.jsx)(n.code,{children:"timeout: 5000"})," would only fail if no response is returned within 5 seconds."]}),"\n",(0,r.jsx)(n.h4,{id:"cors-error",children:"CORS error"}),"\n",(0,r.jsxs)(n.p,{children:["If the console message mentions to corss-origin blocking, then this is a CORS error, see: ",(0,r.jsx)(n.a,{href:"#widget-cors-errors",children:"Fixing Widget CORS Errors"})]}),"\n",(0,r.jsx)(n.h4,{id:"more-info",children:"More Info"}),"\n",(0,r.jsx)(n.p,{children:"If you're able to, you can find more information about why the request may be failing in the Dev Tools under the Network tab, and you can ensure your endpoint is correct and working using a tool like Postman."}),"\n",(0,r.jsx)(n.h3,{id:"widget-cors-errors",children:"Widget CORS Errors"}),"\n",(0,r.jsxs)(n.p,{children:["The most common widget issue is a CORS error. This is a browser security mechanism which prevents the client-side app (Dashy) from from accessing resources on a remote origin, without that server's explicit permission (e.g. with headers like Access-Control-Allow-Origin). See the MDN Docs for more info: ",(0,r.jsx)(n.a,{href:"https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS",children:"Cross-Origin Resource Sharing"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"There are several ways to fix a CORS error:"}),"\n",(0,r.jsx)(n.h4,{id:"option-1---ensure-correct-protocol",children:"Option 1 - Ensure Correct Protocol"}),"\n",(0,r.jsx)(n.p,{children:"You will get a CORS error if you try and access a http service from a https source. So ensure that the URL you are requesting has the right protocol, and is correctly formatted."}),"\n",(0,r.jsx)(n.h4,{id:"option-2---set-headers",children:"Option 2 - Set Headers"}),"\n",(0,r.jsxs)(n.p,{children:["If you have control over the destination (e.g. for a self-hosted service), then you can simply apply the correct headers.\nAdd the ",(0,r.jsx)(n.code,{children:"Access-Control-Allow-Origin"})," header, with the value of either ",(0,r.jsx)(n.code,{children:"*"})," to allow requests from anywhere, or more securely, the host of where Dashy is served from. For example:"]}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: https://url-of-dashy.local\n"})}),"\n",(0,r.jsx)(n.p,{children:"or"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-text",children:"Access-Control-Allow-Origin: *\n"})}),"\n",(0,r.jsxs)(n.p,{children:["For more info on how to set headers, see: ",(0,r.jsx)(n.a,{href:"/docs/management#setting-headers",children:"Setting Headers"})," in the management docs"]}),"\n",(0,r.jsx)(n.h4,{id:"option-3---proxying-request",children:"Option 3 - Proxying Request"}),"\n",(0,r.jsxs)(n.p,{children:["You can route requests through Dashy's built-in CORS proxy. Instructions and more details can be found ",(0,r.jsx)(n.a,{href:"/docs/widgets#proxying-requests",children:"here"}),". If you don't have control over the target origin, and you are running Dashy either through Docker, with the Node server or on Netlify, then this solution will work for you."]}),"\n",(0,r.jsxs)(n.p,{children:["Just add the ",(0,r.jsx)(n.code,{children:"useProxy: true"})," option to the failing widget."]}),"\n",(0,r.jsx)(n.h4,{id:"option-4---use-a-plugin",children:"Option 4 - Use a plugin"}),"\n",(0,r.jsxs)(n.p,{children:["For testing purposes, you can use an addon, which will disable the CORS checks. You can get the Allow-CORS extension for ",(0,r.jsx)(n.a,{href:"https://chrome.google.com/webstore/detail/allow-cors-access-control/lhobafahddgcelffkeicbaginigeejlf?hl=en-US",children:"Chrome"})," or ",(0,r.jsx)(n.a,{href:"https://addons.mozilla.org/en-US/firefox/addon/access-control-allow-origin/",children:"Firefox"}),", more details ",(0,r.jsx)(n.a,{href:"https://mybrowseraddon.com/access-control-allow-origin.html",children:"here"})]}),"\n",(0,r.jsxs)(n.h3,{id:"cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound-",children:["CORS Proxy ",(0,r.jsx)(n.code,{children:"connect ECONNREFUSED ..."})," or ",(0,r.jsx)(n.code,{children:"getaddrinfo ENOTFOUND ..."})]}),"\n",(0,r.jsxs)(n.p,{children:["The target host is unreachable from the Dashy container. If the target is on the same host as Dashy, ",(0,r.jsxs)(n.strong,{children:["don't use ",(0,r.jsx)(n.code,{children:"localhost"})]})," - inside a Docker container that resolves to the container itself, not the host. Use the host's LAN IP, the Docker bridge gateway, or ",(0,r.jsx)(n.code,{children:"host.docker.internal"})," (on Docker Desktop). If the target is on a different Docker network, attach Dashy to that network too."]}),"\n",(0,r.jsxs)(n.h3,{id:"cors-proxy-target-url-host--is-blocked--must-use-http-or-https",children:["CORS Proxy ",(0,r.jsx)(n.code,{children:"Target-URL host '...' is blocked"})," / ",(0,r.jsx)(n.code,{children:"must use http:// or https://"})]}),"\n",(0,r.jsx)(n.p,{children:"To prevent the CORS proxy from being abused as a Server-Side Request Forgery vector, Dashy refuses to proxy a small number of host/scheme combinations by default:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Cloud instance metadata services"})," - ",(0,r.jsx)(n.code,{children:"169.254.169.254"}),", ",(0,r.jsx)(n.code,{children:"metadata.google.internal"}),", and the matching IPv6 forms. These are reserved magic addresses on AWS, Azure, GCP, DigitalOcean, Hetzner, Oracle Cloud and most other providers. A widget that successfully fetches them on a cloud-hosted Dashy can leak the host's IAM credentials, so they're blocked unconditionally."]}),"\n",(0,r.jsxs)(n.li,{children:[(0,r.jsx)(n.strong,{children:"Non-HTTP(S) schemes"})," - ",(0,r.jsx)(n.code,{children:"file://"}),", ",(0,r.jsx)(n.code,{children:"ftp://"}),", ",(0,r.jsx)(n.code,{children:"gopher://"}),", ",(0,r.jsx)(n.code,{children:"javascript:"}),", ",(0,r.jsx)(n.code,{children:"data:"}),", and similar. The proxy is meant for HTTP APIs only."]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:"If you're running Dashy in a fully isolated/private environment and you've deliberately decided you want to allow these (for example, you genuinely need to query your cloud provider's metadata API from a widget), you can opt out of all proxy restrictions by setting the environment variable:"}),"\n",(0,r.jsx)(n.pre,{children:(0,r.jsx)(n.code,{className:"language-bash",children:"DANGEROUSLY_DISABLE_PROXY_RESTRICTIONS=true\n"})}),"\n",(0,r.jsxs)(n.p,{children:["The variable is named so loudly because flipping it on a Dashy instance that's exposed to anything other than fully trusted users re-opens the SSRF surface - anyone who can hit ",(0,r.jsx)(n.code,{children:"/cors-proxy"})," can then use Dashy as a relay to reach internal services. ",(0,r.jsx)(n.strong,{children:"Don't set it on cloud-hosted or internet-exposed deployments."})]}),"\n",(0,r.jsx)(n.p,{children:"Note that this is an all-or-nothing escape hatch, not a per-host allowlist. If you only need to reach one specific host that's currently blocked, please open a feature request describing the use case."}),"\n",(0,r.jsx)(n.h3,{id:"widget-shows-error-incorrectly",children:"Widget Shows Error Incorrectly"}),"\n",(0,r.jsx)(n.p,{children:"When there's an error fetching or displaying a widgets data, then it will be highlighted in yellow, and a message displayed on the UI."}),"\n",(0,r.jsxs)(n.p,{children:["In some instances, this is a false positive, and the widget is actually functioning correctly.\nIf this is the case, you can disable the UI error message of a given widget by setting: ",(0,r.jsx)(n.code,{children:"ignoreErrors: true"})]}),"\n",(0,r.jsx)(n.h3,{id:"weather-forecast-widget-401",children:"Weather Forecast Widget 401"}),"\n",(0,r.jsx)(n.p,{children:"A 401 error means your API key is invalid, it is not an issue with Dashy."}),"\n",(0,r.jsxs)(n.p,{children:["Usually this happens due to an error in your config. If you're unsure, copy and paste the ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather",children:"example"})," config, replacing the API key with your own."]}),"\n",(0,r.jsxs)(n.p,{children:["Check that ",(0,r.jsx)(n.code,{children:"apiKey"})," is correctly specified, and nested within ",(0,r.jsx)(n.code,{children:"options"}),". Ensure your input city is valid."]}),"\n",(0,r.jsxs)(n.p,{children:["To test your API key, try making a request to ",(0,r.jsx)(n.code,{children:"https://api.openweathermap.org/data/2.5/weather?q=London&appid=[your-api-key]"})]}),"\n",(0,r.jsxs)(n.p,{children:["If ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather-forecast",children:"Weather widget"})," is working fine, but you are getting a ",(0,r.jsx)(n.code,{children:"401"})," for the ",(0,r.jsx)(n.a,{href:"/docs/widgets#weather-forecast",children:"Weather Forecast widget"}),", then this is also an OWM API key issue.\nSince the forecasting API requires an upgraded plan. ULPT: You can get a free, premium API key by filling in ",(0,r.jsx)(n.a,{href:"https://home.openweathermap.org/students",children:"this form"}),". It's a student plan, but there's no verification to check that you are still a student."]}),"\n",(0,r.jsx)(n.p,{children:"A future update will be pushed out, to use a free weather forecasting API."}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/803",children:"#803"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/789",children:"#789"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/577",children:"#577"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/621",children:"#621"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/578",children:"#578"}),", ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/discussions/806",children:"#806"})]}),"\n",(0,r.jsx)(n.h3,{id:"widget-displaying-inaccurate-data",children:"Widget Displaying Inaccurate Data"}),"\n",(0,r.jsx)(n.p,{children:"If any widget is not displaying the data you expect, first confirm that your config is correct, then try manually calling the API endpoint."}),"\n",(0,r.jsxs)(n.p,{children:["If the raw API output is correct, yet the widget is rendering incorrect results, then it is likely a bug, and a ticket should be raised. You can start to debug the issue, by looking at the widget's code (",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/components/Widgets",children:"here"}),"), and the browser console + networking tab."]}),"\n",(0,r.jsxs)(n.p,{children:["If the API itself is returning incorrect, incomplete or inaccurate data then an issue needs to be raised ",(0,r.jsx)(n.strong,{children:"with the API provider"})," (not Dashy!). You can find the API provider included within the widget docs, or for a full list see the ",(0,r.jsx)(n.a,{href:"/docs/privacy#widgets",children:"Privacy Docs"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["See also: ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/807",children:"#807"})," (re, domain monitor)"]}),"\n",(0,r.jsxs)(n.h3,{id:"public-ip-widget-not-working-for-ipinfo-or-ipquery-providers",children:["Public IP Widget not working for ",(0,r.jsx)(n.code,{children:"ipinfo"})," or ",(0,r.jsx)(n.code,{children:"ipquery"})," providers"]}),"\n",(0,r.jsxs)(n.p,{children:["If you've set ",(0,r.jsx)(n.code,{children:"options.provider"})," to either ",(0,r.jsx)(n.code,{children:"ipinfo"})," and ",(0,r.jsx)(n.code,{children:"ipquery"})," and the requests are failing, it's likley that they're being blocked. Check your adblocker (uBlock, PrivacyBadger, etc), DNS block lists (PiHole, AdGuard, etc). Or, try proxying the request (with ",(0,r.jsx)(n.code,{children:"useProxy: true"}),") or just try a different provider."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"font-awesome-icons-not-displaying",children:"Font Awesome Icons not Displaying"}),"\n",(0,r.jsxs)(n.p,{children:["Usually, Font Awesome will be automatically enabled if one or more of your icons are using Font-Awesome. If this is not happening, then you can always manually enable (or disable) Font Awesome by setting: ",(0,r.jsx)(n.a,{href:"/docs/configuring#appconfig-optional",children:(0,r.jsx)(n.code,{children:"appConfig"})}),".",(0,r.jsx)(n.code,{children:"enableFontAwesome"})," to ",(0,r.jsx)(n.code,{children:"true"}),"."]}),"\n",(0,r.jsxs)(n.p,{children:["If you are trying to use a premium icon, then you must have a ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/plans",children:"Pro License"}),". You'll then need to specify your Pro plan API key under ",(0,r.jsx)(n.code,{children:"appConfig.fontAwesomeKey"}),". You can find this key, by logging into your FA account, navigate to Account \u2192 ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/kits",children:"Kits"})," \u2192 New Kit \u2192 Copy Kit Code. The code is a 10-digit alpha-numeric code, and is also visible within the new kit's URL, for example: ",(0,r.jsx)(n.code,{children:"81e48ce079"}),"."]}),"\n",(0,r.jsx)(n.p,{align:"center",children:(0,r.jsx)(n.img,{src:"https://i.ibb.co/hZ0D9vs/where-do-i-find-my-font-awesome-key.png",width:"600"})}),"\n",(0,r.jsxs)(n.p,{children:["Be sure that you're specifying the icon category and name correctly. You're icon should look be ",(0,r.jsx)(n.code,{children:"[category] fa-[icon-name]"}),". The following categories are supported: ",(0,r.jsx)(n.code,{children:"far"})," ",(0,r.jsx)(n.em,{children:"(regular)"}),", ",(0,r.jsx)(n.code,{children:"fas"})," ",(0,r.jsx)(n.em,{children:"(solid)"}),", ",(0,r.jsx)(n.code,{children:"fal"}),(0,r.jsx)(n.em,{children:"(light)"}),", ",(0,r.jsx)(n.code,{children:"fad"})," ",(0,r.jsx)(n.em,{children:"(duo-tone)"})," and ",(0,r.jsx)(n.code,{children:"fab"}),(0,r.jsx)(n.em,{children:"(brands)"}),". With the exception of brands, you'll usually want all your icons to be in from same category, so they look uniform."]}),"\n",(0,r.jsxs)(n.p,{children:["Ensure the icon you are trying to use, is available within ",(0,r.jsx)(n.a,{href:"https://fontawesome.com/v5/search",children:"FontAwesome Version 5"})," (we've not yet upgraded to V6, as it works a little differently)."]}),"\n",(0,r.jsxs)(n.p,{children:["Examples: ",(0,r.jsx)(n.code,{children:"fab fa-raspberry-pi"}),", ",(0,r.jsx)(n.code,{children:"fas fa-database"}),", ",(0,r.jsx)(n.code,{children:"fas fa-server"}),", ",(0,r.jsx)(n.code,{children:"fas fa-ethernet"})]}),"\n",(0,r.jsxs)(n.p,{children:["Finally, check the ",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"browser console"})," for any error messages, and raise a ticket if the issue persists."]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"copy-to-clipboard-not-working",children:"Copy to Clipboard not Working"}),"\n",(0,r.jsxs)(n.p,{children:["If the copy to clipboard feature (either under Config --\x3e Export, or Item --\x3e Copy URL) isn't functioning as expected, first check the browser console. If you see ",(0,r.jsx)(n.code,{children:"TypeError: Cannot read properties of undefined (reading 'writeText')"})," then this feature is not supported by your browser.\nThe most common reason for this, is if you not running the app over HTTPS. Copying to the clipboard requires the app to be running in a secure origin / aka have valid HTTPS cert. You can read more about this ",(0,r.jsx)(n.a,{href:"https://stackoverflow.com/a/71876238/979052",children:"here"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"As a workaround, you could either:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Highlight the text and copy / ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"C"})]}),"\n",(0,r.jsxs)(n.li,{children:["Or setup SSL - ",(0,r.jsx)(n.a,{href:"/docs/management#ssl-certificates",children:"here's a guide"})," on doing so"]}),"\n"]}),"\n",(0,r.jsx)(n.hr,{}),"\n",(0,r.jsx)(n.h2,{id:"how-to--reference",children:"How-To / Reference"}),"\n",(0,r.jsx)(n.h3,{id:"how-to-reset-local-settings",children:"How to Reset Local Settings"}),"\n",(0,r.jsx)(n.p,{children:"Some settings are stored locally, in the browser's storage."}),"\n",(0,r.jsx)(n.p,{children:"In some instances cached assets can prevent your settings from being updated, in which case you may wish to reset local data."}),"\n",(0,r.jsx)(n.p,{children:'To clear all local data from the UI, head to the Config Menu, then click "Reset Local Settings", and Confirm when prompted.\nThis will not affect your config file. But be sure that you keep a backup of your config, if you\'ve not written changes it to disk.'}),"\n",(0,r.jsxs)(n.p,{children:["You can also view any and all data that Dashy is storing, using the developer tools. Open your browser's dev tools (usually ",(0,r.jsx)(n.kbd,{children:"F12"}),"), in Chromium head to the Application tab, or in Firefox go to the Storage tab. Select Local Storage, then scroll down the the URL Dashy is running on. You should now see all data being stored, and you can select and delete any fields you wish."]}),"\n",(0,r.jsxs)(n.p,{children:["For a full list of all data that may be cached, see the ",(0,r.jsx)(n.a,{href:"/docs/privacy#browser-storage",children:"Privacy Docs"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"how-to-make-a-bug-report",children:"How to make a bug report"}),"\n",(0,r.jsx)(n.h4,{id:"step-1---where-to-open-issues",children:"Step 1 - Where to open issues"}),"\n",(0,r.jsxs)(n.p,{children:["You will need a GitHub account in order to raise a ticket. You can then ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/issues/new?assignees=lissy93&labels=%F0%9F%90%9B+Bug&template=bug.yml&title=%5BBUG%5D+%3Ctitle%3E",children:"click here"})," to open a new bug report."]}),"\n",(0,r.jsx)(n.h4,{id:"step-2---checking-its-not-already-covered",children:"Step 2 - Checking it's not already covered"}),"\n",(0,r.jsx)(n.p,{children:"Before submitting, please check that:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"A similar ticket has not previously been opened"}),"\n",(0,r.jsxs)(n.li,{children:["The issue is not covered in the ",(0,r.jsx)(n.a,{href:"/docs/troubleshooting",children:"troubleshooting guide"})," or ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/docs#readme",children:"docs"})]}),"\n"]}),"\n",(0,r.jsx)(n.h4,{id:"step-3---describe-the-issue",children:"Step 3 - Describe the Issue"}),"\n",(0,r.jsx)(n.p,{children:"Your ticket will likely be dealt with more effectively if you can explain the issue clearly, and provide all relevant supporting material."}),"\n",(0,r.jsx)(n.p,{children:"Complete the fields, asking for your environment info and version of Dashy.\nThen describe the issue, briefly explaining the steps to reproduce, expected outcome and actual outcome."}),"\n",(0,r.jsx)(n.h4,{id:"step-4---provide-supporting-info",children:"Step 4 - Provide Supporting Info"}),"\n",(0,r.jsx)(n.p,{children:"Where relevant please also include:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:"A screenshot of the issue"}),"\n",(0,r.jsx)(n.li,{children:"The relevant parts of your config file"}),"\n",(0,r.jsxs)(n.li,{children:["Logs\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["If client-side issue, then include the browser logs (",(0,r.jsx)(n.a,{href:"#how-to-open-browser-console",children:"see how"}),")"]}),"\n",(0,r.jsx)(n.li,{children:"If server-side / during deployment, include the terminal output"}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,r.jsx)(n.p,{children:(0,r.jsx)(n.em,{children:"Take care to redact any personal info, (like IP addresses, auth hashes or API keys)."})}),"\n",(0,r.jsx)(n.h4,{id:"step-5---fix-released",children:"Step 5 - Fix Released"}),"\n",(0,r.jsx)(n.p,{children:"A maintainer will aim to respond within 48 hours.\nThe timeframe for resolving your issue, will vary depending on severity of the bug and the complexity of the fix.\nYou will be notified on your ticket, when a fix has been released."}),"\n",(0,r.jsxs)(n.p,{children:["Finally, be sure to remain respectful to other users and project maintainers, in line with the ",(0,r.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.github/CODE_OF_CONDUCT#contributor-covenant-code-of-conduct",children:"Contributor Covenant Code of Conduct"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"how-to-open-browser-console",children:"How-To Open Browser Console"}),"\n",(0,r.jsx)(n.p,{children:"When raising a bug, one crucial piece of info needed is the browser's console output. This will help the developer diagnose and fix the issue."}),"\n",(0,r.jsxs)(n.p,{children:['If you\'ve been asked for this info, but are unsure where to find it, then it is under the "Console" tab, in the browsers developer tools, which can be opened with ',(0,r.jsx)(n.kbd,{children:"F12"}),". You can right-click the console, and select Save As to download the log."]}),"\n",(0,r.jsx)(n.p,{children:"To open dev tools, and jump straight to the console:"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsxs)(n.li,{children:["Win / Linux: ",(0,r.jsx)(n.kbd,{children:"Ctrl"})," + ",(0,r.jsx)(n.kbd,{children:"Shift"})," + ",(0,r.jsx)(n.kbd,{children:"J"})]}),"\n",(0,r.jsxs)(n.li,{children:["MacOS: ",(0,r.jsx)(n.kbd,{children:"Cmd"})," + ",(0,r.jsx)(n.kbd,{children:"Option"})," + ",(0,r.jsx)(n.kbd,{children:"J"})]}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more detailed walk through, see ",(0,r.jsx)(n.a,{href:"https://support.shortpoint.com/support/solutions/articles/1000222881-save-browser-console-file",children:"this article"}),"."]}),"\n",(0,r.jsx)(n.h3,{id:"git-contributions-not-displaying",children:"Git Contributions not Displaying"}),"\n",(0,r.jsxs)(n.p,{children:["If you've contributed to Dashy (or any other project), but your contributions are not showing up on your GH profile, or in Dashy's ",(0,r.jsx)(n.a,{href:"/docs/credits",children:"Credits Page"}),", then this is likely a git config issue."]}),"\n",(0,r.jsxs)(n.p,{children:["These statistics are generated using the username / email associated with commits. This info needs to be setup on your local machine using ",(0,r.jsx)(n.a,{href:"https://git-scm.com/docs/git-config",children:(0,r.jsx)(n.code,{children:"git config"})}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Run the following commands (replacing name + email with your info):"}),"\n",(0,r.jsxs)(n.ul,{children:["\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:'git config --global user.name "John Doe"'})}),"\n",(0,r.jsx)(n.li,{children:(0,r.jsx)(n.code,{children:"git config --global user.email johndoe@example.com"})}),"\n"]}),"\n",(0,r.jsxs)(n.p,{children:["For more info, see ",(0,r.jsx)(n.a,{href:"https://git-scm.com/book/en/v2/Getting-Started-First-Time-Git-Setup",children:"Git First Time Setup Docs"}),"."]}),"\n",(0,r.jsx)(n.p,{children:"Note that only contributions to the master / main branch or a project are counted"})]})}function c(e={}){const{wrapper:n}={...(0,o.R)(),...e.components};return n?(0,r.jsx)(n,{...e,children:(0,r.jsx)(h,{...e})}):h(e)}},8453(e,n,s){s.d(n,{R:()=>t,x:()=>a});var i=s(6540);const r={},o=i.createContext(r);function t(e){const n=i.useContext(o);return i.useMemo(function(){return"function"==typeof e?e(n):{...n,...e}},[n,e])}function a(e){let n;return n=e.disableParentContext?"function"==typeof e.components?e.components(r):e.components||r:t(e.components),i.createElement(o.Provider,{value:n},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/a3713279.7f20aedd.js b/assets/js/a3713279.55b1c4a8.js similarity index 99% rename from assets/js/a3713279.7f20aedd.js rename to assets/js/a3713279.55b1c4a8.js index 8f2a2786..9ed92ec2 100644 --- a/assets/js/a3713279.7f20aedd.js +++ b/assets/js/a3713279.55b1c4a8.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9588],{1989(e,s,n){n.r(s),n.d(s,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>o,metadata:()=>r,toc:()=>a});const r=JSON.parse('{"id":"deployment","title":"Deployment","description":"Welcome to Dashy, so glad you\'re here :) Deployment is super easy, and there are several methods available depending on what type of system you\'re using. If you\'re self-hosting, then deploying with Docker (or similar container engine) is the recommended approach.","source":"@site/docs/deployment.md","sourceDirName":".","slug":"/deployment","permalink":"/docs/deployment","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/deployment.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Quick Start","permalink":"/docs/quick-start"},"next":{"title":"Configuring","permalink":"/docs/configuring"}}');var i=n(4848),t=n(8453);const o={},l="Deployment",d={},a=[{value:"Quick Start",id:"quick-start",level:2},{value:"Deployment Methods",id:"deployment-methods",level:2},{value:"Deploy with Docker",id:"deploy-with-docker",level:2},{value:"Using Docker Compose",id:"using-docker-compose",level:2},{value:"Podman",id:"podman",level:2},{value:"Portainer",id:"portainer",level:2},{value:"Kubernetes",id:"kubernetes",level:2},{value:"Unraid",id:"unraid",level:2},{value:"Home Server Platforms",id:"home-server-platforms",level:2},{value:"Synology NAS",id:"synology-nas",level:2},{value:"Build from Source",id:"build-from-source",level:2},{value:"Deploy to Cloud Service",id:"deploy-to-cloud-service",level:2},{value:"Netlify",id:"netlify",level:3},{value:"Vercel",id:"vercel",level:3},{value:"Easypanel",id:"easypanel",level:3},{value:"EdgeOne Pages",id:"edgeone-pages",level:3},{value:"Play-with-Docker",id:"play-with-docker",level:3},{value:"Hosting with CDN",id:"hosting-with-cdn",level:2},{value:"Requirements",id:"requirements",level:2},{value:"System Requirements",id:"system-requirements",level:3},{value:"Docker",id:"docker",level:3},{value:"Bare Metal",id:"bare-metal",level:3},{value:"CDN / Cloud Deploy",id:"cdn--cloud-deploy",level:3},{value:"Browser Support",id:"browser-support",level:3}];function h(e){const s={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(s.header,{children:(0,i.jsx)(s.h1,{id:"deployment",children:"Deployment"})}),"\n",(0,i.jsx)(s.p,{children:"Welcome to Dashy, so glad you're here :) Deployment is super easy, and there are several methods available depending on what type of system you're using. If you're self-hosting, then deploying with Docker (or similar container engine) is the recommended approach."}),"\n",(0,i.jsx)(s.h2,{id:"quick-start",children:"Quick Start"}),"\n",(0,i.jsxs)(s.p,{children:["If you want to skip the fuss, and ",(0,i.jsx)(s.a,{href:"/docs/quick-start",children:"get straight down to it"}),", then you can spin up a new instance of Dashy by running:"]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -p 8080:8080 lissy93/dashy\n"})}),"\n",(0,i.jsxs)(s.p,{children:["See ",(0,i.jsx)(s.a,{href:"/docs/management",children:"Management Docs"})," for info about securing, monitoring, updating, health checks, auto starting, web server configuration, etc"]}),"\n",(0,i.jsxs)(s.p,{children:["Once you've got Dashy up and running, you'll want to configure it with your own content, for this you can reference the ",(0,i.jsx)(s.a,{href:"/docs/configuring",children:"configuring docs"}),"."]}),"\n",(0,i.jsx)(s.h2,{id:"deployment-methods",children:"Deployment Methods"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#",children:"Deployment"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#quick-start",children:"Quick Start"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#deployment-methods",children:"Deployment Methods"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#deploy-with-docker",children:"Deploy with Docker"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#using-docker-compose",children:"Using Docker Compose"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#podman",children:"Podman"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#portainer",children:"Portainer"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#kubernetes",children:"Kubernetes"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#unraid",children:"Unraid"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#home-server-platforms",children:"Home Server Platforms"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#synology-nas",children:"Synology NAS"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#build-from-source",children:"Build from Source"})}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#deploy-to-cloud-service",children:"Deploy to Cloud Service"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#netlify",children:"Netlify"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#vercel",children:"Vercel"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#easypanel",children:"Easypanel"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#edgeone-pages",children:"EdgeOne Pages"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#play-with-docker",children:"Play-with-Docker"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#hosting-with-cdn",children:"Hosting with CDN"})}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#requirements",children:"Requirements"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#system-requirements",children:"System Requirements"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#docker",children:"Docker"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#bare-metal",children:"Bare Metal"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#cdn--cloud-deploy",children:"CDN / Cloud Deploy"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#browser-support",children:"Browser Support"})}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"deploy-with-docker",children:"Deploy with Docker"}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.strong,{children:"Container Info"}),": ",(0,i.jsxs)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:["\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Architectures-amd64%20%7C%20arm32v7%20%7C%20arm64v8-6ba6e5",alt:"Docker Supported Architecture"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Base_Image-Alpine_3.19-6ba6e5",alt:"Docker Base Image"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Hosted_on-DockerHub%20%26%20GHCR-6ba6e5",alt:"Docker Hosted on"}),"\n"]}),(0,i.jsx)(s.br,{}),"\n",(0,i.jsx)(s.strong,{children:"Status"}),":\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/github/actions/workflow/status/Lissy93/dashy/docker-build-publish.yml?label=Build&color=f4a966",alt:"Build Status"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/pulls/lissy93/dashy?color=ecb2f7",alt:"Docker Pulls"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/stars/lissy93/dashy?color=f7f754&label=Docker%20Stars",alt:"Docker Stars"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/image-size/lissy93/dashy/latest?color=1eea76",alt:"Docker Image Size"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/v/lissy93/dashy/latest?color=a8d8ea&label=Latest%20Version",alt:"Docker Latest Version"})]}),"\n",(0,i.jsxs)(s.p,{children:["Dashy has a built container image hosted on ",(0,i.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:"Docker Hub"}),". You will need ",(0,i.jsx)(s.a,{href:"https://docs.docker.com/get-docker/",children:"Docker"})," installed on your system."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v /path/to/your/user-data:/app/user-data \\\n --name my-dashboard \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:"user-data"})," directory you mount must contain a ",(0,i.jsx)(s.code,{children:"conf.yml"})," file. It can also contain any sub-config files, item icons, fonts, custom CSS, or other assets you want served from the web root. Anything you put in there is available at ",(0,i.jsx)(s.code,{children:"/"})," in the browser."]}),"\n",(0,i.jsx)(s.p,{children:"Explanation of the above options:"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-d"})," Detached mode (not running in the foreground of your terminal)"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-p"})," The port that should be exposed, and the port it should be mapped to in your host system ",(0,i.jsx)(s.code,{children:"[host-port]:[container-port]"}),", leave the container port as ",(0,i.jsx)(s.code,{children:"8080"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-v"})," Mounts the host directory containing your ",(0,i.jsx)(s.code,{children:"conf.yml"})," (and any other assets) into the container at ",(0,i.jsx)(s.code,{children:"/app/user-data"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"--name"})," Give your container a human-readable name"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"--restart=always"})," Spin up the container when the daemon starts, or after it has been stopped"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," The image to run. Replace ",(0,i.jsx)(s.code,{children:":latest"})," with a specific version from the ",(0,i.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy/tags",children:"tags"})," if needed"]}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["For all available options, and to learn more, see the ",(0,i.jsx)(s.a,{href:"https://docs.docker.com/engine/reference/commandline/run/",children:"Docker Run Docs"})]}),"\n",(0,i.jsxs)(s.p,{children:["Dashy is also available through GHCR: ",(0,i.jsx)(s.code,{children:"docker pull ghcr.io/lissy93/dashy:latest"})]}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:"latest"})," image is multi-arch, so the same tag works on amd64, arm64, and arm/v7 (Raspberry Pi 2+). Docker selects the right variant for your host automatically."]}),"\n",(0,i.jsxs)(s.p,{children:["The image defaults to ",(0,i.jsx)(s.code,{children:":latest"}),", but you can instead specify a specific version, e.g. ",(0,i.jsx)(s.code,{children:"docker pull lissy93/dashy:4.0.0"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"using-docker-compose",children:"Using Docker Compose"}),"\n",(0,i.jsxs)(s.p,{children:["Using Docker Compose can be useful for saving your specific config in files, without having to type out a long run command each time. Save compose config as a YAML file, and then run ",(0,i.jsx)(s.code,{children:"docker compose up -d"})," (optionally use the ",(0,i.jsx)(s.code,{children:"-f"})," flag to specify file location, if it isn't located at ",(0,i.jsx)(s.code,{children:"./docker-compose.yml"}),"), ",(0,i.jsx)(s.code,{children:"-d"})," is detached mode (not running in the foreground of your terminal). Compose is also useful if you are using clusters, as the format is very similar to stack files, used with Docker Swarm."]}),"\n",(0,i.jsxs)(s.p,{children:["The following is a complete example of a ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:(0,i.jsx)(s.code,{children:"docker-compose.yml"})})," for Dashy. Run it as is, or uncomment the additional options you need."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-yaml",children:"services:\n dashy:\n # The image to pull + version. Can use `ghcr.io/lissy93/dashy` instead\n image: lissy93/dashy:latest\n # Optional container name\n container_name: dashy\n # Port to serve on (keep container port (second one) as 8080)\n ports:\n - 8080:8080\n # Mount a directory containing your conf.yml and any other assets\n volumes:\n - ./user-data:/app/user-data\n # Add any env vars for server here, if needed\n environment:\n - NODE_ENV=production\n # Auto-start the container on boot\n restart: unless-stopped\n # Healthcheck to determine when container healthy\n healthcheck:\n test: ['CMD', 'node', '/app/services/healthcheck.js']\n interval: 1m30s\n timeout: 10s\n retries: 3\n start_period: 30s\n"})}),"\n",(0,i.jsxs)(s.p,{children:["To pull from GHCR instead of Docker Hub, set ",(0,i.jsx)(s.code,{children:"image: ghcr.io/lissy93/dashy:latest"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"podman",children:"Podman"}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://podman.io/",children:"Podman"})," is a drop-in replacement for Docker that runs containers without a daemon and doesn't require root. If you're on Fedora, RHEL, or just prefer daemonless containers, Podman works with the same images and mostly the same CLI."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"podman run -d \\\n -p 8080:8080 \\\n -v /path/to/your/user-data:/app/user-data:Z \\\n --name dashy \\\n --restart=always \\\n docker.io/lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:":Z"})," suffix on the volume mount handles SELinux relabeling, which you'll need on Fedora/RHEL. If you're not using SELinux, you can leave it off."]}),"\n",(0,i.jsxs)(s.p,{children:["Podman also supports ",(0,i.jsx)(s.code,{children:"podman-compose"})," or ",(0,i.jsx)(s.code,{children:"podman compose"})," (with the compose plugin) using the same ",(0,i.jsx)(s.code,{children:"docker-compose.yml"})," file shown above."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"portainer",children:"Portainer"}),"\n",(0,i.jsxs)(s.p,{children:["If you manage your Docker host through ",(0,i.jsx)(s.a,{href:"https://www.portainer.io/",children:"Portainer"}),", you can deploy Dashy from its UI:"]}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsx)(s.li,{children:"Go to Stacks > Add stack"}),"\n",(0,i.jsxs)(s.li,{children:["Paste the ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:"docker-compose.yml"})," contents, or point to the URL"]}),"\n",(0,i.jsx)(s.li,{children:"Adjust the port and volume mappings as needed"}),"\n",(0,i.jsx)(s.li,{children:"Deploy the stack"}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["Alternatively, go to Containers > Add container and use the image ",(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," with port ",(0,i.jsx)(s.code,{children:"8080"})," mapped."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"kubernetes",children:"Kubernetes"}),"\n",(0,i.jsxs)(s.p,{children:["@vyrtualsynthese has written a Helm Chart for deploying with Kubernetes, available ",(0,i.jsx)(s.a,{href:"https://github.com/vyrtualsynthese/selfhosted-helmcharts/tree/main/charts/dashy",children:"here"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"unraid",children:"Unraid"}),"\n",(0,i.jsxs)(s.p,{children:["Dashy is available through the ",(0,i.jsx)(s.a,{href:"https://forums.unraid.net/topic/38582-plug-in-community-applications/",children:"Community Applications"}),' plugin. Search for "Dashy" in the Apps tab and install from there. The template pre-fills the Docker image, port mapping, and volume paths for you.']}),"\n",(0,i.jsxs)(s.p,{children:["If you'd prefer to set it up manually, go to Docker > Add Container and use ",(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," as the repository. Map port ",(0,i.jsx)(s.code,{children:"8080"}),", and add a path mapping for the host directory containing your ",(0,i.jsx)(s.code,{children:"conf.yml"})," to ",(0,i.jsx)(s.code,{children:"/app/user-data"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"home-server-platforms",children:"Home Server Platforms"}),"\n",(0,i.jsx)(s.p,{children:"Several self-hosting platforms include Dashy in their app stores, giving you a one-click install with a management UI:"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://casaos.io/",children:"CasaOS"})," - Has Dashy in its built-in app store"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://cosmos-cloud.io/",children:"Cosmos Cloud"})," - Install Dashy from the marketplace"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://umbrel.com/",children:"Umbrel"})," - Available in the Umbrel App Store"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://runtipi.io/",children:"Runtipi"})," - Available in the Runtipi App Store"]}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["These all run Dashy as a Docker container under the hood, so configuration works the same way. You'll find your ",(0,i.jsx)(s.code,{children:"conf.yml"})," in whichever directory the platform maps to ",(0,i.jsx)(s.code,{children:"/app/user-data/"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"synology-nas",children:"Synology NAS"}),"\n",(0,i.jsx)(s.p,{children:"Installing dashy is really simply and fast:"}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Install Docker via Synology ",(0,i.jsx)(s.code,{children:"Package Center"}),"."]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Go to ",(0,i.jsx)(s.code,{children:"File Station"})," and open the ",(0,i.jsx)(s.code,{children:"docker"})," folder. Inside the docker folder, create one new folder and name it ",(0,i.jsx)(s.code,{children:"dashy"}),"."]}),"\n",(0,i.jsxs)(s.blockquote,{children:["\n",(0,i.jsx)(s.p,{children:"Note: Be careful to enter only lowercase, not uppercase letters."}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:"Go to Control Panel / Task Scheduler / Create / Scheduled Task / User-defined script."}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Once you click on ",(0,i.jsx)(s.code,{children:"User-defined"})," script a new window will open."]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:"Follow the instructions below:"}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'General: In the Task field type in Install dashy. Uncheck "Enabled" option. Select root User.'}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'Schedule: Select Run on the following date then select "Do not repeat".'}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'Task Settings: Check "Send run details by email", add your email then copy paste the code below in the Run command area. After that click OK.'}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 4000:8080 \\\n -v /volume1/docker/dashy:/app/user-data \\\n --name dashy \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["(Place your ",(0,i.jsx)(s.code,{children:"conf.yml"})," and any sub-configs / icons / assets inside ",(0,i.jsx)(s.code,{children:"/volume1/docker/dashy"})," on the host.)"]}),"\n",(0,i.jsx)(s.p,{children:"dashy should be up within 1-2min after you've started the install task procedure"}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"build-from-source",children:"Build from Source"}),"\n",(0,i.jsxs)(s.p,{children:["If you do not want to use Docker, you can run Dashy directly on your host system. For this, you will need both ",(0,i.jsx)(s.a,{href:"https://git-scm.com/downloads",children:"git"})," and the latest or LTS version of ",(0,i.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"})," installed, and optionally ",(0,i.jsx)(s.a,{href:"https://yarnpkg.com/",children:"yarn"})]}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsxs)(s.li,{children:["Get Code: ",(0,i.jsx)(s.code,{children:"git clone https://github.com/Lissy93/dashy.git"})," and ",(0,i.jsx)(s.code,{children:"cd dashy"})]}),"\n",(0,i.jsxs)(s.li,{children:["Configuration: Fill in your settings in ",(0,i.jsx)(s.code,{children:"./user-data/conf.yml"})]}),"\n",(0,i.jsxs)(s.li,{children:["Install dependencies: ",(0,i.jsx)(s.code,{children:"yarn"})]}),"\n",(0,i.jsxs)(s.li,{children:["Build: ",(0,i.jsx)(s.code,{children:"yarn build"})]}),"\n",(0,i.jsxs)(s.li,{children:["Run: ",(0,i.jsx)(s.code,{children:"yarn start"})]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"deploy-to-cloud-service",children:"Deploy to Cloud Service"}),"\n",(0,i.jsx)(s.p,{children:"Dashy can be deployed to most cloud providers. The Docker guides above work on any VPS, but these providers offer quicker setup for static or containerized deployments."}),"\n",(0,i.jsx)(s.admonition,{type:"note",children:(0,i.jsx)(s.p,{children:"Static hosting providers (Netlify, Vercel, EdgeOne) won't have status checks or config writing to disk, since those features need Dashy's Node server. Everything else works fine."})}),"\n",(0,i.jsx)(s.h3,{id:"netlify",children:"Netlify"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Netlify-00C7B7?logo=netlify&logoColor=white",alt:"Deploy to Netlify"})})}),"\n",(0,i.jsxs)(s.p,{children:["Dashy includes a ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/netlify.toml",children:(0,i.jsx)(s.code,{children:"netlify.toml"})})," so deployment works out of the box. ",(0,i.jsx)(s.a,{href:"https://www.netlify.com/",children:"Netlify"})," is free for personal use, supports custom domains, and deploys automatically from your Git repo."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"vercel",children:"Vercel"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Vercel-000000?logo=vercel&logoColor=white",alt:"Deploy with Vercel"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://vercel.com/",children:"Vercel"})," hosts static frontends with a generous free tier, custom domains, and built-in analytics."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"easypanel",children:"Easypanel"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://easypanel.io/docs/templates/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Easypanel-5765F2?logo=data:image/svg+xml;base64,&logoColor=white",alt:"Deploy to Easypanel"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://easypanel.io",children:"Easypanel"})," is a self-hosted server control panel with a Dashy template. It runs the full Docker image, so all features including the Node server work."]}),"\n",(0,i.jsxs)(s.p,{children:["Template: ",(0,i.jsx)(s.code,{children:"https://easypanel.io/docs/templates/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"edgeone-pages",children:"EdgeOne Pages"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://edgeone.ai/pages/new?repository-url=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://cdnstatic.tencentcs.com/edgeone/pages/deploy.svg",alt:"Deploy to EdgeOne"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://pages.edgeone.ai",children:"EdgeOne Pages"})," is Tencent's edge hosting platform. Static deploy from your Git repo."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://edgeone.ai/pages/new?repository-url=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"play-with-docker",children:"Play-with-Docker"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Try-Play_with_Docker-0db7ed?logo=docker&logoColor=white",alt:"Try in PWD"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://labs.play-with-docker.com/",children:"Play with Docker"})," gives you a free, temporary Docker environment in the browser. Good for trying Dashy without installing anything. Sessions last 4 hours."]}),"\n",(0,i.jsxs)(s.p,{children:["URL: ",(0,i.jsx)(s.code,{children:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"hosting-with-cdn",children:"Hosting with CDN"}),"\n",(0,i.jsxs)(s.p,{children:["Once Dashy has been built, it is effectively just a static web app. This means that it can be served up with pretty much any static host, CDN or web server. To host Dashy through a CDN, the steps are very similar to building from source: clone the project, cd into it, install dependencies, write your config file and build the app. Once build is complete you will have a ",(0,i.jsx)(s.code,{children:"./dist"})," directory within Dashy's root, and this is the build application which is ready to be served up."]}),"\n",(0,i.jsx)(s.p,{children:"However without Dashy's node server, there are a couple of features that will be unavailable to you, including: writing config changes to disk through the UI, and application status checks. Everything else will work fine."}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"requirements",children:"Requirements"}),"\n",(0,i.jsx)(s.h3,{id:"system-requirements",children:"System Requirements"}),"\n",(0,i.jsx)(s.p,{children:"Dashy works well on a Raspberry Pi (tested on Pi 3 and later), but should also run well on any system."}),"\n",(0,i.jsx)(s.h3,{id:"docker",children:"Docker"}),"\n",(0,i.jsx)(s.p,{children:"The initial build causes a spike in resource usage, but once running it's fairly steady. Minimum 1GB memory and 1GB disk space."}),"\n",(0,i.jsx)(s.h3,{id:"bare-metal",children:"Bare Metal"}),"\n",(0,i.jsxs)(s.p,{children:["Requires ",(0,i.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"})," and ",(0,i.jsx)(s.a,{href:"https://yarnpkg.com/",children:"Yarn"}),". The ",(0,i.jsx)(s.code,{children:"engines"})," field in ",(0,i.jsx)(s.code,{children:"package.json"})," specifies ",(0,i.jsx)(s.code,{children:">=18.0.0"}),", but the Docker image is built and tested on Node 24, so that's the recommended version for bare-metal too."]}),"\n",(0,i.jsx)(s.p,{children:"Minimum 512MB memory, 2GB disk space."}),"\n",(0,i.jsx)(s.h3,{id:"cdn--cloud-deploy",children:"CDN / Cloud Deploy"}),"\n",(0,i.jsx)(s.p,{children:"No specific requirements. The built app (without the Node server) is very lightweight and can be served by any static host or CDN. If you're using custom icons or other assets, additional disk space will be needed."}),"\n",(0,i.jsx)(s.h3,{id:"browser-support",children:"Browser Support"}),"\n",(0,i.jsxs)(s.p,{children:["JavaScript is required. Dashy targets browsers with >1% global usage and the last 2 versions of each (via ",(0,i.jsx)(s.a,{href:"https://browsersl.ist/",children:"browserslist"}),"). In practice, any modern browser works fine. Internet Explorer is not supported."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,i.jsxs)(s.table,{children:[(0,i.jsx)(s.thead,{children:(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.th,{children:"Browser"}),(0,i.jsx)(s.th,{children:"Minimum Version"}),(0,i.jsx)(s.th,{children:"Status"})]})}),(0,i.jsxs)(s.tbody,{children:[(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Chrome / Chromium"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Firefox"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Edge"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Safari"}),(0,i.jsx)(s.td,{children:"14+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Opera"}),(0,i.jsx)(s.td,{children:"76+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Samsung Internet"}),(0,i.jsx)(s.td,{children:"15+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Firefox ESR"}),(0,i.jsx)(s.td,{children:"Latest"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Internet Explorer"}),(0,i.jsx)(s.td,{children:"-"}),(0,i.jsx)(s.td,{children:"Not supported"})]})]})]})]})}function c(e={}){const{wrapper:s}={...(0,t.R)(),...e.components};return s?(0,i.jsx)(s,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},8453(e,s,n){n.d(s,{R:()=>o,x:()=>l});var r=n(6540);const i={},t=r.createContext(i);function o(e){const s=r.useContext(t);return r.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function l(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),r.createElement(t.Provider,{value:s},e.children)}}}]); \ No newline at end of file +"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[9588],{1989(e,s,n){n.r(s),n.d(s,{assets:()=>d,contentTitle:()=>l,default:()=>c,frontMatter:()=>o,metadata:()=>r,toc:()=>a});const r=JSON.parse('{"id":"deployment","title":"Deployment","description":"Welcome to Dashy, so glad you\'re here :) Deployment is super easy, and there are several methods available depending on what type of system you\'re using. If you\'re self-hosting, then deploying with Docker (or similar container engine) is the recommended approach.","source":"@site/docs/deployment.md","sourceDirName":".","slug":"/deployment","permalink":"/docs/deployment","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/deployment.md","tags":[],"version":"current","lastUpdatedAt":1779128164000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Quick Start","permalink":"/docs/quick-start"},"next":{"title":"Configuring","permalink":"/docs/configuring"}}');var i=n(4848),t=n(8453);const o={},l="Deployment",d={},a=[{value:"Quick Start",id:"quick-start",level:2},{value:"Deployment Methods",id:"deployment-methods",level:2},{value:"Deploy with Docker",id:"deploy-with-docker",level:2},{value:"Using Docker Compose",id:"using-docker-compose",level:2},{value:"Podman",id:"podman",level:2},{value:"Portainer",id:"portainer",level:2},{value:"Kubernetes",id:"kubernetes",level:2},{value:"Unraid",id:"unraid",level:2},{value:"Home Server Platforms",id:"home-server-platforms",level:2},{value:"Synology NAS",id:"synology-nas",level:2},{value:"Build from Source",id:"build-from-source",level:2},{value:"Deploy to Cloud Service",id:"deploy-to-cloud-service",level:2},{value:"Netlify",id:"netlify",level:3},{value:"Vercel",id:"vercel",level:3},{value:"Easypanel",id:"easypanel",level:3},{value:"EdgeOne Pages",id:"edgeone-pages",level:3},{value:"Play-with-Docker",id:"play-with-docker",level:3},{value:"Hosting with CDN",id:"hosting-with-cdn",level:2},{value:"Requirements",id:"requirements",level:2},{value:"System Requirements",id:"system-requirements",level:3},{value:"Docker",id:"docker",level:3},{value:"Bare Metal",id:"bare-metal",level:3},{value:"CDN / Cloud Deploy",id:"cdn--cloud-deploy",level:3},{value:"Browser Support",id:"browser-support",level:3}];function h(e){const s={a:"a",admonition:"admonition",blockquote:"blockquote",br:"br",code:"code",h1:"h1",h2:"h2",h3:"h3",header:"header",hr:"hr",img:"img",li:"li",ol:"ol",p:"p",pre:"pre",strong:"strong",table:"table",tbody:"tbody",td:"td",th:"th",thead:"thead",tr:"tr",ul:"ul",...(0,t.R)(),...e.components};return(0,i.jsxs)(i.Fragment,{children:[(0,i.jsx)(s.header,{children:(0,i.jsx)(s.h1,{id:"deployment",children:"Deployment"})}),"\n",(0,i.jsx)(s.p,{children:"Welcome to Dashy, so glad you're here :) Deployment is super easy, and there are several methods available depending on what type of system you're using. If you're self-hosting, then deploying with Docker (or similar container engine) is the recommended approach."}),"\n",(0,i.jsx)(s.h2,{id:"quick-start",children:"Quick Start"}),"\n",(0,i.jsxs)(s.p,{children:["If you want to skip the fuss, and ",(0,i.jsx)(s.a,{href:"/docs/quick-start",children:"get straight down to it"}),", then you can spin up a new instance of Dashy by running:"]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -p 8080:8080 lissy93/dashy\n"})}),"\n",(0,i.jsxs)(s.p,{children:["See ",(0,i.jsx)(s.a,{href:"/docs/management",children:"Management Docs"})," for info about securing, monitoring, updating, health checks, auto starting, web server configuration, etc"]}),"\n",(0,i.jsxs)(s.p,{children:["Once you've got Dashy up and running, you'll want to configure it with your own content, for this you can reference the ",(0,i.jsx)(s.a,{href:"/docs/configuring",children:"configuring docs"}),"."]}),"\n",(0,i.jsx)(s.h2,{id:"deployment-methods",children:"Deployment Methods"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#",children:"Deployment"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#quick-start",children:"Quick Start"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#deployment-methods",children:"Deployment Methods"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#deploy-with-docker",children:"Deploy with Docker"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#using-docker-compose",children:"Using Docker Compose"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#podman",children:"Podman"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#portainer",children:"Portainer"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#kubernetes",children:"Kubernetes"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#unraid",children:"Unraid"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#home-server-platforms",children:"Home Server Platforms"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#synology-nas",children:"Synology NAS"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#build-from-source",children:"Build from Source"})}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#deploy-to-cloud-service",children:"Deploy to Cloud Service"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#netlify",children:"Netlify"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#vercel",children:"Vercel"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#easypanel",children:"Easypanel"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#edgeone-pages",children:"EdgeOne Pages"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#play-with-docker",children:"Play-with-Docker"})}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#hosting-with-cdn",children:"Hosting with CDN"})}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"#requirements",children:"Requirements"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#system-requirements",children:"System Requirements"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#docker",children:"Docker"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#bare-metal",children:"Bare Metal"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#cdn--cloud-deploy",children:"CDN / Cloud Deploy"})}),"\n",(0,i.jsx)(s.li,{children:(0,i.jsx)(s.a,{href:"#browser-support",children:"Browser Support"})}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"deploy-with-docker",children:"Deploy with Docker"}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.strong,{children:"Container Info"}),": ",(0,i.jsxs)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:["\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Architectures-amd64%20%7C%20arm32v7%20%7C%20arm64v8-6ba6e5",alt:"Docker Supported Architecture"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Base_Image-Alpine_3.19-6ba6e5",alt:"Docker Base Image"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Hosted_on-DockerHub%20%26%20GHCR-6ba6e5",alt:"Docker Hosted on"}),"\n"]}),(0,i.jsx)(s.br,{}),"\n",(0,i.jsx)(s.strong,{children:"Status"}),":\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/github/actions/workflow/status/Lissy93/dashy/docker-build-publish.yml?label=Build&color=f4a966",alt:"Build Status"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/pulls/lissy93/dashy?color=ecb2f7",alt:"Docker Pulls"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/stars/lissy93/dashy?color=f7f754&label=Docker%20Stars",alt:"Docker Stars"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/image-size/lissy93/dashy/latest?color=1eea76",alt:"Docker Image Size"}),"\n",(0,i.jsx)(s.img,{src:"https://img.shields.io/docker/v/lissy93/dashy/latest?color=a8d8ea&label=Latest%20Version",alt:"Docker Latest Version"})]}),"\n",(0,i.jsxs)(s.p,{children:["Dashy has a built container image hosted on ",(0,i.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy",children:"Docker Hub"}),". You will need ",(0,i.jsx)(s.a,{href:"https://docs.docker.com/get-docker/",children:"Docker"})," installed on your system."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 8080:8080 \\\n -v /path/to/your/user-data:/app/user-data \\\n --name my-dashboard \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:"user-data"})," directory you mount must contain a ",(0,i.jsx)(s.code,{children:"conf.yml"})," file. It can also contain any sub-config files, item icons, fonts, custom CSS, or other assets you want served from the web root. Anything you put in there is available at ",(0,i.jsx)(s.code,{children:"/"})," in the browser."]}),"\n",(0,i.jsx)(s.p,{children:"Explanation of the above options:"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-d"})," Detached mode (not running in the foreground of your terminal)"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-p"})," The port that should be exposed, and the port it should be mapped to in your host system ",(0,i.jsx)(s.code,{children:"[host-port]:[container-port]"}),", leave the container port as ",(0,i.jsx)(s.code,{children:"8080"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"-v"})," Mounts the host directory containing your ",(0,i.jsx)(s.code,{children:"conf.yml"})," (and any other assets) into the container at ",(0,i.jsx)(s.code,{children:"/app/user-data"})]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"--name"})," Give your container a human-readable name"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"--restart=always"})," Spin up the container when the daemon starts, or after it has been stopped"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," The image to run. Replace ",(0,i.jsx)(s.code,{children:":latest"})," with a specific version from the ",(0,i.jsx)(s.a,{href:"https://hub.docker.com/r/lissy93/dashy/tags",children:"tags"})," if needed"]}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["For all available options, and to learn more, see the ",(0,i.jsx)(s.a,{href:"https://docs.docker.com/engine/reference/commandline/run/",children:"Docker Run Docs"})]}),"\n",(0,i.jsxs)(s.p,{children:["Dashy is also available through GHCR: ",(0,i.jsx)(s.code,{children:"docker pull ghcr.io/lissy93/dashy:latest"})]}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:"latest"})," image is multi-arch, so the same tag works on amd64, arm64, and arm/v7 (Raspberry Pi 2+). Docker selects the right variant for your host automatically."]}),"\n",(0,i.jsxs)(s.p,{children:["The image defaults to ",(0,i.jsx)(s.code,{children:":latest"}),", but you can instead specify a specific version, e.g. ",(0,i.jsx)(s.code,{children:"docker pull lissy93/dashy:4.0.0"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"using-docker-compose",children:"Using Docker Compose"}),"\n",(0,i.jsxs)(s.p,{children:["Using Docker Compose can be useful for saving your specific config in files, without having to type out a long run command each time. Save compose config as a YAML file, and then run ",(0,i.jsx)(s.code,{children:"docker compose up -d"})," (optionally use the ",(0,i.jsx)(s.code,{children:"-f"})," flag to specify file location, if it isn't located at ",(0,i.jsx)(s.code,{children:"./docker-compose.yml"}),"), ",(0,i.jsx)(s.code,{children:"-d"})," is detached mode (not running in the foreground of your terminal). Compose is also useful if you are using clusters, as the format is very similar to stack files, used with Docker Swarm."]}),"\n",(0,i.jsxs)(s.p,{children:["The following is a complete example of a ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:(0,i.jsx)(s.code,{children:"docker-compose.yml"})})," for Dashy. Run it as is, or uncomment the additional options you need."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-yaml",children:"services:\n dashy:\n # The image to pull + version. Can use `ghcr.io/lissy93/dashy` instead\n image: lissy93/dashy:latest\n # Optional container name\n container_name: dashy\n # Port to serve on (keep container port (second one) as 8080)\n ports:\n - 8080:8080\n # Mount a directory containing your conf.yml and any other assets\n volumes:\n - ./user-data:/app/user-data\n # Add any env vars for server here, if needed\n environment:\n - NODE_ENV=production\n # Auto-start the container on boot\n restart: unless-stopped\n # Healthcheck to determine when container healthy\n healthcheck:\n test: ['CMD', 'node', '/app/services/healthcheck.js']\n interval: 1m30s\n timeout: 10s\n retries: 3\n start_period: 30s\n"})}),"\n",(0,i.jsxs)(s.p,{children:["To pull from GHCR instead of Docker Hub, set ",(0,i.jsx)(s.code,{children:"image: ghcr.io/lissy93/dashy:latest"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"podman",children:"Podman"}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://podman.io/",children:"Podman"})," is a drop-in replacement for Docker that runs containers without a daemon and doesn't require root. If you're on Fedora, RHEL, or just prefer daemonless containers, Podman works with the same images and mostly the same CLI."]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"podman run -d \\\n -p 8080:8080 \\\n -v /path/to/your/user-data:/app/user-data:Z \\\n --name dashy \\\n --restart=always \\\n docker.io/lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["The ",(0,i.jsx)(s.code,{children:":Z"})," suffix on the volume mount handles SELinux relabeling, which you'll need on Fedora/RHEL. If you're not using SELinux, you can leave it off."]}),"\n",(0,i.jsxs)(s.p,{children:["Podman also supports ",(0,i.jsx)(s.code,{children:"podman-compose"})," or ",(0,i.jsx)(s.code,{children:"podman compose"})," (with the compose plugin) using the same ",(0,i.jsx)(s.code,{children:"docker-compose.yml"})," file shown above."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"portainer",children:"Portainer"}),"\n",(0,i.jsxs)(s.p,{children:["If you manage your Docker host through ",(0,i.jsx)(s.a,{href:"https://www.portainer.io/",children:"Portainer"}),", you can deploy Dashy from its UI:"]}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsx)(s.li,{children:"Go to Stacks > Add stack"}),"\n",(0,i.jsxs)(s.li,{children:["Paste the ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/docker-compose.yml",children:"docker-compose.yml"})," contents, or point to the URL"]}),"\n",(0,i.jsx)(s.li,{children:"Adjust the port and volume mappings as needed"}),"\n",(0,i.jsx)(s.li,{children:"Deploy the stack"}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["Alternatively, go to Containers > Add container and use the image ",(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," with port ",(0,i.jsx)(s.code,{children:"8080"})," mapped."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"kubernetes",children:"Kubernetes"}),"\n",(0,i.jsxs)(s.p,{children:["@vyrtualsynthese has written a Helm Chart for deploying with Kubernetes, available ",(0,i.jsx)(s.a,{href:"https://github.com/vyrtualsynthese/selfhosted-helmcharts/tree/main/charts/dashy",children:"here"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"unraid",children:"Unraid"}),"\n",(0,i.jsxs)(s.p,{children:["Dashy is available through the ",(0,i.jsx)(s.a,{href:"https://forums.unraid.net/topic/38582-plug-in-community-applications/",children:"Community Applications"}),' plugin. Search for "Dashy" in the Apps tab and install from there. The template pre-fills the Docker image, port mapping, and volume paths for you.']}),"\n",(0,i.jsxs)(s.p,{children:["If you'd prefer to set it up manually, go to Docker > Add Container and use ",(0,i.jsx)(s.code,{children:"lissy93/dashy:latest"})," as the repository. Map port ",(0,i.jsx)(s.code,{children:"8080"}),", and add a path mapping for the host directory containing your ",(0,i.jsx)(s.code,{children:"conf.yml"})," to ",(0,i.jsx)(s.code,{children:"/app/user-data"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"home-server-platforms",children:"Home Server Platforms"}),"\n",(0,i.jsx)(s.p,{children:"Several self-hosting platforms include Dashy in their app stores, giving you a one-click install with a management UI:"}),"\n",(0,i.jsxs)(s.ul,{children:["\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://casaos.io/",children:"CasaOS"})," - Has Dashy in its built-in app store"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://cosmos-cloud.io/",children:"Cosmos Cloud"})," - Install Dashy from the marketplace"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://umbrel.com/",children:"Umbrel"})," - Available in the Umbrel App Store"]}),"\n",(0,i.jsxs)(s.li,{children:[(0,i.jsx)(s.a,{href:"https://runtipi.io/",children:"Runtipi"})," - Available in the Runtipi App Store"]}),"\n"]}),"\n",(0,i.jsxs)(s.p,{children:["These all run Dashy as a Docker container under the hood, so configuration works the same way. You'll find your ",(0,i.jsx)(s.code,{children:"conf.yml"})," in whichever directory the platform maps to ",(0,i.jsx)(s.code,{children:"/app/user-data/"}),"."]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"synology-nas",children:"Synology NAS"}),"\n",(0,i.jsx)(s.p,{children:"Installing dashy is really simply and fast:"}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Install Docker via Synology ",(0,i.jsx)(s.code,{children:"Package Center"}),"."]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Go to ",(0,i.jsx)(s.code,{children:"File Station"})," and open the ",(0,i.jsx)(s.code,{children:"docker"})," folder. Inside the docker folder, create one new folder and name it ",(0,i.jsx)(s.code,{children:"dashy"}),"."]}),"\n",(0,i.jsxs)(s.blockquote,{children:["\n",(0,i.jsx)(s.p,{children:"Note: Be careful to enter only lowercase, not uppercase letters."}),"\n"]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:"Go to Control Panel / Task Scheduler / Create / Scheduled Task / User-defined script."}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsxs)(s.p,{children:["Once you click on ",(0,i.jsx)(s.code,{children:"User-defined"})," script a new window will open."]}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:"Follow the instructions below:"}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'General: In the Task field type in Install dashy. Uncheck "Enabled" option. Select root User.'}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'Schedule: Select Run on the following date then select "Do not repeat".'}),"\n"]}),"\n",(0,i.jsxs)(s.li,{children:["\n",(0,i.jsx)(s.p,{children:'Task Settings: Check "Send run details by email", add your email then copy paste the code below in the Run command area. After that click OK.'}),"\n"]}),"\n"]}),"\n",(0,i.jsx)(s.pre,{children:(0,i.jsx)(s.code,{className:"language-bash",children:"docker run -d \\\n -p 4000:8080 \\\n -v /volume1/docker/dashy:/app/user-data \\\n --name dashy \\\n --restart=always \\\n lissy93/dashy:latest\n"})}),"\n",(0,i.jsxs)(s.p,{children:["(Place your ",(0,i.jsx)(s.code,{children:"conf.yml"})," and any sub-configs / icons / assets inside ",(0,i.jsx)(s.code,{children:"/volume1/docker/dashy"})," on the host.)"]}),"\n",(0,i.jsx)(s.p,{children:"dashy should be up within 1-2min after you've started the install task procedure"}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"build-from-source",children:"Build from Source"}),"\n",(0,i.jsxs)(s.p,{children:["If you do not want to use Docker, you can run Dashy directly on your host system. For this, you will need both ",(0,i.jsx)(s.a,{href:"https://git-scm.com/downloads",children:"git"})," and the latest or LTS version of ",(0,i.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"})," installed, and optionally ",(0,i.jsx)(s.a,{href:"https://yarnpkg.com/",children:"yarn"})]}),"\n",(0,i.jsxs)(s.ol,{children:["\n",(0,i.jsxs)(s.li,{children:["Get Code: ",(0,i.jsx)(s.code,{children:"git clone https://github.com/Lissy93/dashy.git"})," and ",(0,i.jsx)(s.code,{children:"cd dashy"})]}),"\n",(0,i.jsxs)(s.li,{children:["Configuration: Fill in your settings in ",(0,i.jsx)(s.code,{children:"./user-data/conf.yml"})]}),"\n",(0,i.jsxs)(s.li,{children:["Install dependencies: ",(0,i.jsx)(s.code,{children:"yarn"})]}),"\n",(0,i.jsxs)(s.li,{children:["Build: ",(0,i.jsx)(s.code,{children:"yarn build"})]}),"\n",(0,i.jsxs)(s.li,{children:["Run: ",(0,i.jsx)(s.code,{children:"yarn start"})]}),"\n"]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"deploy-to-cloud-service",children:"Deploy to Cloud Service"}),"\n",(0,i.jsx)(s.p,{children:"Dashy can be deployed to most cloud providers. The Docker guides above work on any VPS, but these providers offer quicker setup for static or containerized deployments."}),"\n",(0,i.jsx)(s.admonition,{type:"note",children:(0,i.jsx)(s.p,{children:"Static hosting providers (Netlify, Vercel, EdgeOne) won't have status checks or config writing to disk, since those features need Dashy's Node server. Everything else works fine."})}),"\n",(0,i.jsx)(s.h3,{id:"netlify",children:"Netlify"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Netlify-00C7B7?logo=netlify&logoColor=white",alt:"Deploy to Netlify"})})}),"\n",(0,i.jsxs)(s.p,{children:["Dashy includes a ",(0,i.jsx)(s.a,{href:"https://github.com/Lissy93/dashy/blob/master/netlify.toml",children:(0,i.jsx)(s.code,{children:"netlify.toml"})})," so deployment works out of the box. ",(0,i.jsx)(s.a,{href:"https://www.netlify.com/",children:"Netlify"})," is free for personal use, supports custom domains, and deploys automatically from your Git repo."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"vercel",children:"Vercel"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Vercel-000000?logo=vercel&logoColor=white",alt:"Deploy with Vercel"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://vercel.com/",children:"Vercel"})," hosts static frontends with a generous free tier, custom domains, and built-in analytics."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"easypanel",children:"Easypanel"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://easypanel.io/docs/templates/dashy",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Deploy-Easypanel-5765F2?logo=data:image/svg+xml;base64,&logoColor=white",alt:"Deploy to Easypanel"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://easypanel.io",children:"Easypanel"})," is a self-hosted server control panel with a Dashy template. It runs the full Docker image, so all features including the Node server work."]}),"\n",(0,i.jsxs)(s.p,{children:["Template: ",(0,i.jsx)(s.code,{children:"https://easypanel.io/docs/templates/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"edgeone-pages",children:"EdgeOne Pages"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://edgeone.ai/pages/new?repository-url=https://github.com/lissy93/dashy",children:(0,i.jsx)(s.img,{src:"https://cdnstatic.tencentcs.com/edgeone/pages/deploy.svg",alt:"Deploy to EdgeOne"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://pages.edgeone.ai",children:"EdgeOne Pages"})," is Tencent's edge hosting platform. Static deploy from your Git repo."]}),"\n",(0,i.jsxs)(s.p,{children:["Deploy link: ",(0,i.jsx)(s.code,{children:"https://edgeone.ai/pages/new?repository-url=https://github.com/lissy93/dashy"})]}),"\n",(0,i.jsx)(s.h3,{id:"play-with-docker",children:"Play-with-Docker"}),"\n",(0,i.jsx)(s.p,{children:(0,i.jsx)(s.a,{href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml",children:(0,i.jsx)(s.img,{src:"https://img.shields.io/badge/Try-Play_with_Docker-0db7ed?logo=docker&logoColor=white",alt:"Try in PWD"})})}),"\n",(0,i.jsxs)(s.p,{children:[(0,i.jsx)(s.a,{href:"https://labs.play-with-docker.com/",children:"Play with Docker"})," gives you a free, temporary Docker environment in the browser. Good for trying Dashy without installing anything. Sessions last 4 hours."]}),"\n",(0,i.jsxs)(s.p,{children:["URL: ",(0,i.jsx)(s.code,{children:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml"})]}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"hosting-with-cdn",children:"Hosting with CDN"}),"\n",(0,i.jsxs)(s.p,{children:["Once Dashy has been built, it is effectively just a static web app. This means that it can be served up with pretty much any static host, CDN or web server. To host Dashy through a CDN, the steps are very similar to building from source: clone the project, cd into it, install dependencies, write your config file and build the app. Once build is complete you will have a ",(0,i.jsx)(s.code,{children:"./dist"})," directory within Dashy's root, and this is the build application which is ready to be served up."]}),"\n",(0,i.jsx)(s.p,{children:"However without Dashy's node server, there are a couple of features that will be unavailable to you, including: writing config changes to disk through the UI, and application status checks. Everything else will work fine."}),"\n",(0,i.jsx)(s.hr,{}),"\n",(0,i.jsx)(s.h2,{id:"requirements",children:"Requirements"}),"\n",(0,i.jsx)(s.h3,{id:"system-requirements",children:"System Requirements"}),"\n",(0,i.jsx)(s.p,{children:"Dashy works well on a Raspberry Pi (tested on Pi 3 and later), but should also run well on any system."}),"\n",(0,i.jsx)(s.h3,{id:"docker",children:"Docker"}),"\n",(0,i.jsx)(s.p,{children:"The initial build causes a spike in resource usage, but once running it's fairly steady. Minimum 1GB memory and 1GB disk space."}),"\n",(0,i.jsx)(s.h3,{id:"bare-metal",children:"Bare Metal"}),"\n",(0,i.jsxs)(s.p,{children:["Requires ",(0,i.jsx)(s.a,{href:"https://nodejs.org/",children:"Node.js"})," and ",(0,i.jsx)(s.a,{href:"https://yarnpkg.com/",children:"Yarn"}),". The ",(0,i.jsx)(s.code,{children:"engines"})," field in ",(0,i.jsx)(s.code,{children:"package.json"})," specifies ",(0,i.jsx)(s.code,{children:">=18.0.0"}),", but the Docker image is built and tested on Node 24, so that's the recommended version for bare-metal too."]}),"\n",(0,i.jsx)(s.p,{children:"Minimum 512MB memory, 2GB disk space."}),"\n",(0,i.jsx)(s.h3,{id:"cdn--cloud-deploy",children:"CDN / Cloud Deploy"}),"\n",(0,i.jsx)(s.p,{children:"No specific requirements. The built app (without the Node server) is very lightweight and can be served by any static host or CDN. If you're using custom icons or other assets, additional disk space will be needed."}),"\n",(0,i.jsx)(s.h3,{id:"browser-support",children:"Browser Support"}),"\n",(0,i.jsxs)(s.p,{children:["JavaScript is required. Dashy targets browsers with >1% global usage and the last 2 versions of each (via ",(0,i.jsx)(s.a,{href:"https://browsersl.ist/",children:"browserslist"}),"). In practice, any modern browser works fine. Internet Explorer is not supported."]}),"\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n",(0,i.jsxs)(s.table,{children:[(0,i.jsx)(s.thead,{children:(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.th,{children:"Browser"}),(0,i.jsx)(s.th,{children:"Minimum Version"}),(0,i.jsx)(s.th,{children:"Status"})]})}),(0,i.jsxs)(s.tbody,{children:[(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Chrome / Chromium"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Firefox"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Edge"}),(0,i.jsx)(s.td,{children:"90+"}),(0,i.jsx)(s.td,{children:"Fully supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Safari"}),(0,i.jsx)(s.td,{children:"14+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Opera"}),(0,i.jsx)(s.td,{children:"76+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Samsung Internet"}),(0,i.jsx)(s.td,{children:"15+"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Firefox ESR"}),(0,i.jsx)(s.td,{children:"Latest"}),(0,i.jsx)(s.td,{children:"Supported"})]}),(0,i.jsxs)(s.tr,{children:[(0,i.jsx)(s.td,{children:"Internet Explorer"}),(0,i.jsx)(s.td,{children:"-"}),(0,i.jsx)(s.td,{children:"Not supported"})]})]})]})]})}function c(e={}){const{wrapper:s}={...(0,t.R)(),...e.components};return s?(0,i.jsx)(s,{...e,children:(0,i.jsx)(h,{...e})}):h(e)}},8453(e,s,n){n.d(s,{R:()=>o,x:()=>l});var r=n(6540);const i={},t=r.createContext(i);function o(e){const s=r.useContext(t);return r.useMemo(function(){return"function"==typeof e?e(s):{...s,...e}},[s,e])}function l(e){let s;return s=e.disableParentContext?"function"==typeof e.components?e.components(i):e.components||i:o(e.components),r.createElement(t.Provider,{value:s},e.children)}}}]); \ No newline at end of file diff --git a/assets/js/a63906ba.2a9847ea.js b/assets/js/a63906ba.71737a3d.js similarity index 99% rename from assets/js/a63906ba.2a9847ea.js rename to assets/js/a63906ba.71737a3d.js index 7d639151..b21baf1b 100644 --- a/assets/js/a63906ba.2a9847ea.js +++ b/assets/js/a63906ba.71737a3d.js @@ -1 +1 @@ -"use strict";(globalThis.webpackChunkdashy=globalThis.webpackChunkdashy||[]).push([[6903],{2737(e,n,s){s.r(n),s.d(n,{assets:()=>d,contentTitle:()=>o,default:()=>h,frontMatter:()=>a,metadata:()=>i,toc:()=>l});const i=JSON.parse('{"id":"development-guides","title":"Development Guides","description":"A series of short tutorials, to guide you through the most common development tasks.","source":"@site/docs/development-guides.md","sourceDirName":".","slug":"/development-guides","permalink":"/docs/development-guides","draft":false,"unlisted":false,"editUrl":"https://github.com/Lissy93/dashy/edit/master/docs/docs/development-guides.md","tags":[],"version":"current","lastUpdatedAt":1779040438000,"frontMatter":{},"sidebar":"dashySidebar","previous":{"title":"Developing","permalink":"/docs/developing"},"next":{"title":"Privacy & Security","permalink":"/docs/privacy"}}');var t=s(4848),r=s(8453);const a={},o="Development Guides",d={},l=[{value:"Creating a new theme",id:"creating-a-new-theme",level:2},{value:"1. Add Theme Name",id:"1-add-theme-name",level:3},{value:"2. Write some Styles",id:"2-write-some-styles",level:3},{value:"Writing Translations",id:"writing-translations",level:2},{value:"1. Create a new Language File",id:"1-create-a-new-language-file",level:3},{value:"2. Translate",id:"2-translate",level:3},{value:"3. Add your file to the app",id:"3-add-your-file-to-the-app",level:3},{value:"Adding a new option in the config file",id:"adding-a-new-option-in-the-config-file",level:2},{value:"Updating Dependencies",id:"updating-dependencies",level:2},{value:"Developing Netlify Cloud Functions",id:"developing-netlify-cloud-functions",level:2},{value:"1. Run Netlify Dev Server",id:"1-run-netlify-dev-server",level:3},{value:"2. Create a lambda function",id:"2-create-a-lambda-function",level:3},{value:"3. Redirect the Node endpoint to the function",id:"3-redirect-the-node-endpoint-to-the-function",level:3},{value:"Hiding Page Furniture on Certain Routes",id:"hiding-page-furniture-on-certain-routes",level:2},{value:"1. Add the route name to the should hide array",id:"1-add-the-route-name-to-the-should-hide-array",level:3},{value:"2. Add the conditional to the structural component to hide",id:"2-add-the-conditional-to-the-structural-component-to-hide",level:3},{value:"Adding / Using Environmental Variables",id:"adding--using-environmental-variables",level:2},{value:"Building a Widget",id:"building-a-widget",level:2},{value:"Step 0 - Prerequisites",id:"step-0---prerequisites",level:3},{value:"Step 1 - Create Widget",id:"step-1---create-widget",level:3},{value:"Step 2 - Adding Functionality",id:"step-2---adding-functionality",level:3},{value:"Accessing User Options",id:"accessing-user-options",level:4},{value:"Adding an API Endpoint",id:"adding-an-api-endpoint",level:4},{value:"Making an API Request",id:"making-an-api-request",level:4},{value:"Processing Response",id:"processing-response",level:4},{value:"Rendering Response",id:"rendering-response",level:4},{value:"Styling",id:"styling",level:4},{value:"Step 3 - Register",id:"step-3---register",level:3},{value:"Step 4 - Docs",id:"step-4---docs",level:3},{value:"Respecting Config Permissions",id:"respecting-config-permissions",level:2}];function c(e){const n={a:"a",code:"code",h1:"h1",h2:"h2",h3:"h3",h4:"h4",header:"header",hr:"hr",input:"input",li:"li",p:"p",pre:"pre",strong:"strong",ul:"ul",...(0,r.R)(),...e.components};return(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)(n.header,{children:(0,t.jsx)(n.h1,{id:"development-guides",children:"Development Guides"})}),"\n",(0,t.jsx)(n.p,{children:"A series of short tutorials, to guide you through the most common development tasks."}),"\n",(0,t.jsx)(n.p,{children:"Sections:"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#creating-a-new-theme",children:"Creating a new theme"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#writing-translations",children:"Writing Translations"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#adding-a-new-option-in-the-config-file",children:"Adding a new option in the config file"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#updating-dependencies",children:"Updating Dependencies"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#developing-netlify-cloud-functions",children:"Writing Netlify Cloud Functions"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#hiding-page-furniture-on-certain-routes",children:"Hiding Page Furniture"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#adding--using-environmental-variables",children:"Adding / Using Environmental Variables"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#building-a-widget",children:"Building a Widget"})}),"\n",(0,t.jsx)(n.li,{children:(0,t.jsx)(n.a,{href:"#respecting-config-permissions",children:"Respecting Config Permissions"})}),"\n"]}),"\n",(0,t.jsx)(n.h2,{id:"creating-a-new-theme",children:"Creating a new theme"}),"\n",(0,t.jsx)(n.p,{children:"Adding a new theme is really easy. There're two things you need to do: Pass the theme name to Dashy, so that it can be added to the theme selector dropdown menu, and then write some styles!"}),"\n",(0,t.jsx)(n.h3,{id:"1-add-theme-name",children:"1. Add Theme Name"}),"\n",(0,t.jsxs)(n.p,{children:["Choose a snappy name for your theme, and add it to the ",(0,t.jsx)(n.code,{children:"builtInThemes"})," array inside ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/defaults.js#l27",children:(0,t.jsx)(n.code,{children:"defaults.js"})}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"2-write-some-styles",children:"2. Write some Styles"}),"\n",(0,t.jsxs)(n.p,{children:["Put your theme styles inside ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/styles/color-themes.scss",children:(0,t.jsx)(n.code,{children:"color-themes.scss"})}),".\nCreate a new block, and make sure that ",(0,t.jsx)(n.code,{children:"data-theme"})," matches the theme name you chose above. For example:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-css",children:"html[data-theme='tiger'] {\n --primary: #f58233;\n --background: #0b1021;\n}\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Then you can go ahead and write your own custom CSS. Although all CSS is supported here, the best way to define your theme is by setting the CSS variables. You can find a ",(0,t.jsx)(n.a,{href:"/docs/theming#css-variables",children:"list of all CSS variables, here"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["For a full guide on styling, see ",(0,t.jsx)(n.a,{href:"/docs/theming",children:"Theming Docs"}),"."]}),"\n",(0,t.jsxs)(n.p,{children:["Note that if your theme is just for yourself, and you're not submitting a PR, then you can instead just pass it under ",(0,t.jsx)(n.code,{children:"appConfig.cssThemes"})," inside your config file. And then put your theme in your own stylesheet, and pass it into the Docker container - ",(0,t.jsx)(n.a,{href:"/docs/theming#adding-your-own-theme",children:"see how"}),"."]}),"\n",(0,t.jsx)(n.h2,{id:"writing-translations",children:"Writing Translations"}),"\n",(0,t.jsxs)(n.p,{children:["For full docs about Dashy's multi-language support, see ",(0,t.jsx)(n.a,{href:"/docs/multi-language-support",children:"Multi-Language Support"})]}),"\n",(0,t.jsxs)(n.p,{children:["Dashy is using ",(0,t.jsx)(n.a,{href:"https://vue-i18n.intlify.dev/guide/",children:"vue-i18n"})," to manage multi-language support."]}),"\n",(0,t.jsx)(n.p,{children:"Adding a new language is pretty straightforward, with just three steps:"}),"\n",(0,t.jsx)(n.h3,{id:"1-create-a-new-language-file",children:"1. Create a new Language File"}),"\n",(0,t.jsxs)(n.p,{children:["Create a new JSON file in ",(0,t.jsx)(n.code,{children:"./src/assets/locales"})," name is a 2-digit ",(0,t.jsx)(n.a,{href:"https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes",children:"ISO-639 code"})," for your language, E.g. for German ",(0,t.jsx)(n.code,{children:"de.json"}),", French ",(0,t.jsx)(n.code,{children:"fr.json"})," or Spanish ",(0,t.jsx)(n.code,{children:"es.json"})," - You can find a list of all ISO codes at ",(0,t.jsx)(n.a,{href:"https://www.iso.org/obp/ui",children:"iso.org"}),"."]}),"\n",(0,t.jsx)(n.h3,{id:"2-translate",children:"2. Translate"}),"\n",(0,t.jsxs)(n.p,{children:["Using ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/assets/locales/en.json",children:(0,t.jsx)(n.code,{children:"en.json"})})," as an example, translate the JSON values to your language, while leaving the keys as they are. It's fine to leave out certain items, as if they're missing they will fall-back to English. If you see any attribute which include curly braces (",(0,t.jsx)(n.code,{children:"{xxx}"}),"), then leave the inner value of these braces as is, as this is for variables."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'{\n "theme-maker": {\n "export-button": "Benutzerdefinierte Variablen exportieren",\n "reset-button": "Stile zur\xfccksetzen f\xfcr",\n "show-all-button": "Alle Variablen anzeigen",\n "save-button": "Speichern",\n "cancel-button": "Abbrechen",\n "saved-toast": "{theme} Erfolgreich aktualisiert",\n "reset-toast": "Benutzerdefinierte Farben f\xfcr {theme} entfernt"\n },\n}\n'})}),"\n",(0,t.jsx)(n.h3,{id:"3-add-your-file-to-the-app",children:"3. Add your file to the app"}),"\n",(0,t.jsxs)(n.p,{children:["In ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/utils/languages.js",children:(0,t.jsx)(n.code,{children:"./src/utils/languages.js"})}),", you need to do 2 small things:"]}),"\n",(0,t.jsxs)(n.p,{children:["First import your new translation file, do this at the top of the page.\nE.g. ",(0,t.jsx)(n.code,{children:"import de from '@/assets/locales/de.json';"})]}),"\n",(0,t.jsx)(n.p,{children:"Second, add it to the array of languages, e.g:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-javascript",children:"export const languages = [\n {\n name: 'English',\n code: 'en',\n locale: en,\n flag: '\ud83c\uddec\ud83c\udde7',\n },\n {\n name: 'German', // The name of your language\n code: 'de', // The ISO code of your language\n locale: de, // The name of the file you imported (no quotes)\n flag: '\ud83c\udde9\ud83c\uddea', // An optional flag emoji\n },\n];\n"})}),"\n",(0,t.jsxs)(n.p,{children:["You can also add your new language to the readme file, under the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy#language-switching",children:"Language Switching"})," section, and optionally include your name/ username if you'd like to be credited for your work. Done!"]}),"\n",(0,t.jsx)(n.p,{children:"If you are not comfortable with making pull requests, or do not want to modify the code, then feel free to instead send the translated file to me, and I can add it into the application. I will be sure to credit you appropriately."}),"\n",(0,t.jsx)(n.h2,{id:"adding-a-new-option-in-the-config-file",children:"Adding a new option in the config file"}),"\n",(0,t.jsx)(n.p,{children:"This section is for, adding a new setting to the config file."}),"\n",(0,t.jsxs)(n.p,{children:["All of the users config is specified in ",(0,t.jsx)(n.code,{children:"./user-data/conf.yml"})," - see ",(0,t.jsx)(n.a,{href:"/docs/configuring",children:"Configuring Docs"})," for info.\nIt's important to first ensure that there isn't a similar option already available, the new option is definitely necessary, and most importantly that it is fully backwards compatible."]}),"\n",(0,t.jsx)(n.p,{children:"Next choose the appropriate section to place it under"}),"\n",(0,t.jsxs)(n.ul,{children:["\n",(0,t.jsxs)(n.li,{children:["Application settings should be located under ",(0,t.jsx)(n.code,{children:"appConfig"})]}),"\n",(0,t.jsxs)(n.li,{children:["Page info (such as text and metadata) should be under ",(0,t.jsx)(n.code,{children:"pageInfo"})]}),"\n",(0,t.jsxs)(n.li,{children:["Data relating to specific sections should be under ",(0,t.jsx)(n.code,{children:"section[n].displayData"})]}),"\n",(0,t.jsxs)(n.li,{children:["Settings applied to specific items or widgets, should be under ",(0,t.jsx)(n.code,{children:"item[n]"})," or ",(0,t.jsx)(n.code,{children:"widget[n]"})]}),"\n"]}),"\n",(0,t.jsxs)(n.p,{children:["For example, if your option is added under ",(0,t.jsx)(n.code,{children:"appConfig"}),", you can access it within your component using the ",(0,t.jsx)(n.code,{children:"$store"}),", this is typically placed in a computed property, e.g:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-javascript",children:"computed: {\n appConfig() {\n return this.$store.getters.appConfig;\n },\n ...\n},\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Then, where you want to get the users value within your component, use something like: ",(0,t.jsx)(n.code,{children:"this.appConfig.myProperty"}),". If the user hasn't specified the value, Don't forget to have a fallback or default for it."]}),"\n",(0,t.jsxs)(n.p,{children:["If you have a default fallback value, then this would typically be specified in the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/defaults.js",children:(0,t.jsx)(n.code,{children:"defaults.js"})})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["You will now need to add the definition of your new attribute into the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"ConfigSchema"}),". This will make it available in the UI config editor, and also ensure that the config validation check doesn't fail.\nFor example:"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'"fontAwesomeKey": {\n "type": "string",\n "pattern": "^[a-z0-9]{10}$",\n "description": "API key for font-awesome",\n "example": "0821c65656"\n}\n'})}),"\n",(0,t.jsx)(n.p,{children:"or"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-json",children:'"iconSize": {\n "enum": [ "small", "medium", "large" ],\n "default": "medium",\n "description": "The size of each link item / icon"\n}\n'})}),"\n",(0,t.jsxs)(n.p,{children:["Finally, add your new property to the ",(0,t.jsx)(n.a,{href:"/docs/configuring",children:(0,t.jsx)(n.code,{children:"configuring.md"})})," API docs. Put it under the relevant section, and be sure to include field name, data type, a description and mention that it is optional. If your new feature needs more explanation, then you can also document it under the relevant section elsewhere in the documentation."]}),"\n",(0,t.jsx)(n.p,{children:"Checklist:"}),"\n",(0,t.jsxs)(n.ul,{className:"contains-task-list",children:["\n",(0,t.jsxs)(n.li,{className:"task-list-item",children:[(0,t.jsx)(n.input,{type:"checkbox",disabled:!0})," Ensure the new attribute is actually necessary, and nothing similar already exists"]}),"\n",(0,t.jsxs)(n.li,{className:"task-list-item",children:[(0,t.jsx)(n.input,{type:"checkbox",disabled:!0})," Update the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/ConfigSchema.json",children:"Schema"})," with the parameters for your new option"]}),"\n",(0,t.jsxs)(n.li,{className:"task-list-item",children:[(0,t.jsx)(n.input,{type:"checkbox",disabled:!0})," If required, set a default or fallback value (usually in ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/defaults.js",children:(0,t.jsx)(n.code,{children:"defaults.js"})}),")"]}),"\n",(0,t.jsxs)(n.li,{className:"task-list-item",children:[(0,t.jsx)(n.input,{type:"checkbox",disabled:!0})," Document the new value in ",(0,t.jsx)(n.a,{href:"/docs/configuring",children:(0,t.jsx)(n.code,{children:"configuring.md"})}),", and if required under the relevant section in the docs"]}),"\n",(0,t.jsxs)(n.li,{className:"task-list-item",children:[(0,t.jsx)(n.input,{type:"checkbox",disabled:!0})," Ensure your changes are backwards compatible, and that nothing breaks if the attribute isn't specified"]}),"\n"]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"updating-dependencies",children:"Updating Dependencies"}),"\n",(0,t.jsxs)(n.p,{children:["Running ",(0,t.jsx)(n.code,{children:"yarn upgrade"})," will updated all dependencies based on the ranges specified in the ",(0,t.jsx)(n.code,{children:"package.json"}),". The ",(0,t.jsx)(n.code,{children:"yarn.lock"})," file will be updated, as will the contents of ",(0,t.jsx)(n.code,{children:"./node_modules"}),", for more info, see the ",(0,t.jsx)(n.a,{href:"https://classic.yarnpkg.com/en/docs/cli/upgrade/",children:"yarn upgrade documentation"}),". ",(0,t.jsx)(n.a,{href:"https://github.com/raineorshine/npm-check-updates",children:(0,t.jsx)(n.code,{children:"npm-check-updates"})})," is a useful tool to help with this.\nIt is important to thoroughly test after any big dependency updates."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"developing-netlify-cloud-functions",children:"Developing Netlify Cloud Functions"}),"\n",(0,t.jsx)(n.p,{children:"When Dashy is deployed to Netlify, it is effectively running as a static app, and therefore the server-side code for the Node.js endpoints is not available. However Netlify now supports serverless cloud lambda functions, which can be used to replace most functionality."}),"\n",(0,t.jsx)(n.h3,{id:"1-run-netlify-dev-server",children:"1. Run Netlify Dev Server"}),"\n",(0,t.jsxs)(n.p,{children:["First off all, install the Netlify CLI: ",(0,t.jsx)(n.code,{children:"npm install netlify-cli -g"}),"\nThen, from within the root of Dashy's directory, start the server, by running: ",(0,t.jsx)(n.code,{children:"netlify dev"})]}),"\n",(0,t.jsx)(n.h3,{id:"2-create-a-lambda-function",children:"2. Create a lambda function"}),"\n",(0,t.jsxs)(n.p,{children:["This should be saved in the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/services/serverless-functions",children:(0,t.jsx)(n.code,{children:"./services/serverless-functions"})})," directory"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-javascript",children:"exports.handler = async () => ({\n statusCode: 200,\n body: 'Return some data here...',\n});\n"})}),"\n",(0,t.jsx)(n.h3,{id:"3-redirect-the-node-endpoint-to-the-function",children:"3. Redirect the Node endpoint to the function"}),"\n",(0,t.jsxs)(n.p,{children:["In the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/FEATURE/serverless-functions/netlify.toml",children:(0,t.jsx)(n.code,{children:"netlify.toml"})})," file, add a 301 redirect, with the path to the original Node.js endpoint, and the name of your cloud function"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-toml",children:'[[redirects]]\n from = "/status-check"\n to = "/.netlify/functions/cloud-status-check"\n status = 301\n force = true\n'})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"hiding-page-furniture-on-certain-routes",children:"Hiding Page Furniture on Certain Routes"}),"\n",(0,t.jsx)(n.p,{children:"For some pages (such as the login page, the minimal start page, etc) the basic page furniture, (like header, footer, nav, etc) is not needed. This section explains how you can hide furniture on a new view (step 1), or add a component that should be hidden on certain views (step 2)."}),"\n",(0,t.jsx)(n.h3,{id:"1-add-the-route-name-to-the-should-hide-array",children:"1. Add the route name to the should hide array"}),"\n",(0,t.jsxs)(n.p,{children:["In ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/defaults.js",children:(0,t.jsx)(n.code,{children:"./src/utils/config/defaults.js"})}),", there's an array called ",(0,t.jsx)(n.code,{children:"hideFurnitureOn"}),". Append the name of the route (the same as it appears in ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/router.js",children:(0,t.jsx)(n.code,{children:"router.js"})}),") here."]}),"\n",(0,t.jsx)(n.h3,{id:"2-add-the-conditional-to-the-structural-component-to-hide",children:"2. Add the conditional to the structural component to hide"}),"\n",(0,t.jsx)(n.p,{children:"First, import the helper function:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-javascript",children:"import { shouldBeVisible } from '@/utils/config/SectionHelpers';\n"})}),"\n",(0,t.jsx)(n.p,{children:"Then you can create a computed value, that calls this function, passing in the route name:"}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-javascript",children:"export default {\n ...\n computed: {\n ...\n isVisible() {\n return shouldBeVisible(this.$route.name);\n },\n },\n};\n"})}),"\n",(0,t.jsxs)(n.p,{children:["Finally, in the markup of your component, just add a ",(0,t.jsx)(n.code,{children:"v-if"})," statement, referencing your computed value"]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-vue",children:'
\n ...\n
\n'})}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"adding--using-environmental-variables",children:"Adding / Using Environmental Variables"}),"\n",(0,t.jsxs)(n.p,{children:["All environmental variables are optional. Currently there are not many environmental variables used, as most of the user preferences are stored under ",(0,t.jsx)(n.code,{children:"appConfig"})," in the ",(0,t.jsx)(n.code,{children:"conf.yml"})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["You can set variables either in your environment, or using the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/.env",children:(0,t.jsx)(n.code,{children:".env"})})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["Any environmental variables used by the frontend are preceded with ",(0,t.jsx)(n.code,{children:"VITE_APP_"}),". Vite will merge the contents of your ",(0,t.jsx)(n.code,{children:".env"})," file into the app in a similar way to the ",(0,t.jsx)(n.a,{href:"https://github.com/motdotla/dotenv",children:"'dotenv'"})," package, where any variables that you set on your system will always take preference over the contents of any ",(0,t.jsx)(n.code,{children:".env"})," file."]}),"\n",(0,t.jsxs)(n.p,{children:["If add any new variables, ensure that there is always a fallback (define it in ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/utils/config/defaults.js",children:(0,t.jsx)(n.code,{children:"defaults.js"})}),"), so as to not cause breaking changes. Don't commit the contents of your ",(0,t.jsx)(n.code,{children:".env"})," file to git, but instead take a few moments to document what you've added under the appropriate section. Try and follow the concepts outlined in the ",(0,t.jsx)(n.a,{href:"https://12factor.net/config",children:"12 factor app"}),"."]}),"\n",(0,t.jsx)(n.hr,{}),"\n",(0,t.jsx)(n.h2,{id:"building-a-widget",children:"Building a Widget"}),"\n",(0,t.jsx)(n.h3,{id:"step-0---prerequisites",children:"Step 0 - Prerequisites"}),"\n",(0,t.jsxs)(n.p,{children:["If this is your first time working on Dashy, then the ",(0,t.jsx)(n.a,{href:"/docs/developing",children:"Developing Docs"})," instructions for project setup and running. In short, you just need to clone the project, cd into it, install dependencies (",(0,t.jsx)(n.code,{children:"yarn"}),") and then start the development server (",(0,t.jsx)(n.code,{children:"yarn dev"}),")."]}),"\n",(0,t.jsxs)(n.p,{children:["To build a widget, you'll also need some basic knowledge of Vue.js. The ",(0,t.jsx)(n.a,{href:"https://vuejs.org/v2/guide/",children:"official Vue docs"})," provides a good starting point, as does ",(0,t.jsx)(n.a,{href:"https://www.taniarascia.com/getting-started-with-vue/",children:"this guide"})," by Tania Rascia"]}),"\n",(0,t.jsxs)(n.p,{children:["If you just want to jump straight in, then ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/commit/3da76ce2999f57f76a97454c0276301e39957b8e",children:"here"})," is a complete implementation of a new example widget, or take a look at the ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/blob/master/src/components/Widgets/XkcdComic.vue",children:(0,t.jsx)(n.code,{children:"XkcdComic.vue"})})," widget, which is pretty simple."]}),"\n",(0,t.jsx)(n.h3,{id:"step-1---create-widget",children:"Step 1 - Create Widget"}),"\n",(0,t.jsxs)(n.p,{children:["Firstly, create a new ",(0,t.jsx)(n.code,{children:".vue"})," file under ",(0,t.jsx)(n.a,{href:"https://github.com/Lissy93/dashy/tree/master/src/components/Widgets",children:(0,t.jsx)(n.code,{children:"./src/components/Widgets"})}),"."]}),"\n",(0,t.jsx)(n.pre,{children:(0,t.jsx)(n.code,{className:"language-vue",children:"\n\n - - - + + + -

Alternate Views & Opening Methods

+

Alternate Views & Opening Methods

Views

Dashy has three different views:

    @@ -54,6 +54,6 @@

    Even if the target is not set (or is set to sametab), you can still launch any given app in an alternative method. Either right-click to see all options, or use one of the keyboard shortcuts: Alt + Click will open the modal, and Ctrl + Click will open in a new tab.

    If you don't like the custom context menu, it can be disabled by setting appConfig.disableContextMenu: true.

    -

    If you get a 'Refused to Connect' error in the modal or workspace views, then the target app has it's X-Frame-Options HTTP set to block requests from embedded content. You can easily fix this by setting this header to ALLOW, for instructions on how to do so, see the Troubleshooting Docs.

+

If you get a 'Refused to Connect' error in the modal or workspace views, then the target app has it's X-Frame-Options HTTP set to block requests from embedded content. You can easily fix this by setting this header to ALLOW, for instructions on how to do so, see the Troubleshooting Docs.

\ No newline at end of file diff --git a/docs/authentication/index.html b/docs/authentication/index.html index 813b4ed2..a144e92e 100644 --- a/docs/authentication/index.html +++ b/docs/authentication/index.html @@ -10,15 +10,15 @@ - - - + + + -

Authentication

+
+

⬆️ Back to Top

\ No newline at end of file diff --git a/docs/backup-restore/index.html b/docs/backup-restore/index.html index c4441bdf..d62f060b 100644 --- a/docs/backup-restore/index.html +++ b/docs/backup-restore/index.html @@ -10,15 +10,15 @@ - - - + + + -

Cloud Backup and Restore

+

Cloud Backup and Restore

Beyond the cloud backup/restore service, there are several other self-hosted options you can use to backup Dashy, and any other Docker container data. These are outlined in the Management docs, at: Docker Backup Options.

Dashy has a built-in feature for securely backing up your config to a hosted cloud service, and then restoring it on another instance. This feature is totally optional, and if you do not enable it, then Dashy will not make any external network requests.

This is useful not only for backing up your configuration off-site, but it also enables Dashy to be used without having write a YAML config file, and makes it possible to use a public hosted instance, without the need to self-host.

@@ -99,6 +99,6 @@

For more info, see the API Docs.

-

If you are using Postman, you may find this pre-made collection helpful in getting things setup.

+

If you are using Postman, you may find this pre-made collection helpful in getting things setup.

\ No newline at end of file diff --git a/docs/configuring/index.html b/docs/configuring/index.html index 4746a32a..53c55244 100644 --- a/docs/configuring/index.html +++ b/docs/configuring/index.html @@ -10,15 +10,15 @@ - - - + + + -

Configuring

+

Configuring

All app configuration is specified in /user-data/conf.yml which is in YAML Format format. If you're using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done through the UI. From the UI you can also export, backup, reset, validate and download your configuration file.

There are three ways to edit the config

    @@ -231,7 +231,7 @@ -
    FieldTypeRequiredDescription
    titlestringRequiredThe text to display on the link button
    pathstringRequiredThe URL to navigate to when clicked. Can be relative (e.g. /about) or absolute (e.g. https://example.com or http://192.168.1.1)
    targetstringOptionalThe opening method (external links only). Can be either newtab, sametab, top or parent. Defaults to newtab
    +
    FieldTypeRequiredDescription
    titlestringRequiredThe text to display on the link button
    pathstringRequiredThe URL to navigate to when clicked. Can be relative (e.g. /about) or absolute (e.g. https://example.com or http://192.168.1.1)
    targetstringOptionalThe opening method (external links only). Can be either newtab, sametab, newwindow, top or parent. Defaults to newtab

    ⬆️ Back to Top

    pages[] (optional)

    @@ -1285,6 +1285,6 @@ For these, it is recommended to use an this gist

    If you need any help, feel free to Raise an Issue or Start a Discussion

    Happy Configuring 🤓🔧

    -

    ⬆️ Back to Top

+

⬆️ Back to Top

\ No newline at end of file diff --git a/docs/contributing/index.html b/docs/contributing/index.html index e71033d7..337f286e 100644 --- a/docs/contributing/index.html +++ b/docs/contributing/index.html @@ -10,15 +10,15 @@ - - - + + + -

Contributing

+

Contributing

First off, thank you for considering contributing towards Dashy! 🙌 There are several ways that you can help out, and any contributions, however small will always be very much appreciated. You will be appropriately credited in the readme - huge thank you to everyone who has helped so far 💞

@@ -97,6 +97,6 @@ All content is located either in the Credits Page

Auto-generated contributors

Star-Gazers Over Time

-

](https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy)

+

](https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy)

\ No newline at end of file diff --git a/docs/credits/index.html b/docs/credits/index.html index 611d313b..7573eaec 100644 --- a/docs/credits/index.html +++ b/docs/credits/index.html @@ -10,15 +10,15 @@ - - - + + + -

Credits

+

Credits

Sponsors

Sponsors

Contributors

@@ -54,6 +54,6 @@

You

-

Would you like to be listed here? Whatever your skill set, Dashy needs people like you to help support future development. Check out the Contributing Page for ways that you can get involved. Huge thank you to everyone who has already contributed! 💖

+

Would you like to be listed here? Whatever your skill set, Dashy needs people like you to help support future development. Check out the Contributing Page for ways that you can get involved. Huge thank you to everyone who has already contributed! 💖

\ No newline at end of file diff --git a/docs/deployment/index.html b/docs/deployment/index.html index f8125c6b..6343ee30 100644 --- a/docs/deployment/index.html +++ b/docs/deployment/index.html @@ -10,15 +10,15 @@ - - - + + + -

Deployment

+

Deployment

Welcome to Dashy, so glad you're here :) Deployment is super easy, and there are several methods available depending on what type of system you're using. If you're self-hosting, then deploying with Docker (or similar container engine) is the recommended approach.

Quick Start

If you want to skip the fuss, and get straight down to it, then you can spin up a new instance of Dashy by running:

@@ -266,6 +266,6 @@ -
BrowserMinimum VersionStatus
Chrome / Chromium90+Fully supported
Firefox90+Fully supported
Edge90+Fully supported
Safari14+Supported
Opera76+Supported
Samsung Internet15+Supported
Firefox ESRLatestSupported
Internet Explorer-Not supported
+
BrowserMinimum VersionStatus
Chrome / Chromium90+Fully supported
Firefox90+Fully supported
Edge90+Fully supported
Safari14+Supported
Opera76+Supported
Samsung Internet15+Supported
Firefox ESRLatestSupported
Internet Explorer-Not supported
\ No newline at end of file diff --git a/docs/developing/index.html b/docs/developing/index.html index 22825ae5..6a7beaf9 100644 --- a/docs/developing/index.html +++ b/docs/developing/index.html @@ -10,15 +10,15 @@ - - - + + + -

Developing

+

Developing

This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture. If you're adding new features, you may want to check out the Development Guides docs, for tutorials covering basic tasks.

    @@ -219,6 +219,6 @@ For example, FEATURE/420_Awesome-feature or FIX/690_login-ser

    Known Warnings

    When running the build command, several warnings appear. These are not errors, and do not affect the security or performance of the application. They will be addressed in a future update

    WARN A new version of sass-loader is available. Please upgrade for best experience. - Currently we're using an older version of SASS loader, since the more recent releases do not seem to be compatible with the Vue CLI's webpack configuration.

    -

    WARN asset size limit: The following asset(s) exceed the recommended size limit (244 KiB). - For the PWA to support Windows 10, a splash screen asset is required, and is quite large. This throws a warning, however PWA assets are not loaded until needed, so shouldn't have any impact on application performance. A similar warning is thrown for the Raleway font, and that is looking to be addressed.

+

WARN asset size limit: The following asset(s) exceed the recommended size limit (244 KiB). - For the PWA to support Windows 10, a splash screen asset is required, and is quite large. This throws a warning, however PWA assets are not loaded until needed, so shouldn't have any impact on application performance. A similar warning is thrown for the Raleway font, and that is looking to be addressed.

\ No newline at end of file diff --git a/docs/development-guides/index.html b/docs/development-guides/index.html index 1ac9782d..28310d1a 100644 --- a/docs/development-guides/index.html +++ b/docs/development-guides/index.html @@ -10,15 +10,15 @@ - - - + + + -

Development Guides

+

Development Guides

A series of short tutorials, to guide you through the most common development tasks.

Sections:

    @@ -206,6 +206,6 @@ First create a new computed property, like:

    If required, add a message showing that the component isn't available, using the AccessError component. E.g.

    import AccessError from '@/components/Configuration/AccessError';
    <AccessError v-else />
    -

    The $store.getters.permissions object also returns options for when and where config can be saved, using: allowWriteToDisk, and allowSaveLocally - both are booleans.

+

The $store.getters.permissions object also returns options for when and where config can be saved, using: allowWriteToDisk, and allowSaveLocally - both are booleans.

\ No newline at end of file diff --git a/docs/icons/index.html b/docs/icons/index.html index b74c6b2e..63e1b5cc 100644 --- a/docs/icons/index.html +++ b/docs/icons/index.html @@ -10,15 +10,15 @@ - - - + + + -

Icons

+

Icons

Both sections and items can have an icon, which is specified using the icon attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here.

  • @@ -146,6 +146,6 @@

    Notes

    If you are using icons from an external source, these will be fetched on initial page load automatically, if and when needed. But combining icons from multiple services may have a negative impact on performance.

    You can improve load speeds, by downloading your required icons, and serving them locally. Scaling icons to the minimum required dimensions (e.g. 128x128 or 64x64) will also greatly improve application load times.

    -

    For icons from external sources, please see the Privacy Policies and Licenses for that provider.

+

For icons from external sources, please see the Privacy Policies and Licenses for that provider.

\ No newline at end of file diff --git a/docs/index.html b/docs/index.html index 352342b0..1c6fa712 100644 --- a/docs/index.html +++ b/docs/index.html @@ -10,14 +10,14 @@ - - - + + + -
+
\ No newline at end of file diff --git a/docs/license/index.html b/docs/license/index.html index 23c72c91..0fbe3cb7 100644 --- a/docs/license/index.html +++ b/docs/license/index.html @@ -10,15 +10,15 @@ - - - + + + -

License

+

License

MIT License

Copyright (c) 2021 Alicia Sykes

Permission is hereby granted, free of charge, to any person obtaining a copy @@ -37,6 +37,6 @@ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.


-

For more information about the MIT License, see: https://opensource.org/licenses/MIT

+

For more information about the MIT License, see: https://opensource.org/licenses/MIT

\ No newline at end of file diff --git a/docs/management/index.html b/docs/management/index.html index 82a018ca..e78882cb 100644 --- a/docs/management/index.html +++ b/docs/management/index.html @@ -10,15 +10,15 @@ - - - + + + -

App Management

+

App Management

The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains.

Contents

    @@ -507,6 +507,6 @@ For example: scp -r ./dist/* [username]@[server_ip]:/var/www/dashy/htmlYou may wish to upload your image to a container registry for easier access. Note that if you choose to do this on a public registry, please name your container something other than just 'dashy', to avoid confusion with the official image. You can push your build image, by running: docker push ghcr.io/OWNER/IMAGE_NAME:latest. You will first need to authenticate, this can be done by running echo $CR_PAT | docker login ghcr.io -u USERNAME --password-stdin, where CR_PAT is an environmental variable containing a token generated from your GitHub account. For more info, see the Container Registry Docs.

    ⬆️ Back to Top

    -
+
\ No newline at end of file diff --git a/docs/multi-language-support/index.html b/docs/multi-language-support/index.html index f0b5108a..1511a57b 100644 --- a/docs/multi-language-support/index.html +++ b/docs/multi-language-support/index.html @@ -10,15 +10,15 @@ - - - + + + -

Internationalization

+

Internationalization

Internationalization is the process of making an application available in other languages. This is important, as not everyone is a native English speaker. This page explains how you can switch languages, how to add a new language, and how to make text translatable when writing a new component.

  • Setting your Language
  • @@ -91,6 +91,6 @@ For example: alert(this.$t('my-widget.awesome-text')).

    In ./src/components/Settings/SearchBar.vue:

    <template>
    <form>
    <label for="search-input">{{ $t('search.search-label') }}</label>
    <input
    v-model="searchValue"
    :placeholder="$t('search.search-placeholder')"
    />
    </form>
    </template>

    Then in ./src/assets/locales/en.json:

    -
    {
    "search": {
    "search-label": "Search",
    "search-placeholder": "Start typing to filter",
    },
    ...
    }
+
{
"search": {
"search-label": "Search",
"search-placeholder": "Start typing to filter",
},
...
}
\ No newline at end of file diff --git a/docs/pages-and-sections/index.html b/docs/pages-and-sections/index.html index 4fc8214d..308216fd 100644 --- a/docs/pages-and-sections/index.html +++ b/docs/pages-and-sections/index.html @@ -10,15 +10,15 @@ - - - + + + -

Pages and Sections

+

Pages and Sections

Page Metadata

Set your dashboard's branding under pageInfo

pageInfo:
title: My Dashboard # Used for main h1 title, and browser tab text
description: Home server links
logo: /web-icons/my-logo.png # path/URL to optional logo to display next to title
favicon: 'https://example.com/path/to/icon' # path/URL to a favicon (shows in browser tab)
color: '#2a7cf0' # Hex color, to set the browser/address bar color on mobile (supported browsers only)
footer: '© 2026 Me' # Optional text or HTML content, to display in the pages footer
@@ -82,6 +82,6 @@ The config file must, of course be accessible from within Dashy. If your config
- name: Coding
icon: far fa-code
items:
- title: GitHub
url: https://github.com/
- title: StackOverflow
url: http://stackoverflow.com/

But items can also be grouped together, referred to as sub-items. This is useful for a group of less frequently used items, which you don't want to take up too much space.

Item groups may also have an optional title.

-
- name: Coding
icon: far fa-code
items:
- title: Normal Item 1
- title: Normal Item 2
- title: Languages
subItems:
- title: JavaScript
url: https://developer.mozilla.org
icon: si-javascript
- title: TypeScript
url: https://www.typescriptlang.org/docs
icon: si-typescript
- title: Svelte
url: https://svelte.dev/docs
icon: si-svelte
- title: Go
url: https://go.dev/doc
icon: si-go
+
- name: Coding
icon: far fa-code
items:
- title: Normal Item 1
- title: Normal Item 2
- title: Languages
subItems:
- title: JavaScript
url: https://developer.mozilla.org
icon: si-javascript
- title: TypeScript
url: https://www.typescriptlang.org/docs
icon: si-typescript
- title: Svelte
url: https://svelte.dev/docs
icon: si-svelte
- title: Go
url: https://go.dev/doc
icon: si-go
\ No newline at end of file diff --git a/docs/privacy/index.html b/docs/privacy/index.html index c3ba953f..8a85f0e5 100644 --- a/docs/privacy/index.html +++ b/docs/privacy/index.html @@ -10,15 +10,15 @@ - - - + + + -

Privacy & Security

+

Privacy & Security

Dashy was built with privacy in mind. Self-hosting your own apps and services is a great way to protect yourself from the mass data collection employed by big tech companies, and Dashy was designed to make self-hosting easier.

Dashy operates on the premise, that:

@@ -537,6 +537,6 @@ Dashy can use local storage to keep track of your preferences.


Reporting a Security Issue

Please see our Security.md doc for how to report issues. -We have an actively monitored security mailbox supporting PGP, as well as a GitHub Advisories vulnerability reporting program.

+We have an actively monitored security mailbox supporting PGP, as well as a GitHub Advisories vulnerability reporting program.

\ No newline at end of file diff --git a/docs/quick-start/index.html b/docs/quick-start/index.html index 6062a763..7f615c7b 100644 --- a/docs/quick-start/index.html +++ b/docs/quick-start/index.html @@ -10,15 +10,15 @@ - - - + + + -

Quick Start

+ +
\ No newline at end of file diff --git a/docs/release-workflow/index.html b/docs/release-workflow/index.html index c119b30a..00a804e9 100644 --- a/docs/release-workflow/index.html +++ b/docs/release-workflow/index.html @@ -10,15 +10,15 @@ - - - + + + -

Releases and Workflows

+ +
\ No newline at end of file diff --git a/docs/searching/index.html b/docs/searching/index.html index d3331ed3..282af748 100644 --- a/docs/searching/index.html +++ b/docs/searching/index.html @@ -10,15 +10,15 @@ - - - + + + -

Keyboard Shortcuts

+

Keyboard Shortcuts

Searching

One of the primary purposes of Dashy is to allow you to quickly find and launch a given app. To make this as quick as possible, there is no need to touch the mouse, or press a certain key to begin searching - just start typing. Results will be filtered in real-time. No need to worry about case, special characters or small typos, these are taken care of, and your results should appear.

@@ -63,6 +63,6 @@
appConfig:
webSearch:
openUrlsDirectly: true

You can clear your search term at any time, resting the UI to it's initial state, by pressing Esc. -This can also be used to close any open pop-up modals.

+This can also be used to close any open pop-up modals.

\ No newline at end of file diff --git a/docs/showcase/index.html b/docs/showcase/index.html index 8a6b5cde..4dc4518e 100644 --- a/docs/showcase/index.html +++ b/docs/showcase/index.html @@ -10,15 +10,15 @@ - - - + + + -

Dashy Showcase 🌟

+

Dashy Showcase 🌟

@@ -214,6 +214,6 @@ Dashy is awesome!

Template

If you're submitting a pull request, please use a format similar to this:

-
### [Dashboard Name] (required)

> Submitted by [@username](https://github.com/user) (optional)

[An optional text description, or any interesting details] (optional)

![dashboard-screenshot](https://example.com/url-to-screenshot.png) (required)

---

+
### [Dashboard Name] (required)

> Submitted by [@username](https://github.com/user) (optional)

[An optional text description, or any interesting details] (optional)

![dashboard-screenshot](https://example.com/url-to-screenshot.png) (required)

---

\ No newline at end of file diff --git a/docs/status-indicators/index.html b/docs/status-indicators/index.html index 5db02e6e..ab2c69b3 100644 --- a/docs/status-indicators/index.html +++ b/docs/status-indicators/index.html @@ -10,15 +10,15 @@ - - - + + + -

Status Indicators

+

Status Indicators

Dashy has an optional feature that can display a small icon next to each of your running services, indicating it's current status. This can be useful if you are using Dashy as your homelab's start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message.

Enabling Status Indicators

@@ -62,6 +62,6 @@ For example, statusCheckHeaders: { 'X-Custom-Header': 'foob

How it Works

When the app is loaded, if appConfig.statusCheck: true is set, or if any items have the statusCheck: true enabled, then Dashy will make a request, to https://[your-host-name]/status-check?url=[address-or-servce] (may al include GET params for headers and the secure flag), which in turn will ping that running service, and respond with a status code. Response time is calculated from the difference between start and end time of the request.

When the response completes, an indicator will display next to each item. The color denotes the status: Yellow while waiting for the response to return, green if request was successful, red if it failed, and grey if it was unable to make the request all together.

-

All requests are made straight from your server, there is no intermediary. So providing you are hosting Dashy yourself, and are checking the status of other self-hosted services, there shouldn't be any privacy concerns. Requests are made asynchronously, so this won't have any significant impact on page load speeds. However recurring requests (using statusCheckInterval) may run more slowly if the interval between requests is very short.

+

All requests are made straight from your server, there is no intermediary. So providing you are hosting Dashy yourself, and are checking the status of other self-hosted services, there shouldn't be any privacy concerns. Requests are made asynchronously, so this won't have any significant impact on page load speeds. However recurring requests (using statusCheckInterval) may run more slowly if the interval between requests is very short.

\ No newline at end of file diff --git a/docs/theming/index.html b/docs/theming/index.html index 34985aac..ec6c220c 100644 --- a/docs/theming/index.html +++ b/docs/theming/index.html @@ -10,15 +10,15 @@ - - - + + + -

Theming

+
+
\ No newline at end of file diff --git a/docs/troubleshooting/index.html b/docs/troubleshooting/index.html index c4d0cbee..29cee0ce 100644 --- a/docs/troubleshooting/index.html +++ b/docs/troubleshooting/index.html @@ -10,15 +10,15 @@ - - - + + + -

Troubleshooting

+

Troubleshooting

This document contains common problems and their solutions.
Please ensure your issue isn't listed here, before opening a new ticket.

@@ -595,6 +595,6 @@ You will be notified on your ticket, when a fix has been released.

  • git config --global user.email johndoe@example.com
  • For more info, see Git First Time Setup Docs.

    -

    Note that only contributions to the master / main branch or a project are counted

    +

    Note that only contributions to the master / main branch or a project are counted

    \ No newline at end of file diff --git a/docs/widgets/index.html b/docs/widgets/index.html index b3fb6cd2..99b9fa5a 100644 --- a/docs/widgets/index.html +++ b/docs/widgets/index.html @@ -10,15 +10,15 @@ - - - + + + -

    Widgets

    +

    Widgets

    Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API.

    Contents

      @@ -4232,6 +4232,6 @@ Add the Access-Control-Allow-Origin header, with the value of eithe

      For testing purposes, you can use an addon, which will disable the CORS checks. You can get the Allow-CORS extension for Chrome or Firefox, more details here


      Raising an Issue

      -

      If you need to submit a bug report for a failing widget, then please include the full console output (see how) as well as the relevant parts of your config file. Before sending the request, ensure you've read the docs. If you're new to GitHub, an haven't previously contributed to the project, then please fist star the repo to avoid your ticket being closed by the anti-spam bot.

    +

    If you need to submit a bug report for a failing widget, then please include the full console output (see how) as well as the relevant parts of your config file. Before sending the request, ensure you've read the docs. If you're new to GitHub, an haven't previously contributed to the project, then please fist star the repo to avoid your ticket being closed by the anti-spam bot.

    \ No newline at end of file diff --git a/index.html b/index.html index f5efdef2..64970d69 100644 --- a/index.html +++ b/index.html @@ -10,14 +10,14 @@ - - - + + + -

    Dashy

    The Ultimate Homepage for your Homelab

    Dashy is an open source, highly customizable, easy to use, privacy-respecting dashboard app.Keep Reading...
    208Contributors
    25.2kGitHub Stars
    16.2MDocker Downloads
    5MDoc Site Views

    Theming

    With tons of built-in themes to choose form, plus a UI color palette editor, you can have a unique looking dashboard in no time. There is also support for custom CSS, and since all properties use CSS variables, it is easy to override.
    Learn more in the Docs Docs
    Demo of Theming feature in Dashy

    Icons

    Dashy can auto-fetch icons from the favicon of each of your apps/ services. There is also native support for Font Awesome, Material Design Icons, emoji icons and of course normal images.
    Learn more in the Docs Docs
    Demo of Icons feature in Dashy

    Status Indicators

    Get an instant overview of the health of each of your apps with status indicators. Once enabled, a small dot next to each app will show weather it is up and online, with more info like response time visible on hover.
    Learn more in the Docs Docs
    Demo of Status Indicators feature in Dashy

    Authentication

    Need to protect your dashboard, the simple auth feature is super quick to enable, and has support for multiple users with granular controls. Dashy also has built-in support for Keycloak and other SSO providers.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Widgets

    Display dynamic content from any API-enabled service. Dashy comes bundled with 50+ pre-built widgets for self-hosted services, productivity and monitoring.
    Learn more in the Docs Docs
    Demo of Widgets feature in Dashy

    Alternate Views

    As well as the default home, there is also a minimal view, which makes a great fast-loading browser startpage. Plus a workspace view useful for working on multiple apps at once, all without having to leave your dashboard.
    Learn more in the Docs Docs
    Demo of Alternate Views feature in Dashy

    Launching Methods

    Choose how to launch each of your apps by default, or right click for all options. Apps can be opened in a new tab, the same tab, a quick pop-up modal or in the workspace view.
    Learn more in the Docs Docs
    Demo of Launching Methods feature in Dashy

    Search & Shortcuts

    To search, just start typing, results will be filtered instantly. Use the arrow keys or tab to navigate through results, and press enter to launch. You can also create custom shortcuts for frequently used apps, or add custom tags for easier searching. Dashy can also be used to search the web using your favorite search engine.
    Learn more in the Docs Docs
    Demo of Search & Shortcuts feature in Dashy

    Cloud Backup & Sync

    There is an optional, end-to-end encrypted, free backup cloud service. This enables you to have your config backed up off-site, and to sync data between multiple instances easily.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Configuration

    Dashy's config is specified in a simple YAML file. But you can also configure the directly through the UI, and have changes written to, and backed up on disk. Real-time validation and hints are in place to help you.
    Learn more in the Docs Docs
    Demo of Configuration feature in Dashy

    Multi-Language Support

    Dashy's UI has been translated into several languages by several amazing contributors. Currently English, German, French, Dutch and Slovenian are supported. Your language should be applied automatically, or you can change it in the config menu.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Easy Deployment

    Although Dashy can be easily run on bare metal, the quickest method of getting started is with Docker. Just run `docker run -p 8080:80 lissy93/dashy` to pull, build and and run Dashy.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Customizable Layouts

    Structure your dashboard to fit your use case. From the UI, you can choose between different layouts, item sizes, show/ hide components, switch themes plus more. You can customize pretty much every area of your dashboard. There are config options for custom header, footer, nav bar links, title etc. You can also choose to hide any elements you don't need.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Powered by You

    Support us

    If you've found Dashy useful, consider contributing, or dropping us a star!

    Author

    Dashy was initially built by me, Alicia Sykes (@Lissy93 on GitHub), I build free and open source apps, focused on security, privacy, Linux and self-hosting.

    Sponsors

    Huge thanks to the following sponsors for supporting Dashy's development 💖

    Contributors

    Dashy was made possible, thanks to these wonderful contributors 🩵

    lissy93's avatarlissy93 (2839)liss-bot's avatarliss-bot (674)aspenyang's avataraspenyang (41)CrazyWolf13's avatarCrazyWolf13 (40)marekful's avatarmarekful (27)EVOTk's avatarEVOTk (24)snyk-bot's avatarsnyk-bot (24)azerioxal's avatarazerioxal (22)m42e's avatarm42e (19)pinarruiz's avatarpinarruiz (16)casmbu's avatarcasmbu (14)ricardodemauro's avatarricardodemauro imjimmeh's avatarimjimmeh kashif-se's avatarkashif-se lammersbjorn's avatarlammersbjorn conlan0's avatarconlan0 josuablejeru's avatarjosuablejeru ThrustVector's avatarThrustVector albcp's avataralbcp Cereal916's avatarCereal916 lordpansar's avatarlordpansar jammo2k5's avatarjammo2k5 evroon's avatarevroon alhazmy13's avataralhazmy13 alayham's avataralayham altearius's avataraltearius LinuxSBC's avatarLinuxSBC UrekD's avatarUrekD shumittaher's avatarshumittaher hockwill's avatarhockwill z3r0l1nk's avatarz3r0l1nk zcq100's avatarzcq100 wozboz's avatarwozboz he0119's avatarhe0119 dkadioglu's avatardkadioglu Totto16's avatarTotto16 toddejohnson's avatartoddejohnson remygrandin's avatarremygrandin dasunsrule32's avatardasunsrule32 aviolaris's avataraviolaris Tracreed's avatarTracreed rubenandre's avatarrubenandre rtm516's avatarrtm516 kayedspace's avatarkayedspace onedr0p's avataronedr0p stanly0726's avatarstanly0726 kt-alt's avatarkt-alt Bogyie's avatarBogyie webysther's avatarwebysther jtuzar's avatarjtuzar apgyorfi's avatarapgyorfi alucarddelta's avataralucarddelta lucolvin's avatarlucolvin Kf637's avatarKf637 berksmbl's avatarberksmbl JDB321Sailor's avatarJDB321Sailor domportera's avatardomportera hockeymikey's avatarhockeymikey itsmejoeeey's avataritsmejoeeey k073l's avatark073l m42cel's avatarm42cel mmihaly's avatarmmihaly noblepower1337's avatarnoblepower1337 patrickheeney's avatarpatrickheeney rokiden's avatarrokiden zigotica's avatarzigotica ssrangisetti's avatarssrangisetti BySempron's avatarBySempron alexdelprete's avataralexdelprete synack1337's avatarsynack1337 a-mnich's avatara-mnich BOZG's avatarBOZG moemoeq's avatarmoemoeq daentech's avatardaentech deneor's avatardeneor GuilhermeLCS95's avatarGuilhermeLCS95 kit494way's avatarkit494way pwhelan's avatarpwhelan rnowotniak's avatarrnowotniak TJonesy's avatarTJonesy twsouthwick's avatartwsouthwick thomaswienecke's avatarthomaswienecke armageddon421's avatararmageddon421 dintho's avatardintho turnrye's avatarturnrye sachahjkl's avatarsachahjkl shazzx's avatarshazzx ThinkSalat's avatarThinkSalat sxravan's avatarsxravan royshreyaa's avatarroyshreyaa Smexhy's avatarSmexhy TheZachAttack01's avatarTheZachAttack01 stavros-k's avatarstavros-k XenonR's avatarXenonR StevKast's avatarStevKast ThibautSnoeijs's avatarThibautSnoeijs AmadeusGraves's avatarAmadeusGraves rubjo's avatarrubjo RoubenRehman's avatarRoubenRehman rsb-developer's avatarrsb-developer RobLoach's avatarRobLoach PrynsTag's avatarPrynsTag PlusaN's avatarPlusaN onurege3467's avataronurege3467 oka4shi's avataroka4shi pedorich-n's avatarpedorich-n bubylou's avatarbubylou michaelfeinbier's avatarmichaelfeinbier stinkybernie's avatarstinkybernie miclav's avatarmiclav click0's avatarclick0 vinceh121's avatarvinceh121 tazboyz16's avatartazboyz16 sur1v's avatarsur1v soaibsafi's avatarsoaibsafi sasetz's avatarsasetz s-weigand's avatars-weigand pablomalo's avatarpablomalo onwp's avataronwp nOw-Ay's avatarnOw-Ay markxoe's avatarmarkxoe ivyturner's avatarivyturner kxenoxx's avatarkxenoxx jrobles98's avatarjrobles98 jnach's avatarjnach imlonghao's avatarimlonghao icy-comet's avataricy-comet huangshaohuai's avatarhuangshaohuai flechaig's avatarflechaig baifengheixi's avatarbaifengheixi aypt's avataraypt allozavrr's avatarallozavrr XertDev's avatarXertDev willbrowningme's avatarwillbrowningme vishwamartur's avatarvishwamartur pvillaverde's avatarpvillaverde emiran-orange's avataremiran-orange edugof's avataredugof Dylan-Bs's avatarDylan-Bs dyauss's avatardyauss ddenev's avatarddenev dougaldhub's avatardougaldhub dkyeremeh's avatardkyeremeh deepsourcebot's avatardeepsourcebot DawidPietrykowski's avatarDawidPietrykowski skaarj1989's avatarskaarj1989 The127's avatarThe127 yeralin's avataryeralin danfein's avatardanfein clsty's avatarclsty bskim45's avatarbskim45 BhasherBEL's avatarBhasherBEL bgillet's avatarbgillet BeginCI's avatarBeginCI BRAVO68WEB's avatarBRAVO68WEB AaronPorts's avatarAaronPorts AndreasdeReus's avatarAndreasdeReus alydemah's avataralydemah Alexis-BX's avatarAlexis-BX 5idereal's avatar5idereal 0n1cOn3's avatar0n1cOn3 imsakg's avatarimsakg maximemoreillon's avatarmaximemoreillon Glitch3dPenguin's avatarGlitch3dPenguin pho3nixf1re's avatarpho3nixf1re MatesMotionless's avatarMatesMotionless markusdd's avatarmarkusdd asenov's avatarasenov luispabon's avatarluispabon LeoColman's avatarLeoColman larsreinhardt's avatarlarsreinhardt KierenConnell's avatarKierenConnell kieraneglin's avatarkieraneglin SharpCoder's avatarSharpCoder jonas-schulze's avatarjonas-schulze johnhorton2010's avatarjohnhorton2010 Singebob's avatarSingebob Hellhium's avatarHellhium jjmung's avatarjjmung b1thunt3r's avatarb1thunt3r Compunctus's avatarCompunctus nealian's avatarnealian ip2location-com's avatarip2location-com FraglyG's avatarFraglyG hubortje's avatarhubortje gbrown09's avatargbrown09 FormatToday's avatarFormatToday
    +

    Dashy

    The Ultimate Homepage for your Homelab

    Dashy is an open source, highly customizable, easy to use, privacy-respecting dashboard app.Keep Reading...
    208Contributors
    25.2kGitHub Stars
    16.2MDocker Downloads
    5MDoc Site Views

    Theming

    With tons of built-in themes to choose form, plus a UI color palette editor, you can have a unique looking dashboard in no time. There is also support for custom CSS, and since all properties use CSS variables, it is easy to override.
    Learn more in the Docs Docs
    Demo of Theming feature in Dashy

    Icons

    Dashy can auto-fetch icons from the favicon of each of your apps/ services. There is also native support for Font Awesome, Material Design Icons, emoji icons and of course normal images.
    Learn more in the Docs Docs
    Demo of Icons feature in Dashy

    Status Indicators

    Get an instant overview of the health of each of your apps with status indicators. Once enabled, a small dot next to each app will show weather it is up and online, with more info like response time visible on hover.
    Learn more in the Docs Docs
    Demo of Status Indicators feature in Dashy

    Authentication

    Need to protect your dashboard, the simple auth feature is super quick to enable, and has support for multiple users with granular controls. Dashy also has built-in support for Keycloak and other SSO providers.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Widgets

    Display dynamic content from any API-enabled service. Dashy comes bundled with 50+ pre-built widgets for self-hosted services, productivity and monitoring.
    Learn more in the Docs Docs
    Demo of Widgets feature in Dashy

    Alternate Views

    As well as the default home, there is also a minimal view, which makes a great fast-loading browser startpage. Plus a workspace view useful for working on multiple apps at once, all without having to leave your dashboard.
    Learn more in the Docs Docs
    Demo of Alternate Views feature in Dashy

    Launching Methods

    Choose how to launch each of your apps by default, or right click for all options. Apps can be opened in a new tab, the same tab, a quick pop-up modal or in the workspace view.
    Learn more in the Docs Docs
    Demo of Launching Methods feature in Dashy

    Search & Shortcuts

    To search, just start typing, results will be filtered instantly. Use the arrow keys or tab to navigate through results, and press enter to launch. You can also create custom shortcuts for frequently used apps, or add custom tags for easier searching. Dashy can also be used to search the web using your favorite search engine.
    Learn more in the Docs Docs
    Demo of Search & Shortcuts feature in Dashy

    Cloud Backup & Sync

    There is an optional, end-to-end encrypted, free backup cloud service. This enables you to have your config backed up off-site, and to sync data between multiple instances easily.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Configuration

    Dashy's config is specified in a simple YAML file. But you can also configure the directly through the UI, and have changes written to, and backed up on disk. Real-time validation and hints are in place to help you.
    Learn more in the Docs Docs
    Demo of Configuration feature in Dashy

    Multi-Language Support

    Dashy's UI has been translated into several languages by several amazing contributors. Currently English, German, French, Dutch and Slovenian are supported. Your language should be applied automatically, or you can change it in the config menu.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Easy Deployment

    Although Dashy can be easily run on bare metal, the quickest method of getting started is with Docker. Just run `docker run -p 8080:80 lissy93/dashy` to pull, build and and run Dashy.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Customizable Layouts

    Structure your dashboard to fit your use case. From the UI, you can choose between different layouts, item sizes, show/ hide components, switch themes plus more. You can customize pretty much every area of your dashboard. There are config options for custom header, footer, nav bar links, title etc. You can also choose to hide any elements you don't need.
    Learn more in the Docs Docs
    Screenshot Coming Soon

    Powered by You

    Support us

    If you've found Dashy useful, consider contributing, or dropping us a star!

    Author

    Dashy was initially built by me, Alicia Sykes (@Lissy93 on GitHub), I build free and open source apps, focused on security, privacy, Linux and self-hosting.

    Sponsors

    Huge thanks to the following sponsors for supporting Dashy's development 💖

    Contributors

    Dashy was made possible, thanks to these wonderful contributors 🩵

    lissy93's avatarlissy93 (2841)liss-bot's avatarliss-bot (675)aspenyang's avataraspenyang (41)CrazyWolf13's avatarCrazyWolf13 (40)marekful's avatarmarekful (27)EVOTk's avatarEVOTk (24)snyk-bot's avatarsnyk-bot (24)azerioxal's avatarazerioxal (22)m42e's avatarm42e (19)pinarruiz's avatarpinarruiz (16)casmbu's avatarcasmbu (14)ricardodemauro's avatarricardodemauro imjimmeh's avatarimjimmeh kashif-se's avatarkashif-se lammersbjorn's avatarlammersbjorn conlan0's avatarconlan0 josuablejeru's avatarjosuablejeru ThrustVector's avatarThrustVector albcp's avataralbcp Cereal916's avatarCereal916 lordpansar's avatarlordpansar jammo2k5's avatarjammo2k5 evroon's avatarevroon alayham's avataralayham alhazmy13's avataralhazmy13 altearius's avataraltearius LinuxSBC's avatarLinuxSBC UrekD's avatarUrekD shumittaher's avatarshumittaher hockwill's avatarhockwill z3r0l1nk's avatarz3r0l1nk zcq100's avatarzcq100 wozboz's avatarwozboz he0119's avatarhe0119 dkadioglu's avatardkadioglu Totto16's avatarTotto16 toddejohnson's avatartoddejohnson remygrandin's avatarremygrandin dasunsrule32's avatardasunsrule32 aviolaris's avataraviolaris Tracreed's avatarTracreed rubenandre's avatarrubenandre rtm516's avatarrtm516 kayedspace's avatarkayedspace onedr0p's avataronedr0p stanly0726's avatarstanly0726 kt-alt's avatarkt-alt Bogyie's avatarBogyie webysther's avatarwebysther jtuzar's avatarjtuzar apgyorfi's avatarapgyorfi alucarddelta's avataralucarddelta lucolvin's avatarlucolvin Kf637's avatarKf637 berksmbl's avatarberksmbl JDB321Sailor's avatarJDB321Sailor domportera's avatardomportera hockeymikey's avatarhockeymikey itsmejoeeey's avataritsmejoeeey k073l's avatark073l m42cel's avatarm42cel mmihaly's avatarmmihaly noblepower1337's avatarnoblepower1337 patrickheeney's avatarpatrickheeney rokiden's avatarrokiden zigotica's avatarzigotica ssrangisetti's avatarssrangisetti BySempron's avatarBySempron alexdelprete's avataralexdelprete synack1337's avatarsynack1337 a-mnich's avatara-mnich BOZG's avatarBOZG moemoeq's avatarmoemoeq daentech's avatardaentech deneor's avatardeneor GuilhermeLCS95's avatarGuilhermeLCS95 kit494way's avatarkit494way pwhelan's avatarpwhelan rnowotniak's avatarrnowotniak TJonesy's avatarTJonesy twsouthwick's avatartwsouthwick thomaswienecke's avatarthomaswienecke armageddon421's avatararmageddon421 dintho's avatardintho turnrye's avatarturnrye sachahjkl's avatarsachahjkl shazzx's avatarshazzx ThinkSalat's avatarThinkSalat sxravan's avatarsxravan royshreyaa's avatarroyshreyaa Smexhy's avatarSmexhy TheZachAttack01's avatarTheZachAttack01 stavros-k's avatarstavros-k XenonR's avatarXenonR StevKast's avatarStevKast ThibautSnoeijs's avatarThibautSnoeijs AmadeusGraves's avatarAmadeusGraves rubjo's avatarrubjo RoubenRehman's avatarRoubenRehman rsb-developer's avatarrsb-developer RobLoach's avatarRobLoach PrynsTag's avatarPrynsTag PlusaN's avatarPlusaN onurege3467's avataronurege3467 oka4shi's avataroka4shi pedorich-n's avatarpedorich-n bubylou's avatarbubylou michaelfeinbier's avatarmichaelfeinbier stinkybernie's avatarstinkybernie miclav's avatarmiclav click0's avatarclick0 vinceh121's avatarvinceh121 tazboyz16's avatartazboyz16 sur1v's avatarsur1v soaibsafi's avatarsoaibsafi sasetz's avatarsasetz s-weigand's avatars-weigand pablomalo's avatarpablomalo onwp's avataronwp nOw-Ay's avatarnOw-Ay markxoe's avatarmarkxoe ivyturner's avatarivyturner kxenoxx's avatarkxenoxx jrobles98's avatarjrobles98 jnach's avatarjnach imlonghao's avatarimlonghao icy-comet's avataricy-comet huangshaohuai's avatarhuangshaohuai flechaig's avatarflechaig baifengheixi's avatarbaifengheixi aypt's avataraypt allozavrr's avatarallozavrr XertDev's avatarXertDev willbrowningme's avatarwillbrowningme vishwamartur's avatarvishwamartur pvillaverde's avatarpvillaverde emiran-orange's avataremiran-orange edugof's avataredugof Dylan-Bs's avatarDylan-Bs dyauss's avatardyauss ddenev's avatarddenev dougaldhub's avatardougaldhub dkyeremeh's avatardkyeremeh deepsourcebot's avatardeepsourcebot DawidPietrykowski's avatarDawidPietrykowski skaarj1989's avatarskaarj1989 The127's avatarThe127 yeralin's avataryeralin danfein's avatardanfein clsty's avatarclsty bskim45's avatarbskim45 BhasherBEL's avatarBhasherBEL bgillet's avatarbgillet BeginCI's avatarBeginCI BRAVO68WEB's avatarBRAVO68WEB AaronPorts's avatarAaronPorts AndreasdeReus's avatarAndreasdeReus alydemah's avataralydemah Alexis-BX's avatarAlexis-BX 5idereal's avatar5idereal 0n1cOn3's avatar0n1cOn3 imsakg's avatarimsakg maximemoreillon's avatarmaximemoreillon Glitch3dPenguin's avatarGlitch3dPenguin pho3nixf1re's avatarpho3nixf1re MatesMotionless's avatarMatesMotionless markusdd's avatarmarkusdd asenov's avatarasenov luispabon's avatarluispabon LeoColman's avatarLeoColman larsreinhardt's avatarlarsreinhardt KierenConnell's avatarKierenConnell kieraneglin's avatarkieraneglin SharpCoder's avatarSharpCoder jonas-schulze's avatarjonas-schulze johnhorton2010's avatarjohnhorton2010 Singebob's avatarSingebob Hellhium's avatarHellhium jjmung's avatarjjmung b1thunt3r's avatarb1thunt3r Compunctus's avatarCompunctus nealian's avatarnealian ip2location-com's avatarip2location-com FraglyG's avatarFraglyG hubortje's avatarhubortje gbrown09's avatargbrown09 FormatToday's avatarFormatToday
    \ No newline at end of file diff --git a/markdown-page/index.html b/markdown-page/index.html index ce375cd6..54f08a8b 100644 --- a/markdown-page/index.html +++ b/markdown-page/index.html @@ -10,15 +10,15 @@ - - - + + + -

    Markdown page example

    + \ No newline at end of file diff --git a/search-index.json b/search-index.json index ae765045..fed43695 100644 --- a/search-index.json +++ b/search-index.json @@ -1 +1 @@ -[{"documents":[{"i":1,"t":"","u":"/docs/","b":["Documentation"]},{"i":2,"t":"Alternate Views & Opening Methods","u":"/docs/alternate-views","b":["Documentation","Feature Docs"]},{"i":13,"t":"Authentication","u":"/docs/authentication","b":["Documentation","Feature Docs"]},{"i":87,"t":"Cloud Backup and Restore","u":"/docs/backup-restore","b":["Documentation","Feature Docs"]},{"i":110,"t":"Configuring","u":"/docs/configuring","b":["Documentation","Running Dashy"]},{"i":167,"t":"Contributing","u":"/docs/contributing","b":["Documentation","Community"]},{"i":201,"t":"Credits","u":"/docs/credits","b":["Documentation","Misc"]},{"i":215,"t":"Deployment","u":"/docs/deployment","b":["Documentation","Running Dashy"]},{"i":264,"t":"Developing","u":"/docs/developing","b":["Documentation","Community"]},{"i":301,"t":"Development Guides","u":"/docs/development-guides","b":["Documentation","Community"]},{"i":350,"t":"Icons","u":"/docs/icons","b":["Documentation","Feature Docs"]},{"i":380,"t":"License","u":"/docs/license","b":["Documentation","Misc"]},{"i":383,"t":"App Management","u":"/docs/management","b":["Documentation","Running Dashy"]},{"i":507,"t":"Internationalization","u":"/docs/multi-language-support","b":["Documentation","Feature Docs"]},{"i":535,"t":"Pages and Sections","u":"/docs/pages-and-sections","b":["Documentation","Feature Docs"]},{"i":556,"t":"Privacy & Security","u":"/docs/privacy","b":["Documentation","Misc"]},{"i":621,"t":"Quick Start","u":"/docs/quick-start","b":["Quick Start","Running Dashy"]},{"i":641,"t":"Releases and Workflows","u":"/docs/release-workflow","b":["Documentation","Misc"]},{"i":669,"t":"Keyboard Shortcuts","u":"/docs/searching","b":["Documentation","Feature Docs"]},{"i":696,"t":"Dashy Showcase 🌟","u":"/docs/showcase","b":["Documentation","Community"]},{"i":757,"t":"Dashy Showcase 🌟","u":"/docs/showcase/","b":["Documentation","Community"]},{"i":818,"t":"Status Indicators","u":"/docs/status-indicators","b":["Documentation","Feature Docs"]},{"i":838,"t":"Theming","u":"/docs/theming","b":["Documentation","Feature Docs"]},{"i":871,"t":"Troubleshooting","u":"/docs/troubleshooting","b":["Documentation","Running Dashy"]},{"i":1024,"t":"Widgets","u":"/docs/widgets","b":["Documentation","Feature Docs"]}],"index":{"version":"2.3.9","fields":["t"],"fieldVectors":[["t/1",[]],["t/2",[0,1.602,1,1.602,2,0.985,3,1.602,4,1.602]],["t/13",[5,3.442]],["t/87",[6,2.187,7,2.187,8,2.187]],["t/110",[9,3.442]],["t/167",[10,3.442]],["t/201",[11,3.442]],["t/215",[12,3.442]],["t/264",[13,2.826]],["t/301",[13,2.196,14,2.675]],["t/350",[15,3.442]],["t/380",[16,3.442]],["t/383",[17,2.675,18,2.675]],["t/507",[19,3.442]],["t/535",[20,2.675,21,2.675]],["t/556",[2,1.345,22,2.187,23,2.187]],["t/621",[24,2.675,25,2.675]],["t/641",[26,2.675,27,2.675]],["t/669",[28,2.675,29,2.675]],["t/696",[2,1.345,30,1.795,31,1.795]],["t/757",[2,1.345,30,1.795,31,1.795]],["t/818",[32,2.675,33,2.675]],["t/838",[34,3.442]],["t/871",[35,3.442]],["t/1024",[36,3.442]]],"invertedIndex":[["",{"_index":2,"t":{"2":{"position":[[16,1]]},"556":{"position":[[8,1]]},"696":{"position":[[15,2]]},"757":{"position":[[15,2]]}}}],["altern",{"_index":0,"t":{"2":{"position":[[0,9]]}}}],["app",{"_index":17,"t":{"383":{"position":[[0,3]]}}}],["authent",{"_index":5,"t":{"13":{"position":[[0,14]]}}}],["backup",{"_index":7,"t":{"87":{"position":[[6,6]]}}}],["cloud",{"_index":6,"t":{"87":{"position":[[0,5]]}}}],["configur",{"_index":9,"t":{"110":{"position":[[0,11]]}}}],["contribut",{"_index":10,"t":{"167":{"position":[[0,12]]}}}],["credit",{"_index":11,"t":{"201":{"position":[[0,7]]}}}],["dashi",{"_index":30,"t":{"696":{"position":[[0,5]]},"757":{"position":[[0,5]]}}}],["deploy",{"_index":12,"t":{"215":{"position":[[0,10]]}}}],["develop",{"_index":13,"t":{"264":{"position":[[0,10]]},"301":{"position":[[0,11]]}}}],["guid",{"_index":14,"t":{"301":{"position":[[12,6]]}}}],["icon",{"_index":15,"t":{"350":{"position":[[0,5]]}}}],["indic",{"_index":33,"t":{"818":{"position":[[7,10]]}}}],["internation",{"_index":19,"t":{"507":{"position":[[0,20]]}}}],["keyboard",{"_index":28,"t":{"669":{"position":[[0,8]]}}}],["licens",{"_index":16,"t":{"380":{"position":[[0,7]]}}}],["manag",{"_index":18,"t":{"383":{"position":[[4,10]]}}}],["method",{"_index":4,"t":{"2":{"position":[[26,7]]}}}],["open",{"_index":3,"t":{"2":{"position":[[18,7]]}}}],["page",{"_index":20,"t":{"535":{"position":[[0,5]]}}}],["privaci",{"_index":22,"t":{"556":{"position":[[0,7]]}}}],["quick",{"_index":24,"t":{"621":{"position":[[0,5]]}}}],["releas",{"_index":26,"t":{"641":{"position":[[0,8]]}}}],["restor",{"_index":8,"t":{"87":{"position":[[17,7]]}}}],["section",{"_index":21,"t":{"535":{"position":[[10,8]]}}}],["secur",{"_index":23,"t":{"556":{"position":[[10,8]]}}}],["shortcut",{"_index":29,"t":{"669":{"position":[[9,9]]}}}],["showcas",{"_index":31,"t":{"696":{"position":[[6,8]]},"757":{"position":[[6,8]]}}}],["start",{"_index":25,"t":{"621":{"position":[[6,5]]}}}],["statu",{"_index":32,"t":{"818":{"position":[[0,6]]}}}],["theme",{"_index":34,"t":{"838":{"position":[[0,7]]}}}],["troubleshoot",{"_index":35,"t":{"871":{"position":[[0,15]]}}}],["view",{"_index":1,"t":{"2":{"position":[[10,5]]}}}],["widget",{"_index":36,"t":{"1024":{"position":[[0,7]]}}}],["workflow",{"_index":27,"t":{"641":{"position":[[13,9]]}}}]],"pipeline":["stemmer"]}},{"documents":[{"i":3,"t":"Views","u":"/docs/alternate-views","h":"#views","p":2},{"i":5,"t":"Default","u":"/docs/alternate-views","h":"#default","p":2},{"i":7,"t":"Workspace","u":"/docs/alternate-views","h":"#workspace","p":2},{"i":9,"t":"Minimal View","u":"/docs/alternate-views","h":"#minimal-view","p":2},{"i":11,"t":"Opening Methods","u":"/docs/alternate-views","h":"#opening-methods","p":2},{"i":15,"t":"Built-In Auth","u":"/docs/authentication","h":"#built-in-auth","p":13},{"i":17,"t":"Setting Up Authentication","u":"/docs/authentication","h":"#setting-up-authentication","p":13},{"i":19,"t":"Hash Password","u":"/docs/authentication","h":"#hash-password","p":13},{"i":21,"t":"Logging In and Out","u":"/docs/authentication","h":"#logging-in-and-out","p":13},{"i":23,"t":"Enabling Guest Access","u":"/docs/authentication","h":"#enabling-guest-access","p":13},{"i":25,"t":"Granular Access","u":"/docs/authentication","h":"#granular-access","p":13},{"i":27,"t":"Permissions","u":"/docs/authentication","h":"#permissions","p":13},{"i":29,"t":"Using Environment Variables for Passwords","u":"/docs/authentication","h":"#using-environment-variables-for-passwords","p":13},{"i":31,"t":"Adding HTTP Auth to Configuration","u":"/docs/authentication","h":"#adding-http-auth-to-configuration","p":13},{"i":33,"t":"Security","u":"/docs/authentication","h":"#security","p":13},{"i":35,"t":"HTTP Auth","u":"/docs/authentication","h":"#http-auth","p":13},{"i":37,"t":"Using config-file users (recommended)","u":"/docs/authentication","h":"#using-config-file-users-recommended","p":13},{"i":39,"t":"Using static credentials","u":"/docs/authentication","h":"#using-static-credentials","p":13},{"i":41,"t":"Keycloak","u":"/docs/authentication","h":"#keycloak","p":13},{"i":43,"t":"1. Deploy Keycloak","u":"/docs/authentication","h":"#1-deploy-keycloak","p":13},{"i":45,"t":"2. Setup Keycloak Users","u":"/docs/authentication","h":"#2-setup-keycloak-users","p":13},{"i":47,"t":"3. Enable Keycloak in Dashy Config File","u":"/docs/authentication","h":"#3-enable-keycloak-in-dashy-config-file","p":13},{"i":49,"t":"4. Add groups and roles (Optional)","u":"/docs/authentication","h":"#4-add-groups-and-roles-optional","p":13},{"i":51,"t":"CORS Headers","u":"/docs/authentication","h":"#cors-headers","p":13},{"i":53,"t":"How server-side enforcement works","u":"/docs/authentication","h":"#how-server-side-enforcement-works","p":13},{"i":55,"t":"Troubleshooting Keycloak","u":"/docs/authentication","h":"#troubleshooting-keycloak","p":13},{"i":57,"t":"Header Authentication","u":"/docs/authentication","h":"#header-authentication","p":13},{"i":59,"t":"Configuration","u":"/docs/authentication","h":"#configuration","p":13},{"i":61,"t":"How it Works","u":"/docs/authentication","h":"#how-it-works","p":13},{"i":63,"t":"Notes","u":"/docs/authentication","h":"#notes","p":13},{"i":65,"t":"OIDC","u":"/docs/authentication","h":"#oidc","p":13},{"i":67,"t":"How server-side enforcement works","u":"/docs/authentication","h":"#how-server-side-enforcement-works-1","p":13},{"i":69,"t":"authentik","u":"/docs/authentication","h":"#authentik","p":13},{"i":71,"t":"Alternative Authentication Methods","u":"/docs/authentication","h":"#alternative-authentication-methods","p":13},{"i":73,"t":"Reverse proxy auth","u":"/docs/authentication","h":"#reverse-proxy-auth","p":13},{"i":75,"t":"Zero-trust tunnels","u":"/docs/authentication","h":"#zero-trust-tunnels","p":13},{"i":77,"t":"VPN","u":"/docs/authentication","h":"#vpn","p":13},{"i":79,"t":"IP-based access","u":"/docs/authentication","h":"#ip-based-access","p":13},{"i":81,"t":"Web server authentication","u":"/docs/authentication","h":"#web-server-authentication","p":13},{"i":83,"t":"SSO / OAuth providers","u":"/docs/authentication","h":"#sso--oauth-providers","p":13},{"i":85,"t":"Cloud hosting providers","u":"/docs/authentication","h":"#cloud-hosting-providers","p":13},{"i":89,"t":"How it Works","u":"/docs/backup-restore","h":"#how-it-works","p":87},{"i":91,"t":"Creating a Backup","u":"/docs/backup-restore","h":"#creating-a-backup","p":87},{"i":93,"t":"Restoring a Backup","u":"/docs/backup-restore","h":"#restoring-a-backup","p":87},{"i":95,"t":"Privacy & Security","u":"/docs/backup-restore","h":"#privacy--security","p":87},{"i":97,"t":"Fair Use Policy","u":"/docs/backup-restore","h":"#fair-use-policy","p":87},{"i":99,"t":"Self-Hosting the Backup Server","u":"/docs/backup-restore","h":"#self-hosting-the-backup-server","p":87},{"i":100,"t":"Quick Start","u":"/docs/backup-restore","h":"#quick-start","p":87},{"i":102,"t":"Populate wrangler.toml","u":"/docs/backup-restore","h":"#populate-wranglertoml","p":87},{"i":104,"t":"Complete index.js","u":"/docs/backup-restore","h":"#complete-indexjs","p":87},{"i":106,"t":"Commands","u":"/docs/backup-restore","h":"#commands","p":87},{"i":108,"t":"API","u":"/docs/backup-restore","h":"#api","p":87},{"i":112,"t":"There are three ways to edit the config","u":"/docs/configuring","h":"#there-are-three-ways-to-edit-the-config","p":110},{"i":114,"t":"Tips","u":"/docs/configuring","h":"#tips","p":110},{"i":116,"t":"Contents","u":"/docs/configuring","h":"#contents","p":110},{"i":118,"t":"Top-Level Fields","u":"/docs/configuring","h":"#top-level-fields","p":110},{"i":120,"t":"PageInfo","u":"/docs/configuring","h":"#pageinfo","p":110},{"i":122,"t":"pageInfo.navLinks (optional)","u":"/docs/configuring","h":"#pageinfonavlinks-optional","p":110},{"i":124,"t":"pages[] (optional)","u":"/docs/configuring","h":"#pages-optional","p":110},{"i":126,"t":"appConfig (optional)","u":"/docs/configuring","h":"#appconfig-optional","p":110},{"i":128,"t":"appConfig.auth (optional)","u":"/docs/configuring","h":"#appconfigauth-optional","p":110},{"i":130,"t":"appConfig.auth.users (optional)","u":"/docs/configuring","h":"#appconfigauthusers-optional","p":110},{"i":132,"t":"appConfig.auth.keycloak (optional)","u":"/docs/configuring","h":"#appconfigauthkeycloak-optional","p":110},{"i":134,"t":"appConfig.auth.headerAuth (optional)","u":"/docs/configuring","h":"#appconfigauthheaderauth-optional","p":110},{"i":136,"t":"appConfig.auth.oidc (optional)","u":"/docs/configuring","h":"#appconfigauthoidc-optional","p":110},{"i":138,"t":"appConfig.webSearch (optional)","u":"/docs/configuring","h":"#appconfigwebsearch-optional","p":110},{"i":140,"t":"appConfig.hideComponents (optional)","u":"/docs/configuring","h":"#appconfighidecomponents-optional","p":110},{"i":142,"t":"section","u":"/docs/configuring","h":"#section","p":110},{"i":144,"t":"section.item","u":"/docs/configuring","h":"#sectionitem","p":110},{"i":146,"t":"item.displayData (optional)","u":"/docs/configuring","h":"#itemdisplaydata-optional","p":110},{"i":148,"t":"section.widgets (optional)","u":"/docs/configuring","h":"#sectionwidgets-optional","p":110},{"i":150,"t":"section.displayData (optional)","u":"/docs/configuring","h":"#sectiondisplaydata-optional","p":110},{"i":152,"t":"section.icon and section.item.icon","u":"/docs/configuring","h":"#sectionicon-and-sectionitemicon","p":110},{"i":154,"t":"section.displayData.hideForKeycloakUsers, section.displayData.showForKeycloakUsers, item.displayData.hideForKeycloakUsers and item.displayData.showForKeycloakUsers","u":"/docs/configuring","h":"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers","p":110},{"i":156,"t":"Notes","u":"/docs/configuring","h":"#notes","p":110},{"i":157,"t":"Editing Config through the UI","u":"/docs/configuring","h":"#editing-config-through-the-ui","p":110},{"i":159,"t":"About YAML","u":"/docs/configuring","h":"#about-yaml","p":110},{"i":161,"t":"Config Saving Methods","u":"/docs/configuring","h":"#config-saving-methods","p":110},{"i":163,"t":"Preventing Changes","u":"/docs/configuring","h":"#preventing-changes","p":110},{"i":165,"t":"Example","u":"/docs/configuring","h":"#example","p":110},{"i":169,"t":"Take a 2-minute survey","u":"/docs/contributing","h":"#take-a-2-minute-survey","p":167},{"i":171,"t":"Share your dashboard","u":"/docs/contributing","h":"#share-your-dashboard","p":167},{"i":173,"t":"Make a small donation","u":"/docs/contributing","h":"#make-a-small-donation","p":167},{"i":175,"t":"You can also send a one-off small contribution using crypto","u":"/docs/contributing","h":"#you-can-also-send-a-one-off-small-contribution-using-crypto","p":167},{"i":177,"t":"Enable Anonymous Bug Reports","u":"/docs/contributing","h":"#enable-anonymous-bug-reports","p":167},{"i":179,"t":"Add Translations","u":"/docs/contributing","h":"#add-translations","p":167},{"i":181,"t":"Submit a PR","u":"/docs/contributing","h":"#submit-a-pr","p":167},{"i":183,"t":"Improve the Docs","u":"/docs/contributing","h":"#improve-the-docs","p":167},{"i":185,"t":"Raise a bug","u":"/docs/contributing","h":"#raise-a-bug","p":167},{"i":187,"t":"Join the discussion","u":"/docs/contributing","h":"#join-the-discussion","p":167},{"i":189,"t":"Request a feature via BountySource","u":"/docs/contributing","h":"#request-a-feature-via-bountysource","p":167},{"i":191,"t":"Spread the word","u":"/docs/contributing","h":"#spread-the-word","p":167},{"i":193,"t":"Star, Upvote or Leave a Review","u":"/docs/contributing","h":"#star-upvote-or-leave-a-review","p":167},{"i":195,"t":"Follow for More","u":"/docs/contributing","h":"#follow-for-more","p":167},{"i":197,"t":"Contributors","u":"/docs/contributing","h":"#contributors","p":167},{"i":199,"t":"Star-Gazers Over Time","u":"/docs/contributing","h":"#star-gazers-over-time","p":167},{"i":202,"t":"Sponsors","u":"/docs/credits","h":"#sponsors","p":201},{"i":203,"t":"Contributors","u":"/docs/credits","h":"#contributors","p":201},{"i":204,"t":"Newest Stargazers","u":"/docs/credits","h":"#newest-stargazers","p":201},{"i":205,"t":"Dependencies","u":"/docs/credits","h":"#dependencies","p":201},{"i":207,"t":"Core","u":"/docs/credits","h":"#core","p":201},{"i":209,"t":"Utilities","u":"/docs/credits","h":"#utilities","p":201},{"i":211,"t":"Frontend Components","u":"/docs/credits","h":"#frontend-components","p":201},{"i":213,"t":"You","u":"/docs/credits","h":"#you","p":201},{"i":217,"t":"Quick Start","u":"/docs/deployment","h":"#quick-start","p":215},{"i":219,"t":"Deployment Methods","u":"/docs/deployment","h":"#deployment-methods","p":215},{"i":221,"t":"Deploy with Docker","u":"/docs/deployment","h":"#deploy-with-docker","p":215},{"i":223,"t":"Using Docker Compose","u":"/docs/deployment","h":"#using-docker-compose","p":215},{"i":225,"t":"Podman","u":"/docs/deployment","h":"#podman","p":215},{"i":227,"t":"Portainer","u":"/docs/deployment","h":"#portainer","p":215},{"i":229,"t":"Kubernetes","u":"/docs/deployment","h":"#kubernetes","p":215},{"i":231,"t":"Unraid","u":"/docs/deployment","h":"#unraid","p":215},{"i":233,"t":"Home Server Platforms","u":"/docs/deployment","h":"#home-server-platforms","p":215},{"i":235,"t":"Synology NAS","u":"/docs/deployment","h":"#synology-nas","p":215},{"i":237,"t":"Build from Source","u":"/docs/deployment","h":"#build-from-source","p":215},{"i":239,"t":"Deploy to Cloud Service","u":"/docs/deployment","h":"#deploy-to-cloud-service","p":215},{"i":241,"t":"Netlify","u":"/docs/deployment","h":"#netlify","p":215},{"i":243,"t":"Vercel","u":"/docs/deployment","h":"#vercel","p":215},{"i":245,"t":"Easypanel","u":"/docs/deployment","h":"#easypanel","p":215},{"i":247,"t":"EdgeOne Pages","u":"/docs/deployment","h":"#edgeone-pages","p":215},{"i":249,"t":"Play-with-Docker","u":"/docs/deployment","h":"#play-with-docker","p":215},{"i":251,"t":"Hosting with CDN","u":"/docs/deployment","h":"#hosting-with-cdn","p":215},{"i":253,"t":"Requirements","u":"/docs/deployment","h":"#requirements","p":215},{"i":254,"t":"System Requirements","u":"/docs/deployment","h":"#system-requirements","p":215},{"i":256,"t":"Docker","u":"/docs/deployment","h":"#docker","p":215},{"i":258,"t":"Bare Metal","u":"/docs/deployment","h":"#bare-metal","p":215},{"i":260,"t":"CDN / Cloud Deploy","u":"/docs/deployment","h":"#cdn--cloud-deploy","p":215},{"i":262,"t":"Browser Support","u":"/docs/deployment","h":"#browser-support","p":215},{"i":266,"t":"Setting up the Dev Environment","u":"/docs/developing","h":"#setting-up-the-dev-environment","p":264},{"i":267,"t":"Prerequisites","u":"/docs/developing","h":"#prerequisites","p":264},{"i":269,"t":"Running the Project","u":"/docs/developing","h":"#running-the-project","p":264},{"i":271,"t":"Project Commands","u":"/docs/developing","h":"#project-commands","p":264},{"i":273,"t":"Environmental Variables","u":"/docs/developing","h":"#environmental-variables","p":264},{"i":275,"t":"Environment Modes","u":"/docs/developing","h":"#environment-modes","p":264},{"i":277,"t":"Git Strategy","u":"/docs/developing","h":"#git-strategy","p":264},{"i":278,"t":"Git Flow","u":"/docs/developing","h":"#git-flow","p":264},{"i":280,"t":"Git Branch Naming","u":"/docs/developing","h":"#git-branch-naming","p":264},{"i":282,"t":"Commit Emojis","u":"/docs/developing","h":"#commit-emojis","p":264},{"i":284,"t":"PR Guidelines","u":"/docs/developing","h":"#pr-guidelines","p":264},{"i":286,"t":"Resources for Beginners","u":"/docs/developing","h":"#resources-for-beginners","p":264},{"i":288,"t":"App Info","u":"/docs/developing","h":"#app-info","p":264},{"i":289,"t":"Style Guide","u":"/docs/developing","h":"#style-guide","p":264},{"i":291,"t":"Application Structure","u":"/docs/developing","h":"#application-structure","p":264},{"i":293,"t":"Development Tools","u":"/docs/developing","h":"#development-tools","p":264},{"i":294,"t":"Performance - Lighthouse","u":"/docs/developing","h":"#performance---lighthouse","p":264},{"i":296,"t":"Dependencies - BundlePhobia","u":"/docs/developing","h":"#dependencies---bundlephobia","p":264},{"i":298,"t":"Notes","u":"/docs/developing","h":"#notes-1","p":264},{"i":299,"t":"Known Warnings","u":"/docs/developing","h":"#known-warnings","p":264},{"i":303,"t":"Creating a new theme","u":"/docs/development-guides","h":"#creating-a-new-theme","p":301},{"i":305,"t":"1. Add Theme Name","u":"/docs/development-guides","h":"#1-add-theme-name","p":301},{"i":307,"t":"2. Write some Styles","u":"/docs/development-guides","h":"#2-write-some-styles","p":301},{"i":309,"t":"Writing Translations","u":"/docs/development-guides","h":"#writing-translations","p":301},{"i":311,"t":"1. Create a new Language File","u":"/docs/development-guides","h":"#1-create-a-new-language-file","p":301},{"i":313,"t":"2. Translate","u":"/docs/development-guides","h":"#2-translate","p":301},{"i":315,"t":"3. Add your file to the app","u":"/docs/development-guides","h":"#3-add-your-file-to-the-app","p":301},{"i":317,"t":"Adding a new option in the config file","u":"/docs/development-guides","h":"#adding-a-new-option-in-the-config-file","p":301},{"i":319,"t":"Updating Dependencies","u":"/docs/development-guides","h":"#updating-dependencies","p":301},{"i":321,"t":"Developing Netlify Cloud Functions","u":"/docs/development-guides","h":"#developing-netlify-cloud-functions","p":301},{"i":323,"t":"1. Run Netlify Dev Server","u":"/docs/development-guides","h":"#1-run-netlify-dev-server","p":301},{"i":325,"t":"2. Create a lambda function","u":"/docs/development-guides","h":"#2-create-a-lambda-function","p":301},{"i":327,"t":"3. Redirect the Node endpoint to the function","u":"/docs/development-guides","h":"#3-redirect-the-node-endpoint-to-the-function","p":301},{"i":329,"t":"Hiding Page Furniture on Certain Routes","u":"/docs/development-guides","h":"#hiding-page-furniture-on-certain-routes","p":301},{"i":331,"t":"1. Add the route name to the should hide array","u":"/docs/development-guides","h":"#1-add-the-route-name-to-the-should-hide-array","p":301},{"i":333,"t":"2. Add the conditional to the structural component to hide","u":"/docs/development-guides","h":"#2-add-the-conditional-to-the-structural-component-to-hide","p":301},{"i":335,"t":"Adding / Using Environmental Variables","u":"/docs/development-guides","h":"#adding--using-environmental-variables","p":301},{"i":337,"t":"Building a Widget","u":"/docs/development-guides","h":"#building-a-widget","p":301},{"i":338,"t":"Step 0 - Prerequisites","u":"/docs/development-guides","h":"#step-0---prerequisites","p":301},{"i":340,"t":"Step 1 - Create Widget","u":"/docs/development-guides","h":"#step-1---create-widget","p":301},{"i":342,"t":"Step 2 - Adding Functionality","u":"/docs/development-guides","h":"#step-2---adding-functionality","p":301},{"i":344,"t":"Step 3 - Register","u":"/docs/development-guides","h":"#step-3---register","p":301},{"i":346,"t":"Step 4 - Docs","u":"/docs/development-guides","h":"#step-4---docs","p":301},{"i":348,"t":"Respecting Config Permissions","u":"/docs/development-guides","h":"#respecting-config-permissions","p":301},{"i":352,"t":"Favicons","u":"/docs/icons","h":"#favicons","p":350},{"i":354,"t":"Font Awesome","u":"/docs/icons","h":"#font-awesome","p":350},{"i":356,"t":"Simple Icons","u":"/docs/icons","h":"#simple-icons","p":350},{"i":358,"t":"Generative Icons","u":"/docs/icons","h":"#generative-icons","p":350},{"i":360,"t":"Emoji Icons","u":"/docs/icons","h":"#emoji-icons","p":350},{"i":362,"t":"selfh.st Icons","u":"/docs/icons","h":"#selfhst-icons","p":350},{"i":364,"t":"Home-Lab Icons","u":"/docs/icons","h":"#home-lab-icons","p":350},{"i":366,"t":"Material Design Icons","u":"/docs/icons","h":"#material-design-icons","p":350},{"i":368,"t":"Icons by URL","u":"/docs/icons","h":"#icons-by-url","p":350},{"i":370,"t":"Local Icons","u":"/docs/icons","h":"#local-icons","p":350},{"i":372,"t":"Default Icon","u":"/docs/icons","h":"#default-icon","p":350},{"i":374,"t":"No Icon","u":"/docs/icons","h":"#no-icon","p":350},{"i":376,"t":"Icon Collections and Resources","u":"/docs/icons","h":"#icon-collections-and-resources","p":350},{"i":378,"t":"Notes","u":"/docs/icons","h":"#notes","p":350},{"i":381,"t":"MIT License","u":"/docs/license","h":"#mit-license","p":380},{"i":385,"t":"Contents","u":"/docs/management","h":"#contents","p":383},{"i":387,"t":"Providing Assets","u":"/docs/management","h":"#providing-assets","p":383},{"i":389,"t":"File Ownership and Permissions","u":"/docs/management","h":"#file-ownership-and-permissions","p":383},{"i":391,"t":"Running Commands","u":"/docs/management","h":"#running-commands","p":383},{"i":393,"t":"Healthchecks","u":"/docs/management","h":"#healthchecks","p":383},{"i":395,"t":"HTTP Healthcheck Endpoint","u":"/docs/management","h":"#http-healthcheck-endpoint","p":383},{"i":397,"t":"Logs and Performance","u":"/docs/management","h":"#logs-and-performance","p":383},{"i":398,"t":"Container Logs","u":"/docs/management","h":"#container-logs","p":383},{"i":400,"t":"Container Performance","u":"/docs/management","h":"#container-performance","p":383},{"i":402,"t":"Management Apps","u":"/docs/management","h":"#management-apps","p":383},{"i":404,"t":"Advanced Logging and Monitoring","u":"/docs/management","h":"#advanced-logging-and-monitoring","p":383},{"i":406,"t":"Auto-Starting at System Boot","u":"/docs/management","h":"#auto-starting-at-system-boot","p":383},{"i":408,"t":"Updating","u":"/docs/management","h":"#updating","p":383},{"i":410,"t":"Updating Docker Container","u":"/docs/management","h":"#updating-docker-container","p":383},{"i":412,"t":"Automatic Docker Updates","u":"/docs/management","h":"#automatic-docker-updates","p":383},{"i":414,"t":"Updating Dashy from Source","u":"/docs/management","h":"#updating-dashy-from-source","p":383},{"i":416,"t":"Backing Up","u":"/docs/management","h":"#backing-up","p":383},{"i":417,"t":"Backing Up Containers","u":"/docs/management","h":"#backing-up-containers","p":383},{"i":419,"t":"Backing Up Volumes","u":"/docs/management","h":"#backing-up-volumes","p":383},{"i":421,"t":"Dashy-Specific Backup","u":"/docs/management","h":"#dashy-specific-backup","p":383},{"i":423,"t":"Scheduling","u":"/docs/management","h":"#scheduling","p":383},{"i":425,"t":"SSL Certificates","u":"/docs/management","h":"#ssl-certificates","p":383},{"i":427,"t":"Auto-SSL","u":"/docs/management","h":"#auto-ssl","p":383},{"i":429,"t":"Getting a Self-Signed SSL Certificate","u":"/docs/management","h":"#getting-a-self-signed-ssl-certificate","p":383},{"i":431,"t":"Passing a Self-Signed Certificate to Dashy","u":"/docs/management","h":"#passing-a-self-signed-certificate-to-dashy","p":383},{"i":433,"t":"Authentication","u":"/docs/management","h":"#authentication","p":383},{"i":435,"t":"Network Exposure","u":"/docs/management","h":"#network-exposure","p":383},{"i":437,"t":"Managing Containers with Docker Compose","u":"/docs/management","h":"#managing-containers-with-docker-compose","p":383},{"i":439,"t":"Passing in Environmental Variables","u":"/docs/management","h":"#passing-in-environmental-variables","p":383},{"i":441,"t":"Setting Headers","u":"/docs/management","h":"#setting-headers","p":383},{"i":443,"t":"Example Headers","u":"/docs/management","h":"#example-headers","p":383},{"i":445,"t":"Remote Access","u":"/docs/management","h":"#remote-access","p":383},{"i":447,"t":"WireGuard","u":"/docs/management","h":"#wireguard","p":383},{"i":449,"t":"Reverse SSH Tunnel","u":"/docs/management","h":"#reverse-ssh-tunnel","p":383},{"i":451,"t":"TCP Tunnel","u":"/docs/management","h":"#tcp-tunnel","p":383},{"i":453,"t":"Custom Domain","u":"/docs/management","h":"#custom-domain","p":383},{"i":455,"t":"Using DNS","u":"/docs/management","h":"#using-dns","p":383},{"i":457,"t":"Using NGINX","u":"/docs/management","h":"#using-nginx","p":383},{"i":459,"t":"Container Security","u":"/docs/management","h":"#container-security","p":383},{"i":461,"t":"Keep Docker Up-To-Date","u":"/docs/management","h":"#keep-docker-up-to-date","p":383},{"i":463,"t":"Set Resource Quotas","u":"/docs/management","h":"#set-resource-quotas","p":383},{"i":465,"t":"Don't Run as Root","u":"/docs/management","h":"#dont-run-as-root","p":383},{"i":467,"t":"Specify a User","u":"/docs/management","h":"#specify-a-user","p":383},{"i":469,"t":"Limit capabilities","u":"/docs/management","h":"#limit-capabilities","p":383},{"i":471,"t":"Prevent new Privileges being Added","u":"/docs/management","h":"#prevent-new-privileges-being-added","p":383},{"i":473,"t":"Disable Inter-Container Communication","u":"/docs/management","h":"#disable-inter-container-communication","p":383},{"i":475,"t":"Don't Expose the Docker Daemon Socket","u":"/docs/management","h":"#dont-expose-the-docker-daemon-socket","p":383},{"i":477,"t":"Use Read-Only Volumes","u":"/docs/management","h":"#use-read-only-volumes","p":383},{"i":479,"t":"Set the Logging Level","u":"/docs/management","h":"#set-the-logging-level","p":383},{"i":481,"t":"Verify Image before Pulling","u":"/docs/management","h":"#verify-image-before-pulling","p":383},{"i":483,"t":"Specify the Tag","u":"/docs/management","h":"#specify-the-tag","p":383},{"i":485,"t":"Container Security Scanning","u":"/docs/management","h":"#container-security-scanning","p":383},{"i":487,"t":"Registry Security","u":"/docs/management","h":"#registry-security","p":383},{"i":489,"t":"Security Modules","u":"/docs/management","h":"#security-modules","p":383},{"i":491,"t":"Web Server Configuration","u":"/docs/management","h":"#web-server-configuration","p":383},{"i":493,"t":"NGINX","u":"/docs/management","h":"#nginx-1","p":383},{"i":495,"t":"Apache","u":"/docs/management","h":"#apache-1","p":383},{"i":497,"t":"Caddy","u":"/docs/management","h":"#caddy-1","p":383},{"i":499,"t":"Firebase Hosting","u":"/docs/management","h":"#firebase-hosting","p":383},{"i":501,"t":"cPanel","u":"/docs/management","h":"#cpanel","p":383},{"i":503,"t":"Running a Modified Version of the App","u":"/docs/management","h":"#running-a-modified-version-of-the-app","p":383},{"i":505,"t":"Building your Own Container","u":"/docs/management","h":"#building-your-own-container","p":383},{"i":509,"t":"Setting Language","u":"/docs/multi-language-support","h":"#setting-language","p":507},{"i":511,"t":"In the UI","u":"/docs/multi-language-support","h":"#in-the-ui","p":507},{"i":513,"t":"Config File","u":"/docs/multi-language-support","h":"#config-file","p":507},{"i":515,"t":"Available Languages","u":"/docs/multi-language-support","h":"#available-languages","p":507},{"i":517,"t":"Adding a new Language","u":"/docs/multi-language-support","h":"#adding-a-new-language","p":507},{"i":519,"t":"1. Create a new Language File","u":"/docs/multi-language-support","h":"#1-create-a-new-language-file","p":507},{"i":521,"t":"2. Translate","u":"/docs/multi-language-support","h":"#2-translate","p":507},{"i":523,"t":"3. Add your file to the app","u":"/docs/multi-language-support","h":"#3-add-your-file-to-the-app","p":507},{"i":525,"t":"Checking Translation Coverage","u":"/docs/multi-language-support","h":"#checking-translation-coverage","p":507},{"i":527,"t":"Adding New Text to a Component","u":"/docs/multi-language-support","h":"#adding-new-text-to-a-component","p":507},{"i":529,"t":"1. Add Translated Text","u":"/docs/multi-language-support","h":"#1-add-translated-text","p":507},{"i":531,"t":"2. Use Text within Component","u":"/docs/multi-language-support","h":"#2-use-text-within-component","p":507},{"i":533,"t":"Basic Example","u":"/docs/multi-language-support","h":"#basic-example","p":507},{"i":536,"t":"Page Metadata","u":"/docs/pages-and-sections","h":"#page-metadata","p":535},{"i":538,"t":"Multi-Page Support","u":"/docs/pages-and-sections","h":"#multi-page-support","p":535},{"i":540,"t":"Using Local Sub-Pages","u":"/docs/pages-and-sections","h":"#using-local-sub-pages","p":535},{"i":542,"t":"Using Remote Sub-Pages","u":"/docs/pages-and-sections","h":"#using-remote-sub-pages","p":535},{"i":544,"t":"Restrictions","u":"/docs/pages-and-sections","h":"#restrictions","p":535},{"i":546,"t":"URL Structure","u":"/docs/pages-and-sections","h":"#url-structure","p":535},{"i":548,"t":"Layout","u":"/docs/pages-and-sections","h":"#layout","p":535},{"i":550,"t":"Making a section wider or taller","u":"/docs/pages-and-sections","h":"#making-a-section-wider-or-taller","p":535},{"i":552,"t":"Items inside a section","u":"/docs/pages-and-sections","h":"#items-inside-a-section","p":535},{"i":554,"t":"Sub-Items","u":"/docs/pages-and-sections","h":"#sub-items","p":535},{"i":558,"t":"Contents","u":"/docs/privacy","h":"#contents","p":556},{"i":560,"t":"Browser Storage","u":"/docs/privacy","h":"#browser-storage","p":556},{"i":562,"t":"Cookies","u":"/docs/privacy","h":"#cookies","p":556},{"i":564,"t":"Session Storage","u":"/docs/privacy","h":"#session-storage","p":556},{"i":566,"t":"Local Storage","u":"/docs/privacy","h":"#local-storage","p":556},{"i":568,"t":"Deleting Stored Data","u":"/docs/privacy","h":"#deleting-stored-data","p":556},{"i":570,"t":"External Requests","u":"/docs/privacy","h":"#external-requests","p":556},{"i":572,"t":"Icons","u":"/docs/privacy","h":"#icons","p":556},{"i":574,"t":"Themes","u":"/docs/privacy","h":"#themes","p":556},{"i":576,"t":"Status Checking","u":"/docs/privacy","h":"#status-checking","p":556},{"i":578,"t":"Update Checks","u":"/docs/privacy","h":"#update-checks","p":556},{"i":580,"t":"Cloud Backup","u":"/docs/privacy","h":"#cloud-backup","p":556},{"i":582,"t":"Web Search","u":"/docs/privacy","h":"#web-search","p":556},{"i":584,"t":"Initialization Page","u":"/docs/privacy","h":"#initialization-page","p":556},{"i":586,"t":"Anonymous Error Reporting","u":"/docs/privacy","h":"#anonymous-error-reporting","p":556},{"i":588,"t":"Widgets","u":"/docs/privacy","h":"#widgets","p":556},{"i":590,"t":"Dependencies","u":"/docs/privacy","h":"#dependencies","p":556},{"i":592,"t":"Securing your Environment","u":"/docs/privacy","h":"#securing-your-environment","p":556},{"i":594,"t":"Security Features","u":"/docs/privacy","h":"#security-features","p":556},{"i":595,"t":"Subresource Integrity","u":"/docs/privacy","h":"#subresource-integrity","p":556},{"i":597,"t":"SSL","u":"/docs/privacy","h":"#ssl","p":556},{"i":599,"t":"Authentication","u":"/docs/privacy","h":"#authentication","p":556},{"i":601,"t":"Configuration Lockdown","u":"/docs/privacy","h":"#configuration-lockdown","p":556},{"i":603,"t":"Disabling Features","u":"/docs/privacy","h":"#disabling-features","p":556},{"i":605,"t":"Threat Model","u":"/docs/privacy","h":"#threat-model","p":556},{"i":607,"t":"Intended Deployment","u":"/docs/privacy","h":"#intended-deployment","p":556},{"i":609,"t":"Trust Boundaries","u":"/docs/privacy","h":"#trust-boundaries","p":556},{"i":611,"t":"Assets","u":"/docs/privacy","h":"#assets","p":556},{"i":613,"t":"When Dashy is NOT the Right Choice","u":"/docs/privacy","h":"#when-dashy-is-not-the-right-choice","p":556},{"i":615,"t":"Known Limitations","u":"/docs/privacy","h":"#known-limitations","p":556},{"i":617,"t":"Update & Patch Policy","u":"/docs/privacy","h":"#update--patch-policy","p":556},{"i":619,"t":"Reporting a Security Issue","u":"/docs/privacy","h":"#reporting-a-security-issue","p":556},{"i":623,"t":"1. Prerequisites","u":"/docs/quick-start","h":"#1-prerequisites","p":621},{"i":625,"t":"2. Installation","u":"/docs/quick-start","h":"#2-installation","p":621},{"i":627,"t":"3. User Data Directory","u":"/docs/quick-start","h":"#3-user-data-directory","p":621},{"i":629,"t":"4. Configure","u":"/docs/quick-start","h":"#4-configure","p":621},{"i":631,"t":"5. Further Customisation","u":"/docs/quick-start","h":"#5-further-customisation","p":621},{"i":633,"t":"6. Final Note","u":"/docs/quick-start","h":"#6-final-note","p":621},{"i":635,"t":"Alternative Deployment Method 1 - From Source","u":"/docs/quick-start","h":"#alternative-deployment-method-1---from-source","p":621},{"i":637,"t":"Alternative Deployment Method 2 - Netlify","u":"/docs/quick-start","h":"#alternative-deployment-method-2---netlify","p":621},{"i":639,"t":"Alternative Deployment Method 3 - Cloud Services","u":"/docs/quick-start","h":"#alternative-deployment-method-3---cloud-services","p":621},{"i":643,"t":"Release Schedule","u":"/docs/release-workflow","h":"#release-schedule","p":641},{"i":645,"t":"Deployment Process","u":"/docs/release-workflow","h":"#deployment-process","p":641},{"i":647,"t":"Git Strategy","u":"/docs/release-workflow","h":"#git-strategy","p":641},{"i":648,"t":"Git Flow","u":"/docs/release-workflow","h":"#git-flow","p":641},{"i":650,"t":"Git Branch Naming","u":"/docs/release-workflow","h":"#git-branch-naming","p":641},{"i":652,"t":"Commit Emojis","u":"/docs/release-workflow","h":"#commit-emojis","p":641},{"i":654,"t":"PR Guidelines","u":"/docs/release-workflow","h":"#pr-guidelines","p":641},{"i":656,"t":"Automated Workflows","u":"/docs/release-workflow","h":"#automated-workflows","p":641},{"i":658,"t":"CI Checks","u":"/docs/release-workflow","h":"#ci-checks","p":641},{"i":660,"t":"Releases","u":"/docs/release-workflow","h":"#releases","p":641},{"i":662,"t":"Issue Management","u":"/docs/release-workflow","h":"#issue-management","p":641},{"i":664,"t":"Documentation","u":"/docs/release-workflow","h":"#documentation","p":641},{"i":666,"t":"Other","u":"/docs/release-workflow","h":"#other","p":641},{"i":668,"t":"Release Pipeline","u":"/docs/release-workflow","h":"#release-pipeline","p":641},{"i":670,"t":"Searching","u":"/docs/searching","h":"#searching","p":669},{"i":672,"t":"Navigating","u":"/docs/searching","h":"#navigating","p":669},{"i":674,"t":"Launching Apps","u":"/docs/searching","h":"#launching-apps","p":669},{"i":676,"t":"Tags","u":"/docs/searching","h":"#tags","p":669},{"i":678,"t":"Custom Hotkeys","u":"/docs/searching","h":"#custom-hotkeys","p":669},{"i":680,"t":"Web Search","u":"/docs/searching","h":"#web-search","p":669},{"i":682,"t":"Setting Search Engine","u":"/docs/searching","h":"#setting-search-engine","p":669},{"i":684,"t":"Using Custom Search Engine","u":"/docs/searching","h":"#using-custom-search-engine","p":669},{"i":686,"t":"Setting Opening Method","u":"/docs/searching","h":"#setting-opening-method","p":669},{"i":688,"t":"Using Bangs","u":"/docs/searching","h":"#using-bangs","p":669},{"i":690,"t":"Disabling Web Search","u":"/docs/searching","h":"#disabling-web-search","p":669},{"i":692,"t":"Opening URLs Directly","u":"/docs/searching","h":"#opening-urls-directly","p":669},{"i":694,"t":"Clearing Search","u":"/docs/searching","h":"#clearing-search","p":669},{"i":698,"t":"MNDashboard","u":"/docs/showcase","h":"#mndashboard","p":696},{"i":700,"t":"Home Lab 2.0","u":"/docs/showcase","h":"#home-lab-20","p":696},{"i":701,"t":"Dipan's Dash","u":"/docs/showcase","h":"#dipans-dash","p":696},{"i":703,"t":"Ratty222","u":"/docs/showcase","h":"#ratty222","p":696},{"i":705,"t":"Hugalafutro Dashy","u":"/docs/showcase","h":"#hugalafutro-dashy","p":696},{"i":707,"t":"NAS Home Dashboard","u":"/docs/showcase","h":"#nas-home-dashboard","p":696},{"i":709,"t":"Brewhack","u":"/docs/showcase","h":"#brewhack","p":696},{"i":711,"t":"The Dragons Lair","u":"/docs/showcase","h":"#the-dragons-lair","p":696},{"i":713,"t":"Homelab & VPS dashboard","u":"/docs/showcase","h":"#homelab--vps-dashboard","p":696},{"i":715,"t":"Raspberry PI Docker Dashboard","u":"/docs/showcase","h":"#raspberry-pi-docker-dashboard","p":696},{"i":717,"t":"First Week of Self-Hosting","u":"/docs/showcase","h":"#first-week-of-self-hosting","p":696},{"i":719,"t":"EVO Dashboard","u":"/docs/showcase","h":"#evo-dashboard","p":696},{"i":721,"t":"The Private Dashboard","u":"/docs/showcase","h":"#the-private-dashboard","p":696},{"i":723,"t":"Networking Services","u":"/docs/showcase","h":"#networking-services","p":696},{"i":725,"t":"Dashy Live","u":"/docs/showcase","h":"#dashy-live","p":696},{"i":727,"t":"System Monitor","u":"/docs/showcase","h":"#system-monitor","p":696},{"i":729,"t":"Browser Startpage","u":"/docs/showcase","h":"#browser-startpage","p":696},{"i":730,"t":"CFT Toolbox","u":"/docs/showcase","h":"#cft-toolbox","p":696},{"i":731,"t":"Bookmarks","u":"/docs/showcase","h":"#bookmarks","p":696},{"i":732,"t":"Project Management","u":"/docs/showcase","h":"#project-management","p":696},{"i":733,"t":"Dashy Example","u":"/docs/showcase","h":"#dashy-example","p":696},{"i":735,"t":"HomeLAb 3.0","u":"/docs/showcase","h":"#homelab-30","p":696},{"i":737,"t":"Ground Control","u":"/docs/showcase","h":"#ground-control","p":696},{"i":739,"t":"Croco_Grievous","u":"/docs/showcase","h":"#croco_grievous","p":696},{"i":741,"t":"Crypto Dash","u":"/docs/showcase","h":"#crypto-dash","p":696},{"i":743,"t":"Stefantigro","u":"/docs/showcase","h":"#stefantigro","p":696},{"i":745,"t":"HomeLab 3.0","u":"/docs/showcase","h":"#homelab-30-1","p":696},{"i":747,"t":"Morning Dashboard","u":"/docs/showcase","h":"#morning-dashboard","p":696},{"i":749,"t":"Yet Another Homelab","u":"/docs/showcase","h":"#yet-another-homelab","p":696},{"i":750,"t":"Submitting your Dashboard","u":"/docs/showcase","h":"#submitting-your-dashboard","p":696},{"i":751,"t":"How to Submit","u":"/docs/showcase","h":"#how-to-submit","p":696},{"i":753,"t":"What to Include","u":"/docs/showcase","h":"#what-to-include","p":696},{"i":755,"t":"Template","u":"/docs/showcase","h":"#template","p":696},{"i":759,"t":"MNDashboard","u":"/docs/showcase/","h":"#mndashboard","p":757},{"i":761,"t":"Home Lab 2.0","u":"/docs/showcase/","h":"#home-lab-20","p":757},{"i":762,"t":"Dipan's Dash","u":"/docs/showcase/","h":"#dipans-dash","p":757},{"i":764,"t":"Ratty222","u":"/docs/showcase/","h":"#ratty222","p":757},{"i":766,"t":"Hugalafutro Dashy","u":"/docs/showcase/","h":"#hugalafutro-dashy","p":757},{"i":768,"t":"NAS Home Dashboard","u":"/docs/showcase/","h":"#nas-home-dashboard","p":757},{"i":770,"t":"Brewhack","u":"/docs/showcase/","h":"#brewhack","p":757},{"i":772,"t":"The Dragons Lair","u":"/docs/showcase/","h":"#the-dragons-lair","p":757},{"i":774,"t":"Homelab & VPS dashboard","u":"/docs/showcase/","h":"#homelab--vps-dashboard","p":757},{"i":776,"t":"Raspberry PI Docker Dashboard","u":"/docs/showcase/","h":"#raspberry-pi-docker-dashboard","p":757},{"i":778,"t":"First Week of Self-Hosting","u":"/docs/showcase/","h":"#first-week-of-self-hosting","p":757},{"i":780,"t":"EVO Dashboard","u":"/docs/showcase/","h":"#evo-dashboard","p":757},{"i":782,"t":"The Private Dashboard","u":"/docs/showcase/","h":"#the-private-dashboard","p":757},{"i":784,"t":"Networking Services","u":"/docs/showcase/","h":"#networking-services","p":757},{"i":786,"t":"Dashy Live","u":"/docs/showcase/","h":"#dashy-live","p":757},{"i":788,"t":"System Monitor","u":"/docs/showcase/","h":"#system-monitor","p":757},{"i":790,"t":"Browser Startpage","u":"/docs/showcase/","h":"#browser-startpage","p":757},{"i":791,"t":"CFT Toolbox","u":"/docs/showcase/","h":"#cft-toolbox","p":757},{"i":792,"t":"Bookmarks","u":"/docs/showcase/","h":"#bookmarks","p":757},{"i":793,"t":"Project Management","u":"/docs/showcase/","h":"#project-management","p":757},{"i":794,"t":"Dashy Example","u":"/docs/showcase/","h":"#dashy-example","p":757},{"i":796,"t":"HomeLAb 3.0","u":"/docs/showcase/","h":"#homelab-30","p":757},{"i":798,"t":"Ground Control","u":"/docs/showcase/","h":"#ground-control","p":757},{"i":800,"t":"Croco_Grievous","u":"/docs/showcase/","h":"#croco_grievous","p":757},{"i":802,"t":"Crypto Dash","u":"/docs/showcase/","h":"#crypto-dash","p":757},{"i":804,"t":"Stefantigro","u":"/docs/showcase/","h":"#stefantigro","p":757},{"i":806,"t":"HomeLab 3.0","u":"/docs/showcase/","h":"#homelab-30-1","p":757},{"i":808,"t":"Morning Dashboard","u":"/docs/showcase/","h":"#morning-dashboard","p":757},{"i":810,"t":"Yet Another Homelab","u":"/docs/showcase/","h":"#yet-another-homelab","p":757},{"i":811,"t":"Submitting your Dashboard","u":"/docs/showcase/","h":"#submitting-your-dashboard","p":757},{"i":812,"t":"How to Submit","u":"/docs/showcase/","h":"#how-to-submit","p":757},{"i":814,"t":"What to Include","u":"/docs/showcase/","h":"#what-to-include","p":757},{"i":816,"t":"Template","u":"/docs/showcase/","h":"#template","p":757},{"i":820,"t":"Enabling Status Indicators","u":"/docs/status-indicators","h":"#enabling-status-indicators","p":818},{"i":822,"t":"Continuous Checking","u":"/docs/status-indicators","h":"#continuous-checking","p":818},{"i":824,"t":"Using a Different Endpoint","u":"/docs/status-indicators","h":"#using-a-different-endpoint","p":818},{"i":826,"t":"Setting Custom Headers","u":"/docs/status-indicators","h":"#setting-custom-headers","p":818},{"i":828,"t":"Disabling Security","u":"/docs/status-indicators","h":"#disabling-security","p":818},{"i":830,"t":"Allowing Alternative Status Codes","u":"/docs/status-indicators","h":"#allowing-alternative-status-codes","p":818},{"i":832,"t":"Troubleshooting Failing Status Checks","u":"/docs/status-indicators","h":"#troubleshooting-failing-status-checks","p":818},{"i":834,"t":"Color-Blind Accessibility","u":"/docs/status-indicators","h":"#color-blind-accessibility","p":818},{"i":836,"t":"How it Works","u":"/docs/status-indicators","h":"#how-it-works","p":818},{"i":840,"t":"How Theme-Switching Works","u":"/docs/theming","h":"#how-theme-switching-works","p":838},{"i":842,"t":"Adding Your Own Theme","u":"/docs/theming","h":"#adding-your-own-theme","p":838},{"i":843,"t":"Option 1: Colors-only, in conf.yml","u":"/docs/theming","h":"#option-1-colors-only-in-confyml","p":838},{"i":845,"t":"Option 2: Full CSS file","u":"/docs/theming","h":"#option-2-full-css-file","p":838},{"i":847,"t":"Modifying Theme Colors","u":"/docs/theming","h":"#modifying-theme-colors","p":838},{"i":849,"t":"Setting Custom CSS in the UI","u":"/docs/theming","h":"#setting-custom-css-in-the-ui","p":838},{"i":851,"t":"Page-Specific Styles","u":"/docs/theming","h":"#page-specific-styles","p":838},{"i":853,"t":"Loading External Stylesheets","u":"/docs/theming","h":"#loading-external-stylesheets","p":838},{"i":855,"t":"Hard-Coding Section or Item Colors","u":"/docs/theming","h":"#hard-coding-section-or-item-colors","p":838},{"i":857,"t":"Typography","u":"/docs/theming","h":"#typography","p":838},{"i":859,"t":"Changing browser address bar / tab color","u":"/docs/theming","h":"#changing-browser-address-bar--tab-color","p":838},{"i":861,"t":"CSS Variables","u":"/docs/theming","h":"#css-variables","p":838},{"i":863,"t":"Top-Level Variables","u":"/docs/theming","h":"#top-level-variables","p":838},{"i":865,"t":"Targeted Color Variables","u":"/docs/theming","h":"#targeted-color-variables","p":838},{"i":867,"t":"Non-Color Variables","u":"/docs/theming","h":"#non-color-variables","p":838},{"i":869,"t":"Action Colors","u":"/docs/theming","h":"#action-colors","p":838},{"i":873,"t":"Contents","u":"/docs/troubleshooting","h":"#contents","p":871},{"i":875,"t":"Config not saving","u":"/docs/troubleshooting","h":"#config-not-saving","p":871},{"i":877,"t":"Permission denied or read-only filesystem (EACCES, EROFS)","u":"/docs/troubleshooting","h":"#permission-denied-or-read-only-filesystem-eacces-erofs","p":871},{"i":879,"t":"Kubernetes ConfigMap mount is read-only","u":"/docs/troubleshooting","h":"#kubernetes-configmap-mount-is-read-only","p":871},{"i":881,"t":"SELinux or AppArmor blocks the write","u":"/docs/troubleshooting","h":"#selinux-or-apparmor-blocks-the-write","p":871},{"i":883,"t":"Backup step fails so save aborts","u":"/docs/troubleshooting","h":"#backup-step-fails-so-save-aborts","p":871},{"i":885,"t":"Save button is missing or returns 403 Forbidden","u":"/docs/troubleshooting","h":"#save-button-is-missing-or-returns-403-forbidden","p":871},{"i":887,"t":"Save unavailable on Vercel, Netlify or other static hosts","u":"/docs/troubleshooting","h":"#save-unavailable-on-vercel-netlify-or-other-static-hosts","p":871},{"i":889,"t":"/config-manager/save returns 404 or HTML","u":"/docs/troubleshooting","h":"#config-managersave-returns-404-or-html","p":871},{"i":891,"t":"\"Invalid filename\" when saving a sub-page","u":"/docs/troubleshooting","h":"#invalid-filename-when-saving-a-sub-page","p":871},{"i":893,"t":"\"Cannot save to an external URL\"","u":"/docs/troubleshooting","h":"#cannot-save-to-an-external-url","p":871},{"i":895,"t":"Saved successfully but the UI shows the old config","u":"/docs/troubleshooting","h":"#saved-successfully-but-the-ui-shows-the-old-config","p":871},{"i":897,"t":"Container crashes or restart loop after saving (3.1.0 and 3.1.1 only)","u":"/docs/troubleshooting","h":"#container-crashes-or-restart-loop-after-saving-310-and-311-only","p":871},{"i":899,"t":"Intentionally read-only mode","u":"/docs/troubleshooting","h":"#intentionally-read-only-mode","p":871},{"i":901,"t":"Refused to Connect in Modal or Workspace View","u":"/docs/troubleshooting","h":"#refused-to-connect-in-modal-or-workspace-view","p":871},{"i":903,"t":"NGINX","u":"/docs/troubleshooting","h":"#nginx","p":871},{"i":905,"t":"Caddy","u":"/docs/troubleshooting","h":"#caddy","p":871},{"i":907,"t":"Apache","u":"/docs/troubleshooting","h":"#apache","p":871},{"i":909,"t":"LightHttpd","u":"/docs/troubleshooting","h":"#lighthttpd","p":871},{"i":911,"t":"404 / Routing issues","u":"/docs/troubleshooting","h":"#404--routing-issues","p":871},{"i":912,"t":"404 On Static Hosting","u":"/docs/troubleshooting","h":"#404-on-static-hosting","p":871},{"i":914,"t":"404 after Launch from Mobile Home Screen","u":"/docs/troubleshooting","h":"#404-after-launch-from-mobile-home-screen","p":871},{"i":916,"t":"404 On Multi-Page Apps","u":"/docs/troubleshooting","h":"#404-on-multi-page-apps","p":871},{"i":918,"t":"Dashy hosted at a sub-path (e.g. example.com/dashy)","u":"/docs/troubleshooting","h":"#dashy-hosted-at-a-sub-path-eg-examplecomdashy","p":871},{"i":920,"t":"Sub-pages","u":"/docs/troubleshooting","h":"#sub-pages","p":871},{"i":921,"t":"Sub-page shows \"Unable to find config for ...\"","u":"/docs/troubleshooting","h":"#sub-page-shows-unable-to-find-config-for-","p":871},{"i":923,"t":"Sub-page missing from nav, or won't open when clicked","u":"/docs/troubleshooting","h":"#sub-page-missing-from-nav-or-wont-open-when-clicked","p":871},{"i":925,"t":"Sub-page ignores its theme, layout or appConfig","u":"/docs/troubleshooting","h":"#sub-page-ignores-its-theme-layout-or-appconfig","p":871},{"i":927,"t":"Sub-config files return 404","u":"/docs/troubleshooting","h":"#sub-config-files-return-404","p":871},{"i":929,"t":"Remote Config Not Loading","u":"/docs/troubleshooting","h":"#remote-config-not-loading","p":871},{"i":931,"t":"Build & memory errors","u":"/docs/troubleshooting","h":"#build--memory-errors","p":871},{"i":932,"t":"Yarn Error","u":"/docs/troubleshooting","h":"#yarn-error","p":871},{"i":934,"t":"yarn build fails inside the container","u":"/docs/troubleshooting","h":"#yarn-build-fails-inside-the-container","p":871},{"i":936,"t":"High CPU or RAM Usage on Startup","u":"/docs/troubleshooting","h":"#high-cpu-or-ram-usage-on-startup","p":871},{"i":938,"t":"Ineffective mark-compacts near heap limit Allocation failed","u":"/docs/troubleshooting","h":"#ineffective-mark-compacts-near-heap-limit-allocation-failed","p":871},{"i":940,"t":"Command failed with signal \"SIGKILL\"","u":"/docs/troubleshooting","h":"#command-failed-with-signal-sigkill","p":871},{"i":942,"t":"Node Sass does not yet support your current environment","u":"/docs/troubleshooting","h":"#node-sass-does-not-yet-support-your-current-environment","p":871},{"i":944,"t":"Unreachable Code Error","u":"/docs/troubleshooting","h":"#unreachable-code-error","p":871},{"i":946,"t":"Error: Cannot find module './_baseValues'","u":"/docs/troubleshooting","h":"#error-cannot-find-module-_basevalues","p":871},{"i":948,"t":"Auth & OIDC","u":"/docs/troubleshooting","h":"#auth--oidc","p":871},{"i":949,"t":"Auth Validation Error: \"should be object\"","u":"/docs/troubleshooting","h":"#auth-validation-error-should-be-object","p":871},{"i":951,"t":"Keycloak Redirect Error","u":"/docs/troubleshooting","h":"#keycloak-redirect-error","p":871},{"i":953,"t":"OIDC or Keycloak failure on numeric client IDs","u":"/docs/troubleshooting","h":"#oidc-or-keycloak-failure-on-numeric-client-ids","p":871},{"i":955,"t":"Redirect loop after login","u":"/docs/troubleshooting","h":"#redirect-loop-after-login","p":871},{"i":957,"t":"invalid_redirect_uri","u":"/docs/troubleshooting","h":"#invalid_redirect_uri","p":871},{"i":959,"t":"Login works in the browser but the dashboard refuses to save anything (403)","u":"/docs/troubleshooting","h":"#login-works-in-the-browser-but-the-dashboard-refuses-to-save-anything-403","p":871},{"i":961,"t":"Logged in but no admin controls","u":"/docs/troubleshooting","h":"#logged-in-but-no-admin-controls","p":871},{"i":963,"t":"Login works but Dashy errors on the callback with \"OIDC signinCallback returned no user\"","u":"/docs/troubleshooting","h":"#login-works-but-dashy-errors-on-the-callback-with-oidc-signincallback-returned-no-user","p":871},{"i":965,"t":"Sign-out leaves you stuck on Authentik","u":"/docs/troubleshooting","h":"#sign-out-leaves-you-stuck-on-authentik","p":871},{"i":967,"t":"Untrusted certificate from Authentik","u":"/docs/troubleshooting","h":"#untrusted-certificate-from-authentik","p":871},{"i":969,"t":"Numeric client_id getting truncated","u":"/docs/troubleshooting","h":"#numeric-client_id-getting-truncated","p":871},{"i":971,"t":"Docker & image issues","u":"/docs/troubleshooting","h":"#docker--image-issues","p":871},{"i":972,"t":"App Not Starting After Update to 2.0.4","u":"/docs/troubleshooting","h":"#app-not-starting-after-update-to-204","p":871},{"i":974,"t":"Mount Type Mismatch","u":"/docs/troubleshooting","h":"#mount-type-mismatch","p":871},{"i":976,"t":"DockerHub toomanyrequests","u":"/docs/troubleshooting","h":"#dockerhub-toomanyrequests","p":871},{"i":978,"t":"Old image tags fail to pull","u":"/docs/troubleshooting","h":"#old-image-tags-fail-to-pull","p":871},{"i":980,"t":"Healthcheck Failing in Docker","u":"/docs/troubleshooting","h":"#healthcheck-failing-in-docker","p":871},{"i":982,"t":"Docker Login Fails on Ubuntu","u":"/docs/troubleshooting","h":"#docker-login-fails-on-ubuntu","p":871},{"i":984,"t":"Styles and Assets not Updating","u":"/docs/troubleshooting","h":"#styles-and-assets-not-updating","p":871},{"i":986,"t":"Config Validation Errors","u":"/docs/troubleshooting","h":"#config-validation-errors","p":871},{"i":988,"t":"Invalid Host Header while running through ngrok","u":"/docs/troubleshooting","h":"#invalid-host-header-while-running-through-ngrok","p":871},{"i":990,"t":"Warnings in the Console during deploy","u":"/docs/troubleshooting","h":"#warnings-in-the-console-during-deploy","p":871},{"i":992,"t":"Status Checks Failing","u":"/docs/troubleshooting","h":"#status-checks-failing","p":871},{"i":994,"t":"Widgets","u":"/docs/troubleshooting","h":"#widgets","p":871},{"i":995,"t":"Widget Errors","u":"/docs/troubleshooting","h":"#widget-errors","p":871},{"i":997,"t":"Widget CORS Errors","u":"/docs/troubleshooting","h":"#widget-cors-errors","p":871},{"i":999,"t":"CORS Proxy connect ECONNREFUSED ... or getaddrinfo ENOTFOUND ...","u":"/docs/troubleshooting","h":"#cors-proxy-connect-econnrefused--or-getaddrinfo-enotfound-","p":871},{"i":1001,"t":"CORS Proxy Target-URL host '...' is blocked / must use http:// or https://","u":"/docs/troubleshooting","h":"#cors-proxy-target-url-host--is-blocked--must-use-http-or-https","p":871},{"i":1003,"t":"Widget Shows Error Incorrectly","u":"/docs/troubleshooting","h":"#widget-shows-error-incorrectly","p":871},{"i":1005,"t":"Weather Forecast Widget 401","u":"/docs/troubleshooting","h":"#weather-forecast-widget-401","p":871},{"i":1007,"t":"Widget Displaying Inaccurate Data","u":"/docs/troubleshooting","h":"#widget-displaying-inaccurate-data","p":871},{"i":1009,"t":"Public IP Widget not working for ipinfo or ipquery providers","u":"/docs/troubleshooting","h":"#public-ip-widget-not-working-for-ipinfo-or-ipquery-providers","p":871},{"i":1011,"t":"Font Awesome Icons not Displaying","u":"/docs/troubleshooting","h":"#font-awesome-icons-not-displaying","p":871},{"i":1013,"t":"Copy to Clipboard not Working","u":"/docs/troubleshooting","h":"#copy-to-clipboard-not-working","p":871},{"i":1015,"t":"How-To / Reference","u":"/docs/troubleshooting","h":"#how-to--reference","p":871},{"i":1016,"t":"How to Reset Local Settings","u":"/docs/troubleshooting","h":"#how-to-reset-local-settings","p":871},{"i":1018,"t":"How to make a bug report","u":"/docs/troubleshooting","h":"#how-to-make-a-bug-report","p":871},{"i":1020,"t":"How-To Open Browser Console","u":"/docs/troubleshooting","h":"#how-to-open-browser-console","p":871},{"i":1022,"t":"Git Contributions not Displaying","u":"/docs/troubleshooting","h":"#git-contributions-not-displaying","p":871},{"i":1026,"t":"Contents","u":"/docs/widgets","h":"#contents","p":1024},{"i":1028,"t":"General Widgets","u":"/docs/widgets","h":"#general-widgets","p":1024},{"i":1029,"t":"Clock","u":"/docs/widgets","h":"#clock","p":1024},{"i":1031,"t":"Weather","u":"/docs/widgets","h":"#weather","p":1024},{"i":1033,"t":"Weather Forecast","u":"/docs/widgets","h":"#weather-forecast","p":1024},{"i":1035,"t":"RSS Feed","u":"/docs/widgets","h":"#rss-feed","p":1024},{"i":1037,"t":"Image","u":"/docs/widgets","h":"#image","p":1024},{"i":1039,"t":"Public IP","u":"/docs/widgets","h":"#public-ip","p":1024},{"i":1041,"t":"IP Blacklist","u":"/docs/widgets","h":"#ip-blacklist","p":1024},{"i":1043,"t":"Domain Monitor","u":"/docs/widgets","h":"#domain-monitor","p":1024},{"i":1045,"t":"Crypto Watch List","u":"/docs/widgets","h":"#crypto-watch-list","p":1024},{"i":1047,"t":"Crypto Token Price History","u":"/docs/widgets","h":"#crypto-token-price-history","p":1024},{"i":1049,"t":"Wallet Balance","u":"/docs/widgets","h":"#wallet-balance","p":1024},{"i":1051,"t":"Code Stats","u":"/docs/widgets","h":"#code-stats","p":1024},{"i":1053,"t":"Mullvad Status","u":"/docs/widgets","h":"#mullvad-status","p":1024},{"i":1055,"t":"addy.io","u":"/docs/widgets","h":"#addyio","p":1024},{"i":1057,"t":"Vulnerability Feed","u":"/docs/widgets","h":"#vulnerability-feed","p":1024},{"i":1059,"t":"Exchange Rates","u":"/docs/widgets","h":"#exchange-rates","p":1024},{"i":1061,"t":"Public Holidays","u":"/docs/widgets","h":"#public-holidays","p":1024},{"i":1063,"t":"Covid-19 Status","u":"/docs/widgets","h":"#covid-19-status","p":1024},{"i":1065,"t":"Sports Scores","u":"/docs/widgets","h":"#sports-scores","p":1024},{"i":1067,"t":"News Headlines","u":"/docs/widgets","h":"#news-headlines","p":1024},{"i":1069,"t":"TFL Status","u":"/docs/widgets","h":"#tfl-status","p":1024},{"i":1071,"t":"Stock Price History","u":"/docs/widgets","h":"#stock-price-history","p":1024},{"i":1073,"t":"ETH Gas Prices","u":"/docs/widgets","h":"#eth-gas-prices","p":1024},{"i":1075,"t":"Joke","u":"/docs/widgets","h":"#joke","p":1024},{"i":1077,"t":"Chuck Norris quotes","u":"/docs/widgets","h":"#chuck-norris-quotes","p":1024},{"i":1079,"t":"XKCD Comics","u":"/docs/widgets","h":"#xkcd-comics","p":1024},{"i":1081,"t":"Flight Data","u":"/docs/widgets","h":"#flight-data","p":1024},{"i":1083,"t":"Astronomy Picture of the Day","u":"/docs/widgets","h":"#astronomy-picture-of-the-day","p":1024},{"i":1085,"t":"GitHub Trending","u":"/docs/widgets","h":"#github-trending","p":1024},{"i":1087,"t":"GitHub Profile Stats","u":"/docs/widgets","h":"#github-profile-stats","p":1024},{"i":1089,"t":"HealthChecks Status","u":"/docs/widgets","h":"#healthchecks-status","p":1024},{"i":1091,"t":"Hackernews Trending","u":"/docs/widgets","h":"#hackernews-trending","p":1024},{"i":1093,"t":"MVG Departure","u":"/docs/widgets","h":"#mvg-departure","p":1024},{"i":1095,"t":"MVG Connection","u":"/docs/widgets","h":"#mvg-connection","p":1024},{"i":1097,"t":"Custom List","u":"/docs/widgets","h":"#custom-list","p":1024},{"i":1099,"t":"Custom search","u":"/docs/widgets","h":"#custom-search","p":1024},{"i":1101,"t":"RescueTime Overview","u":"/docs/widgets","h":"#rescuetime-overview","p":1024},{"i":1103,"t":"Minecraft Server","u":"/docs/widgets","h":"#minecraft-server","p":1024},{"i":1105,"t":"Self-Hosted Services Widgets","u":"/docs/widgets","h":"#self-hosted-services-widgets","p":1024},{"i":1106,"t":"System Info","u":"/docs/widgets","h":"#system-info","p":1024},{"i":1108,"t":"Cron Monitoring (Health Checks)","u":"/docs/widgets","h":"#cron-monitoring-health-checks","p":1024},{"i":1110,"t":"CPU History (NetData)","u":"/docs/widgets","h":"#cpu-history-netdata","p":1024},{"i":1112,"t":"Memory History (NetData)","u":"/docs/widgets","h":"#memory-history-netdata","p":1024},{"i":1114,"t":"Load History (NetData)","u":"/docs/widgets","h":"#load-history-netdata","p":1024},{"i":1116,"t":"Pi-Hole Stats","u":"/docs/widgets","h":"#pi-hole-stats","p":1024},{"i":1118,"t":"Pi-Hole Stats v6","u":"/docs/widgets","h":"#pi-hole-stats-v6","p":1024},{"i":1120,"t":"Pi-Hole Queries","u":"/docs/widgets","h":"#pi-hole-queries","p":1024},{"i":1122,"t":"Pi-Hole Queries v6","u":"/docs/widgets","h":"#pi-hole-queries-v6","p":1024},{"i":1124,"t":"Pi-Hole Recent Traffic","u":"/docs/widgets","h":"#pi-hole-recent-traffic","p":1024},{"i":1126,"t":"Pi-Hole Recent Traffic v6","u":"/docs/widgets","h":"#pi-hole-recent-traffic-v6","p":1024},{"i":1128,"t":"Stat Ping Statuses","u":"/docs/widgets","h":"#stat-ping-statuses","p":1024},{"i":1130,"t":"Synology Download Station","u":"/docs/widgets","h":"#synology-download-station","p":1024},{"i":1132,"t":"AdGuard Home Block Stats","u":"/docs/widgets","h":"#adguard-home-block-stats","p":1024},{"i":1134,"t":"AdGuard Home Filters","u":"/docs/widgets","h":"#adguard-home-filters","p":1024},{"i":1136,"t":"AdGuard Home DNS Info","u":"/docs/widgets","h":"#adguard-home-dns-info","p":1024},{"i":1138,"t":"AdGuard Home Top Domains","u":"/docs/widgets","h":"#adguard-home-top-domains","p":1024},{"i":1140,"t":"Nextcloud User","u":"/docs/widgets","h":"#nextcloud-user","p":1024},{"i":1142,"t":"Nextcloud User Statuses","u":"/docs/widgets","h":"#nextcloud-user-statuses","p":1024},{"i":1144,"t":"Nextcloud Notifications","u":"/docs/widgets","h":"#nextcloud-notifications","p":1024},{"i":1146,"t":"Nextcloud System","u":"/docs/widgets","h":"#nextcloud-system","p":1024},{"i":1148,"t":"Nextcloud Stats","u":"/docs/widgets","h":"#nextcloud-stats","p":1024},{"i":1150,"t":"Nextcloud PHP OPcache Stats","u":"/docs/widgets","h":"#nextcloud-php-opcache-stats","p":1024},{"i":1152,"t":"ntfy stream","u":"/docs/widgets","h":"#ntfy-stream","p":1024},{"i":1154,"t":"Proxmox lists","u":"/docs/widgets","h":"#proxmox-lists","p":1024},{"i":1156,"t":"Sabnzbd","u":"/docs/widgets","h":"#sabnzbd","p":1024},{"i":1158,"t":"Gluetun VPN Info","u":"/docs/widgets","h":"#gluetun-vpn-info","p":1024},{"i":1160,"t":"Drone CI Builds","u":"/docs/widgets","h":"#drone-ci-builds","p":1024},{"i":1162,"t":"Filebrowser","u":"/docs/widgets","h":"#filebrowser","p":1024},{"i":1164,"t":"Linkding","u":"/docs/widgets","h":"#linkding","p":1024},{"i":1166,"t":"Uptime Kuma","u":"/docs/widgets","h":"#uptime-kuma","p":1024},{"i":1168,"t":"Uptime Kuma Status Page","u":"/docs/widgets","h":"#uptime-kuma-status-page","p":1024},{"i":1170,"t":"Tactical RMM","u":"/docs/widgets","h":"#tactical-rmm","p":1024},{"i":1172,"t":"System Resource Monitoring","u":"/docs/widgets","h":"#system-resource-monitoring","p":1024},{"i":1173,"t":"Glances","u":"/docs/widgets","h":"#glances","p":1024},{"i":1175,"t":"Current CPU Usage","u":"/docs/widgets","h":"#current-cpu-usage","p":1024},{"i":1177,"t":"Current CPU Usage Speedometer","u":"/docs/widgets","h":"#current-cpu-usage-speedometer","p":1024},{"i":1179,"t":"CPU Usage Per Core","u":"/docs/widgets","h":"#cpu-usage-per-core","p":1024},{"i":1181,"t":"CPU Usage History","u":"/docs/widgets","h":"#cpu-usage-history","p":1024},{"i":1183,"t":"Current Memory Usage","u":"/docs/widgets","h":"#current-memory-usage","p":1024},{"i":1185,"t":"Current Memory Usage Speedometer","u":"/docs/widgets","h":"#current-memory-usage-speedometer","p":1024},{"i":1187,"t":"Memory Usage History","u":"/docs/widgets","h":"#memory-usage-history","p":1024},{"i":1189,"t":"Disk Space","u":"/docs/widgets","h":"#disk-space","p":1024},{"i":1191,"t":"Disk IO","u":"/docs/widgets","h":"#disk-io","p":1024},{"i":1193,"t":"System Load","u":"/docs/widgets","h":"#system-load","p":1024},{"i":1195,"t":"Uptime","u":"/docs/widgets","h":"#uptime","p":1024},{"i":1197,"t":"System Load History","u":"/docs/widgets","h":"#system-load-history","p":1024},{"i":1199,"t":"Network Interfaces","u":"/docs/widgets","h":"#network-interfaces","p":1024},{"i":1201,"t":"Network Traffic","u":"/docs/widgets","h":"#network-traffic","p":1024},{"i":1203,"t":"Resource Usage Alerts","u":"/docs/widgets","h":"#resource-usage-alerts","p":1024},{"i":1205,"t":"IP Address","u":"/docs/widgets","h":"#ip-address","p":1024},{"i":1207,"t":"CPU Temp","u":"/docs/widgets","h":"#cpu-temp","p":1024},{"i":1209,"t":"Compact Metrics","u":"/docs/widgets","h":"#compact-metrics","p":1024},{"i":1211,"t":"Dynamic Widgets","u":"/docs/widgets","h":"#dynamic-widgets","p":1024},{"i":1212,"t":"Iframe Widget","u":"/docs/widgets","h":"#iframe-widget","p":1024},{"i":1214,"t":"HTML Embedded Widget","u":"/docs/widgets","h":"#html-embedded-widget","p":1024},{"i":1216,"t":"API Response","u":"/docs/widgets","h":"#api-response","p":1024},{"i":1218,"t":"Prometheus Data","u":"/docs/widgets","h":"#prometheus-data","p":1024},{"i":1220,"t":"Data Feed","u":"/docs/widgets","h":"#data-feed","p":1024},{"i":1222,"t":"Usage & Customizations","u":"/docs/widgets","h":"#usage--customizations","p":1024},{"i":1223,"t":"Widget Usage Guide","u":"/docs/widgets","h":"#widget-usage-guide","p":1024},{"i":1225,"t":"Continuous Updates","u":"/docs/widgets","h":"#continuous-updates","p":1024},{"i":1227,"t":"Proxying Requests","u":"/docs/widgets","h":"#proxying-requests","p":1024},{"i":1229,"t":"Handling Secrets","u":"/docs/widgets","h":"#handling-secrets","p":1024},{"i":1231,"t":"Setting Timeout","u":"/docs/widgets","h":"#setting-timeout","p":1024},{"i":1233,"t":"Adding Labels","u":"/docs/widgets","h":"#adding-labels","p":1024},{"i":1235,"t":"Ignoring Errors","u":"/docs/widgets","h":"#ignoring-errors","p":1024},{"i":1237,"t":"Widget Styling","u":"/docs/widgets","h":"#widget-styling","p":1024},{"i":1239,"t":"Customizing Charts","u":"/docs/widgets","h":"#customizing-charts","p":1024},{"i":1241,"t":"Language Translations","u":"/docs/widgets","h":"#language-translations","p":1024},{"i":1243,"t":"Widget UI Options","u":"/docs/widgets","h":"#widget-ui-options","p":1024},{"i":1245,"t":"Build your own Widget","u":"/docs/widgets","h":"#build-your-own-widget","p":1024},{"i":1247,"t":"Requesting a Widget","u":"/docs/widgets","h":"#requesting-a-widget","p":1024},{"i":1249,"t":"Troubleshooting Widget Errors","u":"/docs/widgets","h":"#troubleshooting-widget-errors","p":1024},{"i":1251,"t":"Raising an Issue","u":"/docs/widgets","h":"#raising-an-issue","p":1024}],"index":{"version":"2.3.9","fields":["t"],"fieldVectors":[["t/3",[0,6.931]],["t/5",[1,7.381]],["t/7",[2,7.381]],["t/9",[0,5.726,3,6.661]],["t/11",[4,5.227,5,4.747]],["t/15",[6,6.661,7,5.043]],["t/17",[8,3.681,9,4.296,10,4.296]],["t/19",[11,6.661,12,6.097]],["t/21",[13,5.043,14,6.097]],["t/23",[15,4.641,16,5.674,17,4.453]],["t/25",[17,5.227,18,6.661]],["t/27",[19,6.596]],["t/29",[12,4.524,20,2.93,21,3.878,22,3.522]],["t/31",[7,3.741,23,3.431,24,4.042,25,3.878]],["t/33",[26,5.464]],["t/35",[7,5.043,24,5.448]],["t/37",[20,2.595,27,2.683,28,2.9,29,3.21,30,4.377]],["t/39",[20,3.364,31,4.878,32,5.674]],["t/41",[33,5.913]],["t/43",[33,4.161,34,3.759,35,3.759]],["t/45",[29,3.624,33,3.624,36,3.206,37,4.942]],["t/47",[15,3.213,27,2.408,28,2.602,33,2.881,38,2.881,39,2.498]],["t/49",[40,3.763,41,3.119,42,4.377,43,4.377,44,2.555]],["t/51",[45,5.448,46,5.043]],["t/53",[47,3.522,48,4.524,49,4.524,50,3.349]],["t/55",[33,4.885,51,5.726]],["t/57",[10,5.043,46,5.043]],["t/59",[25,6.328]],["t/61",[50,5.464]],["t/63",[52,6.328]],["t/65",[53,6.596]],["t/67",[47,3.522,48,4.524,49,4.524,50,3.349]],["t/69",[54,6.931]],["t/71",[5,4.043,10,4.296,55,4.453]],["t/73",[7,4.296,56,5.194,57,4.641]],["t/75",[58,5.674,59,5.194,60,4.878]],["t/77",[61,7.381]],["t/79",[17,4.453,62,4.453,63,5.674]],["t/81",[10,4.296,47,4.043,64,4.453]],["t/83",[65,4.942,66,2.93,67,4.942,68,4.042]],["t/85",[68,4.641,69,4.296,70,3.681]],["t/89",[50,5.464]],["t/91",[71,5.043,72,5.043]],["t/93",[72,5.043,73,6.661]],["t/95",[26,3.845,66,3.364,74,5.674]],["t/97",[20,3.364,75,5.674,76,5.194]],["t/99",[47,3.522,70,3.206,72,3.741,77,3.741]],["t/100",[78,6.097,79,5.448]],["t/102",[80,6.661,81,6.661]],["t/104",[82,6.661,83,6.661]],["t/106",[84,6.596]],["t/108",[85,7.381]],["t/112",[27,3.03,86,4.942,87,4.942,88,4.524]],["t/114",[89,8.063]],["t/116",[90,6.328]],["t/118",[91,4.878,92,4.878,93,5.674]],["t/120",[94,8.063]],["t/122",[44,3.888,95,6.661]],["t/124",[44,3.888,96,4.015]],["t/126",[44,3.888,97,6.097]],["t/128",[44,3.888,98,6.661]],["t/130",[44,3.888,99,6.661]],["t/132",[44,3.888,100,6.661]],["t/134",[44,3.888,101,6.661]],["t/136",[44,3.888,102,6.661]],["t/138",[44,3.888,103,6.661]],["t/140",[44,3.888,104,6.661]],["t/142",[105,6.596]],["t/144",[106,8.063]],["t/146",[44,3.888,107,6.661]],["t/148",[44,3.888,108,6.661]],["t/150",[44,3.888,109,6.661]],["t/152",[110,6.661,111,6.661]],["t/154",[112,4.942,113,4.942,114,4.942,115,4.942]],["t/156",[52,6.328]],["t/157",[27,3.03,88,4.524,116,4.524,117,3.878]],["t/159",[118,8.063]],["t/161",[5,4.043,27,3.479,119,3.845]],["t/163",[120,6.097,121,6.097]],["t/165",[122,6.328]],["t/169",[36,3.206,123,4.942,124,4.942,125,4.942]],["t/171",[126,6.661,127,4.015]],["t/173",[128,4.878,129,5.194,130,5.674]],["t/175",[20,2.329,129,3.596,131,3.928,132,3.928,133,3.596,134,3.083]],["t/177",[15,4.042,135,4.524,136,4.248,137,4.042]],["t/179",[41,4.747,138,4.885]],["t/181",[139,5.227,140,5.726]],["t/183",[141,6.661,142,6.097]],["t/185",[136,5.726,143,6.097]],["t/187",[144,6.661,145,6.661]],["t/189",[146,4.042,147,4.248,148,4.942,149,4.942]],["t/191",[150,6.661,151,6.661]],["t/193",[152,4.524,153,4.942,154,4.524,155,4.942]],["t/195",[156,6.661,157,6.661]],["t/197",[158,7.381]],["t/199",[152,4.524,159,4.942,160,4.942,161,4.942]],["t/202",[162,8.063]],["t/203",[158,7.381]],["t/204",[163,6.661,164,6.661]],["t/205",[165,6.596]],["t/207",[166,7.381]],["t/209",[167,8.063]],["t/211",[168,6.661,169,5.448]],["t/213",[]],["t/217",[78,6.097,79,5.448]],["t/219",[5,4.747,35,4.413]],["t/221",[35,4.413,170,4.157]],["t/223",[20,3.364,170,3.541,171,5.194]],["t/225",[172,8.063]],["t/227",[173,8.063]],["t/229",[174,7.381]],["t/231",[175,8.063]],["t/233",[47,4.043,176,3.759,177,5.674]],["t/235",[178,6.097,179,5.726]],["t/237",[180,4.885,181,5.726]],["t/239",[35,3.759,69,4.296,182,4.453]],["t/241",[183,6.328]],["t/243",[184,7.381]],["t/245",[185,8.063]],["t/247",[96,4.015,186,6.661]],["t/249",[170,4.157,187,6.661]],["t/251",[70,4.321,188,6.097]],["t/253",[189,7.381]],["t/254",[189,6.097,190,4.624]],["t/256",[170,5.032]],["t/258",[191,6.661,192,6.661]],["t/260",[35,3.274,66,2.93,69,3.741,188,4.524]],["t/262",[193,4.885,194,5.726]],["t/266",[8,3.206,9,3.741,21,3.878,195,4.524]],["t/267",[196,6.931]],["t/269",[197,5.043,198,5.448]],["t/271",[84,5.448,198,5.448]],["t/273",[22,4.747,199,5.726]],["t/275",[21,5.227,200,6.097]],["t/277",[201,4.885,202,6.097]],["t/278",[201,4.885,203,6.097]],["t/280",[201,4.161,204,5.194,205,4.641]],["t/282",[206,6.097,207,5.726]],["t/284",[140,5.726,208,6.097]],["t/286",[209,5.227,210,6.661]],["t/288",[211,4.747,212,5.448]],["t/289",[213,5.227,214,6.097]],["t/291",[215,6.661,216,5.726]],["t/293",[217,6.097,218,6.661]],["t/294",[219,5.726,220,6.661]],["t/296",[165,5.448,221,6.661]],["t/298",[52,6.328]],["t/299",[222,6.097,223,6.097]],["t/303",[71,4.296,224,4.043,225,4.161]],["t/305",[34,3.274,41,3.522,205,4.042,225,3.624]],["t/307",[36,3.681,213,4.453,226,4.878]],["t/309",[138,4.885,226,5.726]],["t/311",[28,2.9,34,2.9,71,3.314,224,3.119,227,3.314]],["t/313",[36,4.321,138,4.885]],["t/315",[28,3.274,38,3.624,41,3.522,211,3.522]],["t/317",[23,3.038,27,2.683,28,2.9,44,2.555,224,3.119]],["t/319",[165,5.448,228,4.513]],["t/321",[69,3.741,183,3.878,217,4.524,229,4.042]],["t/323",[34,2.9,47,3.119,183,3.435,195,4.007,197,3.314]],["t/325",[36,3.206,71,3.741,229,4.042,230,4.942]],["t/327",[38,3.21,229,3.58,231,3.763,232,4.007,233,3.763]],["t/329",[96,2.638,234,3.763,235,4.377,236,4.377,237,3.763]],["t/331",[34,2.602,41,2.799,205,3.213,234,3.377,237,3.377,238,3.928]],["t/333",[36,2.548,41,2.799,169,3.213,216,3.377,234,3.377,239,3.928]],["t/335",[20,2.595,22,3.119,23,3.038,66,2.595,199,3.763]],["t/337",[180,4.885,240,3.722]],["t/338",[196,4.878,241,4.296,242,5.674]],["t/340",[34,3.274,71,3.741,240,2.762,241,3.741]],["t/342",[23,3.431,36,3.206,229,4.042,241,3.741]],["t/344",[38,4.161,241,4.296,243,5.674]],["t/346",[40,4.878,142,5.194,241,4.296]],["t/348",[19,4.641,27,3.479,244,5.674]],["t/352",[245,8.063]],["t/354",[246,6.097,247,6.097]],["t/356",[248,6.661,249,4.236]],["t/358",[249,4.236,250,6.097]],["t/360",[207,5.726,249,4.236]],["t/362",[249,4.236,251,6.661]],["t/364",[176,3.759,249,3.608,252,4.878]],["t/366",[249,3.608,253,5.674,254,5.674]],["t/368",[249,4.236,255,5.227]],["t/370",[249,4.236,256,5.448]],["t/372",[1,6.097,249,4.236]],["t/374",[249,5.128]],["t/376",[209,4.453,249,3.608,257,5.674]],["t/378",[52,6.328]],["t/381",[258,6.661,259,6.661]],["t/385",[90,6.328]],["t/387",[68,5.448,260,5.726]],["t/389",[19,4.641,28,3.759,261,5.674]],["t/391",[84,5.448,197,5.043]],["t/393",[262,6.596]],["t/395",[24,4.641,233,4.878,262,4.641]],["t/397",[13,5.043,219,5.726]],["t/398",[13,5.043,263,4.413]],["t/400",[219,5.726,263,4.413]],["t/402",[211,4.747,264,5.227]],["t/404",[13,4.296,265,5.674,266,4.296]],["t/406",[79,4.042,190,3.431,267,4.524,268,4.942]],["t/408",[228,5.464]],["t/410",[170,3.541,228,3.845,263,3.759]],["t/412",[170,3.541,228,3.845,269,5.674]],["t/414",[39,3.608,181,4.878,228,3.845]],["t/416",[9,5.043,270,5.726]],["t/417",[9,4.296,263,3.759,270,4.878]],["t/419",[9,4.296,270,4.878,271,5.194]],["t/421",[39,3.608,72,4.296,272,5.194]],["t/423",[273,7.381]],["t/425",[274,5.448,275,5.448]],["t/427",[267,6.097,274,5.448]],["t/429",[77,3.314,274,3.58,275,3.58,276,4.007,277,3.763]],["t/431",[39,2.784,77,3.314,275,3.58,277,3.763,278,4.007]],["t/433",[10,6.104]],["t/435",[279,5.227,280,6.661]],["t/437",[170,3.084,171,4.524,263,3.274,264,3.878]],["t/439",[22,4.043,199,4.878,278,5.194]],["t/441",[8,4.321,46,5.043]],["t/443",[46,5.043,122,5.227]],["t/445",[17,5.227,281,5.726]],["t/447",[282,8.063]],["t/449",[56,5.194,60,4.878,283,5.674]],["t/451",[60,5.726,284,6.661]],["t/453",[285,4.624,286,5.726]],["t/455",[20,3.95,287,6.097]],["t/457",[20,3.95,288,5.726]],["t/459",[26,4.513,263,4.413]],["t/461",[9,3.741,170,3.084,289,4.942,290,4.942]],["t/463",[8,3.681,209,4.453,291,5.674]],["t/465",[197,4.296,292,5.194,293,5.674]],["t/467",[29,4.885,294,6.097]],["t/469",[295,5.726,296,6.661]],["t/471",[23,3.038,120,4.007,224,3.119,297,4.377,298,4.377]],["t/473",[263,3.274,299,4.042,300,4.942,301,4.942]],["t/475",[170,2.732,292,4.007,302,4.377,303,4.377,304,4.377]],["t/477",[20,3.364,271,5.194,305,4.641]],["t/479",[8,3.681,13,4.296,92,4.878]],["t/481",[306,4.942,307,4.042,308,4.942,309,4.524]],["t/483",[294,6.097,310,5.726]],["t/485",[26,3.845,263,3.759,311,5.674]],["t/487",[26,4.513,312,6.661]],["t/489",[26,4.513,313,6.097]],["t/491",[25,4.453,47,4.043,64,4.453]],["t/493",[288,6.931]],["t/495",[314,7.381]],["t/497",[315,7.381]],["t/499",[70,4.321,316,6.661]],["t/501",[317,8.063]],["t/503",[197,3.741,211,3.522,318,4.524,319,4.942]],["t/505",[180,4.885,263,4.413]],["t/509",[8,4.321,227,5.043]],["t/511",[117,6.328]],["t/513",[27,4.084,28,4.413]],["t/515",[227,5.043,320,6.661]],["t/517",[23,3.939,224,4.043,227,4.296]],["t/519",[28,2.9,34,2.9,71,3.314,224,3.119,227,3.314]],["t/521",[36,4.321,138,4.885]],["t/523",[28,3.274,38,3.624,41,3.522,211,3.522]],["t/525",[138,4.161,321,4.043,322,5.674]],["t/527",[23,3.431,169,4.042,224,3.522,323,4.248]],["t/529",[34,3.274,41,3.522,138,3.624,323,4.248]],["t/531",[20,2.595,36,2.839,169,3.58,323,3.763,324,4.377]],["t/533",[122,5.227,325,6.661]],["t/536",[96,4.015,326,6.661]],["t/538",[96,3.42,194,4.878,327,5.194]],["t/540",[20,2.93,96,2.978,256,4.042,328,3.349]],["t/542",[20,2.93,96,2.978,281,4.248,328,3.349]],["t/544",[329,8.063]],["t/546",[216,5.726,255,5.227]],["t/548",[330,7.381]],["t/550",[105,4.042,128,4.248,331,4.942,332,4.942]],["t/552",[105,4.641,333,4.878,334,5.194]],["t/554",[328,4.513,333,5.726]],["t/558",[90,6.328]],["t/560",[193,4.885,335,5.726]],["t/562",[336,8.063]],["t/564",[335,5.726,337,6.661]],["t/566",[256,5.448,335,5.726]],["t/568",[338,5.674,339,5.674,340,4.296]],["t/570",[146,5.448,341,5.726]],["t/572",[249,5.128]],["t/574",[225,5.913]],["t/576",[321,4.747,342,4.513]],["t/578",[228,4.513,321,4.747]],["t/580",[69,5.043,72,5.043]],["t/582",[64,5.227,343,4.747]],["t/584",[96,4.015,344,6.661]],["t/586",[135,5.194,137,4.641,345,3.541]],["t/588",[240,4.506]],["t/590",[165,6.596]],["t/592",[21,5.227,26,4.513]],["t/594",[26,4.513,147,5.726]],["t/595",[346,6.661,347,6.661]],["t/597",[274,6.596]],["t/599",[10,6.104]],["t/601",[25,5.227,348,6.661]],["t/603",[147,5.726,299,5.448]],["t/605",[349,6.661,350,6.661]],["t/607",[35,4.413,351,6.661]],["t/609",[59,6.097,352,6.661]],["t/611",[260,6.931]],["t/613",[39,3.608,353,5.674,354,5.674]],["t/615",[222,6.097,295,5.726]],["t/617",[66,2.93,76,4.524,228,3.349,355,4.942]],["t/619",[26,3.845,137,4.641,356,4.453]],["t/623",[34,4.413,196,5.726]],["t/625",[36,4.321,357,6.661]],["t/627",[29,3.624,38,3.624,340,3.741,358,4.942]],["t/629",[25,5.227,40,5.726]],["t/631",[359,5.674,360,5.674,361,5.674]],["t/633",[52,4.453,362,5.674,363,5.674]],["t/635",[5,3.119,34,2.9,35,2.9,55,3.435,181,3.763]],["t/637",[5,3.119,35,2.9,36,2.839,55,3.435,183,3.435]],["t/639",[5,2.799,35,2.602,38,2.881,55,3.083,69,2.974,182,3.083]],["t/643",[273,6.097,364,5.726]],["t/645",[35,4.413,365,6.661]],["t/647",[201,4.885,202,6.097]],["t/648",[201,4.885,203,6.097]],["t/650",[201,4.161,204,5.194,205,4.641]],["t/652",[206,6.097,207,5.726]],["t/654",[140,5.726,208,6.097]],["t/656",[366,6.661,367,6.661]],["t/658",[321,4.747,368,6.097]],["t/660",[364,6.931]],["t/662",[264,5.227,356,5.227]],["t/664",[369,8.063]],["t/666",[]],["t/668",[364,5.726,370,6.661]],["t/670",[343,5.746]],["t/672",[371,8.063]],["t/674",[211,4.747,372,6.097]],["t/676",[310,6.931]],["t/678",[285,4.624,373,6.661]],["t/680",[64,5.227,343,4.747]],["t/682",[8,3.681,343,4.043,374,5.194]],["t/684",[20,2.93,285,3.431,343,3.522,374,4.524]],["t/686",[4,4.453,5,4.043,8,3.681]],["t/688",[20,3.95,375,6.661]],["t/690",[64,4.453,299,4.641,343,4.043]],["t/692",[4,4.453,255,4.453,376,5.674]],["t/694",[343,4.747,377,6.661]],["t/698",[378,7.381]],["t/700",[176,3.759,252,4.878,379,5.194]],["t/701",[380,6.097,381,5.448]],["t/703",[382,7.381]],["t/705",[39,4.236,383,6.097]],["t/707",[127,3.42,176,3.759,179,4.878]],["t/709",[384,7.381]],["t/711",[385,6.097,386,6.097]],["t/713",[66,2.93,127,2.978,387,3.522,388,4.524]],["t/715",[127,2.978,170,3.084,389,4.524,390,3.522]],["t/717",[70,3.206,77,3.741,391,4.524,392,4.524]],["t/719",[127,4.015,393,6.097]],["t/721",[127,4.015,394,6.097]],["t/723",[182,5.227,279,5.227]],["t/725",[39,4.236,395,6.097]],["t/727",[190,4.624,266,5.043]],["t/729",[193,4.885,396,6.097]],["t/730",[397,6.097,398,6.097]],["t/731",[399,7.381]],["t/732",[198,5.448,264,5.227]],["t/733",[39,4.236,122,5.227]],["t/735",[387,4.747,400,5.448]],["t/737",[401,6.097,402,5.726]],["t/739",[403,7.381]],["t/741",[134,5.227,381,5.448]],["t/743",[404,7.381]],["t/745",[387,4.747,400,5.448]],["t/747",[127,4.015,405,6.097]],["t/749",[387,4.747,406,6.097]],["t/750",[127,4.015,139,5.227]],["t/751",[139,6.328]],["t/753",[407,7.381]],["t/755",[408,7.381]],["t/759",[378,7.381]],["t/761",[176,3.759,252,4.878,379,5.194]],["t/762",[380,6.097,381,5.448]],["t/764",[382,7.381]],["t/766",[39,4.236,383,6.097]],["t/768",[127,3.42,176,3.759,179,4.878]],["t/770",[384,7.381]],["t/772",[385,6.097,386,6.097]],["t/774",[66,2.93,127,2.978,387,3.522,388,4.524]],["t/776",[127,2.978,170,3.084,389,4.524,390,3.522]],["t/778",[70,3.206,77,3.741,391,4.524,392,4.524]],["t/780",[127,4.015,393,6.097]],["t/782",[127,4.015,394,6.097]],["t/784",[182,5.227,279,5.227]],["t/786",[39,4.236,395,6.097]],["t/788",[190,4.624,266,5.043]],["t/790",[193,4.885,396,6.097]],["t/791",[397,6.097,398,6.097]],["t/792",[399,7.381]],["t/793",[198,5.448,264,5.227]],["t/794",[39,4.236,122,5.227]],["t/796",[387,4.747,400,5.448]],["t/798",[401,6.097,402,5.726]],["t/800",[403,7.381]],["t/802",[134,5.227,381,5.448]],["t/804",[404,7.381]],["t/806",[387,4.747,400,5.448]],["t/808",[127,4.015,405,6.097]],["t/810",[387,4.747,406,6.097]],["t/811",[127,4.015,139,5.227]],["t/812",[139,6.328]],["t/814",[407,7.381]],["t/816",[408,7.381]],["t/820",[15,4.641,342,3.845,409,5.674]],["t/822",[321,4.747,410,6.097]],["t/824",[20,3.364,233,4.878,411,5.674]],["t/826",[8,3.681,46,4.296,285,3.939]],["t/828",[26,4.513,299,5.448]],["t/830",[55,3.878,342,3.349,412,4.942,413,4.042]],["t/832",[51,4.248,321,3.522,342,3.349,414,3.431]],["t/834",[17,4.453,415,4.043,416,5.674]],["t/836",[50,5.464]],["t/840",[50,3.845,225,4.161,417,5.674]],["t/842",[23,4.624,225,4.885]],["t/843",[34,3.274,44,2.885,415,3.522,418,4.942]],["t/845",[28,2.9,36,2.839,44,2.555,419,4.377,420,3.763]],["t/847",[225,4.161,318,5.194,415,4.043]],["t/849",[8,3.206,117,3.878,285,3.431,420,4.248]],["t/851",[96,3.42,213,4.453,272,5.194]],["t/853",[341,4.878,421,4.453,422,5.674]],["t/855",[105,3.58,333,3.763,413,3.58,415,3.119,423,4.377]],["t/857",[424,8.063]],["t/859",[66,2.113,121,3.261,193,2.613,415,2.539,425,3.261,426,3.563,427,3.563]],["t/861",[22,4.747,420,5.726]],["t/863",[22,4.043,91,4.878,92,4.878]],["t/865",[22,4.043,415,4.043,428,5.194]],["t/867",[22,4.043,415,4.043,429,5.674]],["t/869",[415,4.747,430,6.661]],["t/873",[90,6.328]],["t/875",[27,4.084,119,4.513]],["t/877",[19,3.213,305,3.213,431,3.928,432,3.928,433,3.928,434,3.928]],["t/879",[174,4.524,305,4.042,435,4.942,436,4.524]],["t/881",[226,4.248,437,4.942,438,4.942,439,4.248]],["t/883",[72,3.314,119,2.966,241,3.314,414,3.038,440,4.377]],["t/885",[119,2.662,441,3.928,442,3.596,443,3.213,444,3.596,445,3.928]],["t/887",[31,3.377,70,2.548,119,2.662,183,3.083,184,3.596,446,3.928]],["t/889",[27,2.683,443,3.58,447,4.377,448,3.314,449,4.007]],["t/891",[96,2.638,119,2.966,328,2.966,450,4.007,451,4.377]],["t/893",[119,3.845,255,4.453,341,4.878]],["t/895",[27,2.408,117,3.083,119,2.662,452,3.928,453,3.377,454,3.596]],["t/897",[119,2.414,263,2.36,455,3.563,456,3.563,457,3.261,458,3.563,459,3.563]],["t/899",[200,5.194,305,4.641,460,5.674]],["t/901",[0,3.763,2,4.007,461,4.007,462,3.763,463,4.377]],["t/903",[288,6.931]],["t/905",[315,7.381]],["t/907",[314,7.381]],["t/909",[464,8.063]],["t/911",[66,2.93,237,4.248,356,3.878,448,3.741]],["t/912",[31,4.878,70,3.681,448,4.296]],["t/914",[176,2.9,372,4.007,448,3.314,465,4.377,466,4.377]],["t/916",[96,2.978,211,3.522,327,4.524,448,3.741]],["t/918",[39,2.498,70,2.548,328,2.662,467,3.928,468,3.928,469,3.928]],["t/920",[96,4.015,328,4.513]],["t/921",[27,2.184,66,2.113,96,2.147,328,2.414,453,3.063,470,3.563,471,3.261]],["t/923",[4,2.796,96,2.147,328,2.414,442,3.261,472,3.563,473,3.563,474,3.563]],["t/925",[96,2.367,97,3.596,225,2.881,328,2.662,330,3.596,475,3.596]],["t/927",[27,2.683,28,2.9,328,2.966,443,3.58,448,3.314]],["t/929",[27,3.479,281,4.878,421,4.453]],["t/931",[66,2.93,180,3.624,345,3.084,476,3.878]],["t/932",[345,4.157,477,6.097]],["t/934",[180,3.21,263,2.9,334,4.007,414,3.038,477,4.007]],["t/936",[478,4.377,479,3.21,480,4.377,481,2.9,482,4.377]],["t/938",[295,2.802,414,2.263,483,3.259,484,3.259,485,2.984,486,3.259,487,3.259,488,3.259]],["t/940",[84,4.042,414,3.431,489,4.942,490,4.942]],["t/942",[21,3.435,194,3.763,232,4.007,491,4.377,492,3.435]],["t/944",[345,3.541,413,4.641,493,5.674]],["t/946",[313,4.524,345,3.084,471,4.524,494,4.942]],["t/948",[7,4.296,53,4.641,66,3.364]],["t/949",[7,3.741,345,3.084,495,4.524,496,4.942]],["t/951",[33,4.161,231,4.878,345,3.541]],["t/953",[33,2.881,53,3.213,497,3.928,498,3.596,499,3.928,500,3.928]],["t/955",[231,4.878,457,5.194,501,4.641]],["t/957",[502,8.063]],["t/959",[50,2.209,119,2.209,127,1.964,193,2.39,444,2.984,461,2.984,501,2.666,503,3.259]],["t/961",[13,4.296,402,4.878,504,5.674]],["t/963",[29,2.203,39,1.91,50,2.035,53,2.457,345,1.875,443,2.457,501,2.457,505,3.004,506,3.004]],["t/965",[14,4.007,54,3.763,154,4.007,277,3.763,507,4.377]],["t/967",[54,4.878,275,4.641,508,5.674]],["t/969",[276,4.524,498,4.524,509,4.942,510,4.942]],["t/971",[66,2.93,170,3.084,307,4.042,356,3.878]],["t/972",[79,4.042,211,3.522,228,3.349,511,4.942]],["t/974",[436,5.194,512,5.674,513,5.674]],["t/976",[514,6.661,515,6.661]],["t/978",[307,3.58,309,4.007,310,3.763,414,3.038,454,4.007]],["t/980",[170,3.541,262,4.641,414,3.939]],["t/982",[170,3.084,414,3.431,501,4.042,516,4.942]],["t/984",[213,4.453,228,3.845,260,4.878]],["t/986",[27,3.479,345,3.541,495,5.194]],["t/988",[46,2.974,70,2.548,116,3.596,197,2.974,450,3.596,517,3.928]],["t/990",[35,3.274,223,4.524,518,4.524,519,4.942]],["t/992",[321,4.043,342,3.845,414,3.939]],["t/994",[240,4.506]],["t/995",[240,3.722,345,4.157]],["t/997",[45,4.641,240,3.171,345,3.541]],["t/999",[45,2.666,57,2.666,66,3.104,462,2.802,520,3.259,521,3.259,522,3.259]],["t/1001",[20,1.54,24,3.553,45,2.124,57,2.124,66,2.576,70,1.684,255,2.038,428,2.377,439,2.232]],["t/1003",[240,2.762,345,3.084,453,4.248,523,4.942]],["t/1005",[240,2.762,524,4.248,525,4.524,526,4.942]],["t/1007",[240,2.762,340,3.741,527,4.248,528,4.942]],["t/1009",[50,2.414,62,2.796,68,2.914,240,1.991,529,3.063,530,3.563,531,3.563]],["t/1011",[246,4.524,247,4.524,249,3.143,527,4.248]],["t/1013",[50,3.845,532,5.674,533,5.674]],["t/1015",[66,3.95,534,6.661]],["t/1016",[8,3.681,256,4.641,535,5.674]],["t/1018",[128,4.878,136,4.878,137,4.641]],["t/1020",[4,4.453,193,4.161,518,5.194]],["t/1022",[133,5.194,201,4.161,527,4.878]],["t/1026",[90,6.328]],["t/1028",[240,3.722,250,6.097]],["t/1029",[536,8.063]],["t/1031",[524,6.931]],["t/1033",[524,5.726,525,6.097]],["t/1035",[537,6.661,538,5.726]],["t/1037",[307,6.596]],["t/1039",[62,5.227,529,5.726]],["t/1041",[62,5.227,539,6.661]],["t/1043",[266,5.043,286,5.726]],["t/1045",[134,4.453,540,5.674,541,4.878]],["t/1047",[134,3.878,542,4.942,543,4.248,544,3.522]],["t/1049",[545,6.661,546,6.661]],["t/1051",[413,5.448,547,4.747]],["t/1053",[342,4.513,548,6.661]],["t/1055",[549,8.063]],["t/1057",[538,5.726,550,6.661]],["t/1059",[551,6.661,552,6.661]],["t/1061",[529,5.726,553,6.661]],["t/1063",[342,3.845,554,5.674,555,5.674]],["t/1065",[556,6.661,557,6.661]],["t/1067",[224,4.747,558,6.661]],["t/1069",[342,4.513,559,6.661]],["t/1071",[543,4.878,544,4.043,560,5.674]],["t/1073",[543,4.878,561,5.674,562,5.674]],["t/1075",[563,8.063]],["t/1077",[564,5.674,565,5.674,566,5.674]],["t/1079",[567,6.661,568,6.661]],["t/1081",[340,5.043,569,6.661]],["t/1083",[570,5.674,571,5.674,572,5.674]],["t/1085",[573,6.097,574,6.097]],["t/1087",[547,4.043,573,5.194,575,5.674]],["t/1089",[262,5.448,342,4.513]],["t/1091",[574,6.097,576,6.661]],["t/1093",[577,6.097,578,6.661]],["t/1095",[462,5.726,577,6.097]],["t/1097",[285,4.624,541,5.726]],["t/1099",[285,4.624,343,4.747]],["t/1101",[579,6.661,580,6.661]],["t/1103",[47,4.747,581,6.661]],["t/1105",[70,3.206,77,3.741,182,3.878,240,2.762]],["t/1106",[190,4.624,212,5.448]],["t/1108",[266,3.741,321,3.522,582,4.942,583,4.942]],["t/1110",[479,4.161,544,4.043,584,4.878]],["t/1112",[476,4.453,544,4.043,584,4.878]],["t/1114",[421,4.453,544,4.043,584,4.878]],["t/1116",[390,4.043,547,4.043,585,4.296]],["t/1118",[390,3.522,547,3.522,585,3.741,586,4.248]],["t/1120",[390,4.043,585,4.296,587,5.194]],["t/1122",[390,3.522,585,3.741,586,4.248,587,4.524]],["t/1124",[390,3.522,585,3.741,588,4.524,589,4.248]],["t/1126",[390,3.119,585,3.314,586,3.763,588,4.007,589,3.763]],["t/1128",[547,4.043,590,5.674,591,5.194]],["t/1130",[178,5.194,592,5.674,593,5.674]],["t/1132",[176,3.274,439,4.248,547,3.522,594,4.042]],["t/1134",[176,3.759,594,4.641,595,5.674]],["t/1136",[176,3.274,212,4.042,287,4.524,594,4.042]],["t/1138",[91,4.248,176,3.274,286,4.248,594,4.042]],["t/1140",[29,4.885,596,5.043]],["t/1142",[29,4.161,591,5.194,596,4.296]],["t/1144",[596,5.043,597,6.661]],["t/1146",[190,4.624,596,5.043]],["t/1148",[547,4.747,596,5.043]],["t/1150",[547,3.522,596,3.741,598,4.942,599,4.942]],["t/1152",[600,6.661,601,6.661]],["t/1154",[541,5.726,602,6.661]],["t/1156",[603,8.063]],["t/1158",[61,5.194,212,4.641,604,5.674]],["t/1160",[180,4.161,368,5.194,605,5.674]],["t/1162",[606,8.063]],["t/1164",[607,8.063]],["t/1166",[608,5.726,609,6.097]],["t/1168",[96,2.978,342,3.349,608,4.248,609,4.524]],["t/1170",[610,6.661,611,6.661]],["t/1172",[190,3.939,209,4.453,266,4.296]],["t/1173",[612,8.063]],["t/1175",[479,4.161,481,3.759,492,4.453]],["t/1177",[479,3.624,481,3.274,492,3.878,613,4.524]],["t/1179",[166,4.524,479,3.624,481,3.274,614,4.942]],["t/1181",[479,4.161,481,3.759,544,4.043]],["t/1183",[476,4.453,481,3.759,492,4.453]],["t/1185",[476,3.878,481,3.274,492,3.878,613,4.524]],["t/1187",[476,4.453,481,3.759,544,4.043]],["t/1189",[615,6.097,616,6.661]],["t/1191",[615,6.097,617,6.661]],["t/1193",[190,4.624,421,5.227]],["t/1195",[608,6.931]],["t/1197",[190,3.939,421,4.453,544,4.043]],["t/1199",[279,5.227,618,6.661]],["t/1201",[279,5.227,589,5.726]],["t/1203",[209,4.453,481,3.759,619,5.674]],["t/1205",[62,5.227,425,6.097]],["t/1207",[479,4.885,620,6.661]],["t/1209",[485,6.097,621,6.661]],["t/1211",[240,3.722,622,6.661]],["t/1212",[240,3.722,623,6.661]],["t/1214",[240,3.171,449,5.194,624,5.674]],["t/1216",[85,6.097,625,6.661]],["t/1218",[340,5.043,626,6.661]],["t/1220",[340,5.043,538,5.726]],["t/1222",[66,3.364,285,3.939,481,3.759]],["t/1223",[214,5.194,240,3.171,481,3.759]],["t/1225",[228,4.513,410,6.097]],["t/1227",[57,5.448,146,5.448]],["t/1229",[627,6.661,628,6.661]],["t/1231",[8,4.321,629,6.661]],["t/1233",[23,4.624,630,6.661]],["t/1235",[345,4.157,475,6.097]],["t/1237",[213,5.227,240,3.722]],["t/1239",[285,4.624,631,6.661]],["t/1241",[138,4.885,227,5.043]],["t/1243",[44,3.312,117,4.453,240,3.171]],["t/1245",[180,4.885,240,3.722]],["t/1247",[146,5.448,240,3.722]],["t/1249",[51,4.878,240,3.171,345,3.541]],["t/1251",[143,6.097,356,5.227]]],"invertedIndex":[["",{"_index":66,"t":{"83":{"position":[[4,1]]},"95":{"position":[[8,1]]},"260":{"position":[[4,1]]},"335":{"position":[[7,1]]},"617":{"position":[[7,1]]},"713":{"position":[[8,1]]},"774":{"position":[[8,1]]},"859":{"position":[[29,1]]},"911":{"position":[[4,1]]},"921":{"position":[[42,4]]},"931":{"position":[[6,1]]},"948":{"position":[[5,1]]},"971":{"position":[[7,1]]},"999":{"position":[[32,3],[61,3]]},"1001":{"position":[[27,5],[44,1]]},"1015":{"position":[[7,1]]},"1222":{"position":[[6,1]]}}}],["0",{"_index":242,"t":{"338":{"position":[[5,1]]}}}],["1",{"_index":34,"t":{"43":{"position":[[0,2]]},"305":{"position":[[0,2]]},"311":{"position":[[0,2]]},"323":{"position":[[0,2]]},"331":{"position":[[0,2]]},"340":{"position":[[5,1]]},"519":{"position":[[0,2]]},"529":{"position":[[0,2]]},"623":{"position":[[0,2]]},"635":{"position":[[30,1]]},"843":{"position":[[7,2]]}}}],["19",{"_index":555,"t":{"1063":{"position":[[6,2]]}}}],["2",{"_index":36,"t":{"45":{"position":[[0,2]]},"169":{"position":[[7,1]]},"307":{"position":[[0,2]]},"313":{"position":[[0,2]]},"325":{"position":[[0,2]]},"333":{"position":[[0,2]]},"342":{"position":[[5,1]]},"521":{"position":[[0,2]]},"531":{"position":[[0,2]]},"625":{"position":[[0,2]]},"637":{"position":[[30,1]]},"845":{"position":[[7,2]]}}}],["2.0",{"_index":379,"t":{"700":{"position":[[9,3]]},"761":{"position":[[9,3]]}}}],["2.0.4",{"_index":511,"t":{"972":{"position":[[33,5]]}}}],["3",{"_index":38,"t":{"47":{"position":[[0,2]]},"315":{"position":[[0,2]]},"327":{"position":[[0,2]]},"344":{"position":[[5,1]]},"523":{"position":[[0,2]]},"627":{"position":[[0,2]]},"639":{"position":[[30,1]]}}}],["3.0",{"_index":400,"t":{"735":{"position":[[8,3]]},"745":{"position":[[8,3]]},"796":{"position":[[8,3]]},"806":{"position":[[8,3]]}}}],["3.1.0",{"_index":458,"t":{"897":{"position":[[47,6]]}}}],["3.1.1",{"_index":459,"t":{"897":{"position":[[58,5]]}}}],["4",{"_index":40,"t":{"49":{"position":[[0,2]]},"346":{"position":[[5,1]]},"629":{"position":[[0,2]]}}}],["401",{"_index":526,"t":{"1005":{"position":[[24,3]]}}}],["403",{"_index":444,"t":{"885":{"position":[[34,3]]},"959":{"position":[[70,5]]}}}],["404",{"_index":448,"t":{"889":{"position":[[29,3]]},"911":{"position":[[0,3]]},"912":{"position":[[0,3]]},"914":{"position":[[0,3]]},"916":{"position":[[0,3]]},"927":{"position":[[24,3]]}}}],["5",{"_index":359,"t":{"631":{"position":[[0,2]]}}}],["6",{"_index":362,"t":{"633":{"position":[[0,2]]}}}],["_basevalu",{"_index":494,"t":{"946":{"position":[[26,15]]}}}],["abort",{"_index":440,"t":{"883":{"position":[[26,6]]}}}],["access",{"_index":17,"t":{"23":{"position":[[15,6]]},"25":{"position":[[9,6]]},"79":{"position":[[9,6]]},"445":{"position":[[7,6]]},"834":{"position":[[12,13]]}}}],["action",{"_index":430,"t":{"869":{"position":[[0,6]]}}}],["ad",{"_index":23,"t":{"31":{"position":[[0,6]]},"317":{"position":[[0,6]]},"335":{"position":[[0,6]]},"342":{"position":[[9,6]]},"471":{"position":[[29,5]]},"517":{"position":[[0,6]]},"527":{"position":[[0,6]]},"842":{"position":[[0,6]]},"1233":{"position":[[0,6]]}}}],["add",{"_index":41,"t":{"49":{"position":[[3,3]]},"179":{"position":[[0,3]]},"305":{"position":[[3,3]]},"315":{"position":[[3,3]]},"331":{"position":[[3,3]]},"333":{"position":[[3,3]]},"523":{"position":[[3,3]]},"529":{"position":[[3,3]]}}}],["address",{"_index":425,"t":{"859":{"position":[[17,7]]},"1205":{"position":[[3,7]]}}}],["addy.io",{"_index":549,"t":{"1055":{"position":[[0,7]]}}}],["adguard",{"_index":594,"t":{"1132":{"position":[[0,7]]},"1134":{"position":[[0,7]]},"1136":{"position":[[0,7]]},"1138":{"position":[[0,7]]}}}],["admin",{"_index":504,"t":{"961":{"position":[[17,5]]}}}],["advanc",{"_index":265,"t":{"404":{"position":[[0,8]]}}}],["alert",{"_index":619,"t":{"1203":{"position":[[15,6]]}}}],["alloc",{"_index":488,"t":{"938":{"position":[[42,10]]}}}],["allow",{"_index":412,"t":{"830":{"position":[[0,8]]}}}],["altern",{"_index":55,"t":{"71":{"position":[[0,11]]},"635":{"position":[[0,11]]},"637":{"position":[[0,11]]},"639":{"position":[[0,11]]},"830":{"position":[[9,11]]}}}],["anonym",{"_index":135,"t":{"177":{"position":[[7,9]]},"586":{"position":[[0,9]]}}}],["anoth",{"_index":406,"t":{"749":{"position":[[4,7]]},"810":{"position":[[4,7]]}}}],["anyth",{"_index":503,"t":{"959":{"position":[[61,8]]}}}],["apach",{"_index":314,"t":{"495":{"position":[[0,6]]},"907":{"position":[[0,6]]}}}],["api",{"_index":85,"t":{"108":{"position":[[0,3]]},"1216":{"position":[[0,3]]}}}],["app",{"_index":211,"t":{"288":{"position":[[0,3]]},"315":{"position":[[24,3]]},"402":{"position":[[11,4]]},"503":{"position":[[34,3]]},"523":{"position":[[24,3]]},"674":{"position":[[10,4]]},"916":{"position":[[18,4]]},"972":{"position":[[0,3]]}}}],["apparmor",{"_index":438,"t":{"881":{"position":[[11,8]]}}}],["appconfig",{"_index":97,"t":{"126":{"position":[[0,9]]},"925":{"position":[[38,9]]}}}],["appconfig.auth",{"_index":98,"t":{"128":{"position":[[0,14]]}}}],["appconfig.auth.headerauth",{"_index":101,"t":{"134":{"position":[[0,25]]}}}],["appconfig.auth.keycloak",{"_index":100,"t":{"132":{"position":[[0,23]]}}}],["appconfig.auth.oidc",{"_index":102,"t":{"136":{"position":[[0,19]]}}}],["appconfig.auth.us",{"_index":99,"t":{"130":{"position":[[0,20]]}}}],["appconfig.hidecompon",{"_index":104,"t":{"140":{"position":[[0,24]]}}}],["appconfig.websearch",{"_index":103,"t":{"138":{"position":[[0,19]]}}}],["applic",{"_index":215,"t":{"291":{"position":[[0,11]]}}}],["array",{"_index":238,"t":{"331":{"position":[[41,5]]}}}],["asset",{"_index":260,"t":{"387":{"position":[[10,6]]},"611":{"position":[[0,6]]},"984":{"position":[[11,6]]}}}],["astronomi",{"_index":570,"t":{"1083":{"position":[[0,9]]}}}],["auth",{"_index":7,"t":{"15":{"position":[[9,4]]},"31":{"position":[[12,4]]},"35":{"position":[[5,4]]},"73":{"position":[[14,4]]},"948":{"position":[[0,4]]},"949":{"position":[[0,4]]}}}],["authent",{"_index":10,"t":{"17":{"position":[[11,14]]},"57":{"position":[[7,14]]},"71":{"position":[[12,14]]},"81":{"position":[[11,14]]},"433":{"position":[[0,14]]},"599":{"position":[[0,14]]}}}],["authentik",{"_index":54,"t":{"69":{"position":[[0,9]]},"965":{"position":[[29,9]]},"967":{"position":[[27,9]]}}}],["auto",{"_index":267,"t":{"406":{"position":[[0,4]]},"427":{"position":[[0,4]]}}}],["autom",{"_index":366,"t":{"656":{"position":[[0,9]]}}}],["automat",{"_index":269,"t":{"412":{"position":[[0,9]]}}}],["avail",{"_index":320,"t":{"515":{"position":[[0,9]]}}}],["awesom",{"_index":247,"t":{"354":{"position":[[5,7]]},"1011":{"position":[[5,7]]}}}],["back",{"_index":270,"t":{"416":{"position":[[0,7]]},"417":{"position":[[0,7]]},"419":{"position":[[0,7]]}}}],["backup",{"_index":72,"t":{"91":{"position":[[11,6]]},"93":{"position":[[12,6]]},"99":{"position":[[17,6]]},"421":{"position":[[15,6]]},"580":{"position":[[6,6]]},"883":{"position":[[0,6]]}}}],["balanc",{"_index":546,"t":{"1049":{"position":[[7,7]]}}}],["bang",{"_index":375,"t":{"688":{"position":[[6,5]]}}}],["bar",{"_index":426,"t":{"859":{"position":[[25,3]]}}}],["bare",{"_index":191,"t":{"258":{"position":[[0,4]]}}}],["base",{"_index":63,"t":{"79":{"position":[[3,5]]}}}],["basic",{"_index":325,"t":{"533":{"position":[[0,5]]}}}],["be",{"_index":298,"t":{"471":{"position":[[23,5]]}}}],["befor",{"_index":308,"t":{"481":{"position":[[13,6]]}}}],["beginn",{"_index":210,"t":{"286":{"position":[[14,9]]}}}],["blacklist",{"_index":539,"t":{"1041":{"position":[[3,9]]}}}],["blind",{"_index":416,"t":{"834":{"position":[[6,5]]}}}],["block",{"_index":439,"t":{"881":{"position":[[20,6]]},"1001":{"position":[[36,7]]},"1132":{"position":[[13,5]]}}}],["bookmark",{"_index":399,"t":{"731":{"position":[[0,9]]},"792":{"position":[[0,9]]}}}],["boot",{"_index":268,"t":{"406":{"position":[[24,4]]}}}],["boundari",{"_index":352,"t":{"609":{"position":[[6,10]]}}}],["bountysourc",{"_index":149,"t":{"189":{"position":[[22,12]]}}}],["branch",{"_index":204,"t":{"280":{"position":[[4,6]]},"650":{"position":[[4,6]]}}}],["brewhack",{"_index":384,"t":{"709":{"position":[[0,8]]},"770":{"position":[[0,8]]}}}],["browser",{"_index":193,"t":{"262":{"position":[[0,7]]},"560":{"position":[[0,7]]},"729":{"position":[[0,7]]},"790":{"position":[[0,7]]},"859":{"position":[[9,7]]},"959":{"position":[[19,7]]},"1020":{"position":[[12,7]]}}}],["bug",{"_index":136,"t":{"177":{"position":[[17,3]]},"185":{"position":[[8,3]]},"1018":{"position":[[14,3]]}}}],["build",{"_index":180,"t":{"237":{"position":[[0,5]]},"337":{"position":[[0,8]]},"505":{"position":[[0,8]]},"931":{"position":[[0,5]]},"934":{"position":[[5,5]]},"1160":{"position":[[9,6]]},"1245":{"position":[[0,5]]}}}],["built",{"_index":6,"t":{"15":{"position":[[0,5]]}}}],["bundlephobia",{"_index":221,"t":{"296":{"position":[[15,12]]}}}],["button",{"_index":441,"t":{"885":{"position":[[5,6]]}}}],["caddi",{"_index":315,"t":{"497":{"position":[[0,5]]},"905":{"position":[[0,5]]}}}],["callback",{"_index":505,"t":{"963":{"position":[[36,8]]}}}],["capabl",{"_index":296,"t":{"469":{"position":[[6,12]]}}}],["cdn",{"_index":188,"t":{"251":{"position":[[13,3]]},"260":{"position":[[0,3]]}}}],["certain",{"_index":236,"t":{"329":{"position":[[25,7]]}}}],["certif",{"_index":275,"t":{"425":{"position":[[4,12]]},"429":{"position":[[26,11]]},"431":{"position":[[22,11]]},"967":{"position":[[10,11]]}}}],["cft",{"_index":397,"t":{"730":{"position":[[0,3]]},"791":{"position":[[0,3]]}}}],["chang",{"_index":121,"t":{"163":{"position":[[11,7]]},"859":{"position":[[0,8]]}}}],["chart",{"_index":631,"t":{"1239":{"position":[[12,6]]}}}],["check",{"_index":321,"t":{"525":{"position":[[0,8]]},"576":{"position":[[7,8]]},"578":{"position":[[7,6]]},"658":{"position":[[3,6]]},"822":{"position":[[11,8]]},"832":{"position":[[31,6]]},"992":{"position":[[7,6]]},"1108":{"position":[[24,7]]}}}],["choic",{"_index":354,"t":{"613":{"position":[[28,6]]}}}],["chuck",{"_index":564,"t":{"1077":{"position":[[0,5]]}}}],["ci",{"_index":368,"t":{"658":{"position":[[0,2]]},"1160":{"position":[[6,2]]}}}],["clear",{"_index":377,"t":{"694":{"position":[[0,8]]}}}],["click",{"_index":474,"t":{"923":{"position":[[46,7]]}}}],["client",{"_index":499,"t":{"953":{"position":[[36,6]]}}}],["client_id",{"_index":509,"t":{"969":{"position":[[8,9]]}}}],["clipboard",{"_index":533,"t":{"1013":{"position":[[8,9]]}}}],["clock",{"_index":536,"t":{"1029":{"position":[[0,5]]}}}],["cloud",{"_index":69,"t":{"85":{"position":[[0,5]]},"239":{"position":[[10,5]]},"260":{"position":[[6,5]]},"321":{"position":[[19,5]]},"580":{"position":[[0,5]]},"639":{"position":[[34,5]]}}}],["code",{"_index":413,"t":{"830":{"position":[[28,5]]},"855":{"position":[[5,6]]},"944":{"position":[[12,4]]},"1051":{"position":[[0,4]]}}}],["collect",{"_index":257,"t":{"376":{"position":[[5,11]]}}}],["color",{"_index":415,"t":{"834":{"position":[[0,5]]},"843":{"position":[[10,6]]},"847":{"position":[[16,6]]},"855":{"position":[[28,6]]},"859":{"position":[[35,5]]},"865":{"position":[[9,5]]},"867":{"position":[[4,5]]},"869":{"position":[[7,6]]}}}],["comic",{"_index":568,"t":{"1079":{"position":[[5,6]]}}}],["command",{"_index":84,"t":{"106":{"position":[[0,8]]},"271":{"position":[[8,8]]},"391":{"position":[[8,8]]},"940":{"position":[[0,7]]}}}],["commit",{"_index":206,"t":{"282":{"position":[[0,6]]},"652":{"position":[[0,6]]}}}],["commun",{"_index":301,"t":{"473":{"position":[[24,13]]}}}],["compact",{"_index":485,"t":{"938":{"position":[[17,8]]},"1209":{"position":[[0,7]]}}}],["complet",{"_index":82,"t":{"104":{"position":[[0,8]]}}}],["compon",{"_index":169,"t":{"211":{"position":[[9,10]]},"333":{"position":[[41,9]]},"527":{"position":[[21,9]]},"531":{"position":[[19,9]]}}}],["compos",{"_index":171,"t":{"223":{"position":[[13,7]]},"437":{"position":[[32,7]]}}}],["condit",{"_index":239,"t":{"333":{"position":[[11,11]]}}}],["conf.yml",{"_index":418,"t":{"843":{"position":[[26,8]]}}}],["config",{"_index":27,"t":{"37":{"position":[[6,6]]},"47":{"position":[[28,6]]},"112":{"position":[[33,6]]},"157":{"position":[[8,6]]},"161":{"position":[[0,6]]},"317":{"position":[[27,6]]},"348":{"position":[[11,6]]},"513":{"position":[[0,6]]},"875":{"position":[[0,6]]},"889":{"position":[[0,7]]},"895":{"position":[[44,6]]},"921":{"position":[[31,6]]},"927":{"position":[[4,6]]},"929":{"position":[[7,6]]},"986":{"position":[[0,6]]}}}],["configmap",{"_index":435,"t":{"879":{"position":[[11,9]]}}}],["configur",{"_index":25,"t":{"31":{"position":[[20,13]]},"59":{"position":[[0,13]]},"491":{"position":[[11,13]]},"601":{"position":[[0,13]]},"629":{"position":[[3,9]]}}}],["connect",{"_index":462,"t":{"901":{"position":[[11,7]]},"999":{"position":[[11,7]]},"1095":{"position":[[4,10]]}}}],["consol",{"_index":518,"t":{"990":{"position":[[16,7]]},"1020":{"position":[[20,7]]}}}],["contain",{"_index":263,"t":{"398":{"position":[[0,9]]},"400":{"position":[[0,9]]},"410":{"position":[[16,9]]},"417":{"position":[[11,10]]},"437":{"position":[[9,10]]},"459":{"position":[[0,9]]},"473":{"position":[[14,9]]},"485":{"position":[[0,9]]},"505":{"position":[[18,9]]},"897":{"position":[[0,9]]},"934":{"position":[[28,9]]}}}],["content",{"_index":90,"t":{"116":{"position":[[0,8]]},"385":{"position":[[0,8]]},"558":{"position":[[0,8]]},"873":{"position":[[0,8]]},"1026":{"position":[[0,8]]}}}],["continu",{"_index":410,"t":{"822":{"position":[[0,10]]},"1225":{"position":[[0,10]]}}}],["contribut",{"_index":133,"t":{"175":{"position":[[34,12]]},"1022":{"position":[[4,13]]}}}],["contributor",{"_index":158,"t":{"197":{"position":[[0,12]]},"203":{"position":[[0,12]]}}}],["control",{"_index":402,"t":{"737":{"position":[[7,7]]},"798":{"position":[[7,7]]},"961":{"position":[[23,8]]}}}],["cooki",{"_index":336,"t":{"562":{"position":[[0,7]]}}}],["copi",{"_index":532,"t":{"1013":{"position":[[0,4]]}}}],["cor",{"_index":45,"t":{"51":{"position":[[0,4]]},"997":{"position":[[7,4]]},"999":{"position":[[0,4]]},"1001":{"position":[[0,4]]}}}],["core",{"_index":166,"t":{"207":{"position":[[0,4]]},"1179":{"position":[[14,4]]}}}],["coverag",{"_index":322,"t":{"525":{"position":[[21,8]]}}}],["covid",{"_index":554,"t":{"1063":{"position":[[0,5]]}}}],["cpanel",{"_index":317,"t":{"501":{"position":[[0,6]]}}}],["cpu",{"_index":479,"t":{"936":{"position":[[5,3]]},"1110":{"position":[[0,3]]},"1175":{"position":[[8,3]]},"1177":{"position":[[8,3]]},"1179":{"position":[[0,3]]},"1181":{"position":[[0,3]]},"1207":{"position":[[0,3]]}}}],["crash",{"_index":455,"t":{"897":{"position":[[10,7]]}}}],["creat",{"_index":71,"t":{"91":{"position":[[0,8]]},"303":{"position":[[0,8]]},"311":{"position":[[3,6]]},"325":{"position":[[3,6]]},"340":{"position":[[9,6]]},"519":{"position":[[3,6]]}}}],["credenti",{"_index":32,"t":{"39":{"position":[[13,11]]}}}],["croco_griev",{"_index":403,"t":{"739":{"position":[[0,14]]},"800":{"position":[[0,14]]}}}],["cron",{"_index":582,"t":{"1108":{"position":[[0,4]]}}}],["crypto",{"_index":134,"t":{"175":{"position":[[53,6]]},"741":{"position":[[0,6]]},"802":{"position":[[0,6]]},"1045":{"position":[[0,6]]},"1047":{"position":[[0,6]]}}}],["css",{"_index":420,"t":{"845":{"position":[[15,3]]},"849":{"position":[[15,3]]},"861":{"position":[[0,3]]}}}],["current",{"_index":492,"t":{"942":{"position":[[36,7]]},"1175":{"position":[[0,7]]},"1177":{"position":[[0,7]]},"1183":{"position":[[0,7]]},"1185":{"position":[[0,7]]}}}],["custom",{"_index":285,"t":{"453":{"position":[[0,6]]},"678":{"position":[[0,6]]},"684":{"position":[[6,6]]},"826":{"position":[[8,6]]},"849":{"position":[[8,6]]},"1097":{"position":[[0,6]]},"1099":{"position":[[0,6]]},"1222":{"position":[[8,14]]},"1239":{"position":[[0,11]]}}}],["customis",{"_index":361,"t":{"631":{"position":[[11,13]]}}}],["daemon",{"_index":303,"t":{"475":{"position":[[24,6]]}}}],["dash",{"_index":381,"t":{"701":{"position":[[8,4]]},"741":{"position":[[7,4]]},"762":{"position":[[8,4]]},"802":{"position":[[7,4]]}}}],["dashboard",{"_index":127,"t":{"171":{"position":[[11,9]]},"707":{"position":[[9,9]]},"713":{"position":[[14,9]]},"715":{"position":[[20,9]]},"719":{"position":[[4,9]]},"721":{"position":[[12,9]]},"747":{"position":[[8,9]]},"750":{"position":[[16,9]]},"768":{"position":[[9,9]]},"774":{"position":[[14,9]]},"776":{"position":[[20,9]]},"780":{"position":[[4,9]]},"782":{"position":[[12,9]]},"808":{"position":[[8,9]]},"811":{"position":[[16,9]]},"959":{"position":[[35,9]]}}}],["dashi",{"_index":39,"t":{"47":{"position":[[22,5]]},"414":{"position":[[9,5]]},"421":{"position":[[0,5]]},"431":{"position":[[37,5]]},"613":{"position":[[5,5]]},"705":{"position":[[12,5]]},"725":{"position":[[0,5]]},"733":{"position":[[0,5]]},"766":{"position":[[12,5]]},"786":{"position":[[0,5]]},"794":{"position":[[0,5]]},"918":{"position":[[0,5]]},"963":{"position":[[16,5]]}}}],["data",{"_index":340,"t":{"568":{"position":[[16,4]]},"627":{"position":[[8,4]]},"1007":{"position":[[29,4]]},"1081":{"position":[[7,4]]},"1218":{"position":[[11,4]]},"1220":{"position":[[0,4]]}}}],["date",{"_index":290,"t":{"461":{"position":[[18,4]]}}}],["day",{"_index":572,"t":{"1083":{"position":[[25,3]]}}}],["default",{"_index":1,"t":{"5":{"position":[[0,7]]},"372":{"position":[[0,7]]}}}],["delet",{"_index":338,"t":{"568":{"position":[[0,8]]}}}],["deni",{"_index":431,"t":{"877":{"position":[[11,6]]}}}],["departur",{"_index":578,"t":{"1093":{"position":[[4,9]]}}}],["depend",{"_index":165,"t":{"205":{"position":[[0,12]]},"296":{"position":[[0,12]]},"319":{"position":[[9,12]]},"590":{"position":[[0,12]]}}}],["deploy",{"_index":35,"t":{"43":{"position":[[3,6]]},"219":{"position":[[0,10]]},"221":{"position":[[0,6]]},"239":{"position":[[0,6]]},"260":{"position":[[12,6]]},"607":{"position":[[9,10]]},"635":{"position":[[12,10]]},"637":{"position":[[12,10]]},"639":{"position":[[12,10]]},"645":{"position":[[0,10]]},"990":{"position":[[31,6]]}}}],["design",{"_index":254,"t":{"366":{"position":[[9,6]]}}}],["dev",{"_index":195,"t":{"266":{"position":[[15,3]]},"323":{"position":[[15,3]]}}}],["develop",{"_index":217,"t":{"293":{"position":[[0,11]]},"321":{"position":[[0,10]]}}}],["differ",{"_index":411,"t":{"824":{"position":[[8,9]]}}}],["dipan'",{"_index":380,"t":{"701":{"position":[[0,7]]},"762":{"position":[[0,7]]}}}],["directli",{"_index":376,"t":{"692":{"position":[[13,8]]}}}],["directori",{"_index":358,"t":{"627":{"position":[[13,9]]}}}],["disabl",{"_index":299,"t":{"473":{"position":[[0,7]]},"603":{"position":[[0,9]]},"690":{"position":[[0,9]]},"828":{"position":[[0,9]]}}}],["discuss",{"_index":145,"t":{"187":{"position":[[9,10]]}}}],["disk",{"_index":615,"t":{"1189":{"position":[[0,4]]},"1191":{"position":[[0,4]]}}}],["display",{"_index":527,"t":{"1007":{"position":[[7,10]]},"1011":{"position":[[23,10]]},"1022":{"position":[[22,10]]}}}],["dn",{"_index":287,"t":{"455":{"position":[[6,3]]},"1136":{"position":[[13,3]]}}}],["doc",{"_index":142,"t":{"183":{"position":[[12,4]]},"346":{"position":[[9,4]]}}}],["docker",{"_index":170,"t":{"221":{"position":[[12,6]]},"223":{"position":[[6,6]]},"249":{"position":[[10,6]]},"256":{"position":[[0,6]]},"410":{"position":[[9,6]]},"412":{"position":[[10,6]]},"437":{"position":[[25,6]]},"461":{"position":[[5,6]]},"475":{"position":[[17,6]]},"715":{"position":[[13,6]]},"776":{"position":[[13,6]]},"971":{"position":[[0,6]]},"980":{"position":[[23,6]]},"982":{"position":[[0,6]]}}}],["dockerhub",{"_index":514,"t":{"976":{"position":[[0,9]]}}}],["document",{"_index":369,"t":{"664":{"position":[[0,13]]}}}],["domain",{"_index":286,"t":{"453":{"position":[[7,6]]},"1043":{"position":[[0,6]]},"1138":{"position":[[17,7]]}}}],["don't",{"_index":292,"t":{"465":{"position":[[0,5]]},"475":{"position":[[0,5]]}}}],["donat",{"_index":130,"t":{"173":{"position":[[13,8]]}}}],["download",{"_index":592,"t":{"1130":{"position":[[9,8]]}}}],["dragon",{"_index":385,"t":{"711":{"position":[[4,7]]},"772":{"position":[[4,7]]}}}],["drone",{"_index":605,"t":{"1160":{"position":[[0,5]]}}}],["dure",{"_index":519,"t":{"990":{"position":[[24,6]]}}}],["dynam",{"_index":622,"t":{"1211":{"position":[[0,7]]}}}],["e.g",{"_index":468,"t":{"918":{"position":[[27,5]]}}}],["eacc",{"_index":433,"t":{"877":{"position":[[42,8]]}}}],["easypanel",{"_index":185,"t":{"245":{"position":[[0,9]]}}}],["econnrefus",{"_index":520,"t":{"999":{"position":[[19,12]]}}}],["edgeon",{"_index":186,"t":{"247":{"position":[[0,7]]}}}],["edit",{"_index":88,"t":{"112":{"position":[[24,4]]},"157":{"position":[[0,7]]}}}],["embed",{"_index":624,"t":{"1214":{"position":[[5,8]]}}}],["emoji",{"_index":207,"t":{"282":{"position":[[7,6]]},"360":{"position":[[0,5]]},"652":{"position":[[7,6]]}}}],["enabl",{"_index":15,"t":{"23":{"position":[[0,8]]},"47":{"position":[[3,6]]},"177":{"position":[[0,6]]},"820":{"position":[[0,8]]}}}],["endpoint",{"_index":233,"t":{"327":{"position":[[21,8]]},"395":{"position":[[17,8]]},"824":{"position":[[18,8]]}}}],["enforc",{"_index":49,"t":{"53":{"position":[[16,11]]},"67":{"position":[[16,11]]}}}],["engin",{"_index":374,"t":{"682":{"position":[[15,6]]},"684":{"position":[[20,6]]}}}],["enotfound",{"_index":522,"t":{"999":{"position":[[51,9]]}}}],["environ",{"_index":21,"t":{"29":{"position":[[6,11]]},"266":{"position":[[19,11]]},"275":{"position":[[0,11]]},"592":{"position":[[14,11]]},"942":{"position":[[44,11]]}}}],["environment",{"_index":199,"t":{"273":{"position":[[0,13]]},"335":{"position":[[15,13]]},"439":{"position":[[11,13]]}}}],["erof",{"_index":434,"t":{"877":{"position":[[51,6]]}}}],["error",{"_index":345,"t":{"586":{"position":[[10,5]]},"931":{"position":[[15,6]]},"932":{"position":[[5,5]]},"944":{"position":[[17,5]]},"946":{"position":[[0,6]]},"949":{"position":[[16,6]]},"951":{"position":[[18,5]]},"963":{"position":[[22,6]]},"986":{"position":[[18,6]]},"995":{"position":[[7,6]]},"997":{"position":[[12,6]]},"1003":{"position":[[13,5]]},"1235":{"position":[[9,6]]},"1249":{"position":[[23,6]]}}}],["eth",{"_index":561,"t":{"1073":{"position":[[0,3]]}}}],["evo",{"_index":393,"t":{"719":{"position":[[0,3]]},"780":{"position":[[0,3]]}}}],["exampl",{"_index":122,"t":{"165":{"position":[[0,7]]},"443":{"position":[[0,7]]},"533":{"position":[[6,7]]},"733":{"position":[[6,7]]},"794":{"position":[[6,7]]}}}],["example.com/dashi",{"_index":469,"t":{"918":{"position":[[33,18]]}}}],["exchang",{"_index":551,"t":{"1059":{"position":[[0,8]]}}}],["expos",{"_index":302,"t":{"475":{"position":[[6,6]]}}}],["exposur",{"_index":280,"t":{"435":{"position":[[8,8]]}}}],["extern",{"_index":341,"t":{"570":{"position":[[0,8]]},"853":{"position":[[8,8]]},"893":{"position":[[19,8]]}}}],["fail",{"_index":414,"t":{"832":{"position":[[16,7]]},"883":{"position":[[12,5]]},"934":{"position":[[11,5]]},"938":{"position":[[53,6]]},"940":{"position":[[8,6]]},"978":{"position":[[15,4]]},"980":{"position":[[12,7]]},"982":{"position":[[13,5]]},"992":{"position":[[14,7]]}}}],["failur",{"_index":497,"t":{"953":{"position":[[17,7]]}}}],["fair",{"_index":75,"t":{"97":{"position":[[0,4]]}}}],["favicon",{"_index":245,"t":{"352":{"position":[[0,8]]}}}],["featur",{"_index":147,"t":{"189":{"position":[[10,7]]},"594":{"position":[[9,8]]},"603":{"position":[[10,8]]}}}],["feed",{"_index":538,"t":{"1035":{"position":[[4,4]]},"1057":{"position":[[14,4]]},"1220":{"position":[[5,4]]}}}],["field",{"_index":93,"t":{"118":{"position":[[10,6]]}}}],["file",{"_index":28,"t":{"37":{"position":[[13,4]]},"47":{"position":[[35,4]]},"311":{"position":[[25,4]]},"315":{"position":[[12,4]]},"317":{"position":[[34,4]]},"389":{"position":[[0,4]]},"513":{"position":[[7,4]]},"519":{"position":[[25,4]]},"523":{"position":[[12,4]]},"845":{"position":[[19,4]]},"927":{"position":[[11,5]]}}}],["filebrows",{"_index":606,"t":{"1162":{"position":[[0,11]]}}}],["filenam",{"_index":451,"t":{"891":{"position":[[9,9]]}}}],["filesystem",{"_index":432,"t":{"877":{"position":[[31,10]]}}}],["filter",{"_index":595,"t":{"1134":{"position":[[13,7]]}}}],["final",{"_index":363,"t":{"633":{"position":[[3,5]]}}}],["find",{"_index":471,"t":{"921":{"position":[[26,4]]},"946":{"position":[[14,4]]}}}],["firebas",{"_index":316,"t":{"499":{"position":[[0,8]]}}}],["first",{"_index":391,"t":{"717":{"position":[[0,5]]},"778":{"position":[[0,5]]}}}],["flight",{"_index":569,"t":{"1081":{"position":[[0,6]]}}}],["flow",{"_index":203,"t":{"278":{"position":[[4,4]]},"648":{"position":[[4,4]]}}}],["follow",{"_index":156,"t":{"195":{"position":[[0,6]]}}}],["font",{"_index":246,"t":{"354":{"position":[[0,4]]},"1011":{"position":[[0,4]]}}}],["forbidden",{"_index":445,"t":{"885":{"position":[[38,9]]}}}],["forecast",{"_index":525,"t":{"1005":{"position":[[8,8]]},"1033":{"position":[[8,8]]}}}],["frontend",{"_index":168,"t":{"211":{"position":[[0,8]]}}}],["full",{"_index":419,"t":{"845":{"position":[[10,4]]}}}],["function",{"_index":229,"t":{"321":{"position":[[25,9]]},"325":{"position":[[19,8]]},"327":{"position":[[37,8]]},"342":{"position":[[16,13]]}}}],["furnitur",{"_index":235,"t":{"329":{"position":[[12,9]]}}}],["further",{"_index":360,"t":{"631":{"position":[[3,7]]}}}],["ga",{"_index":562,"t":{"1073":{"position":[[4,3]]}}}],["gazer",{"_index":159,"t":{"199":{"position":[[5,6]]}}}],["gener",{"_index":250,"t":{"358":{"position":[[0,10]]},"1028":{"position":[[0,7]]}}}],["get",{"_index":276,"t":{"429":{"position":[[0,7]]},"969":{"position":[[18,7]]}}}],["getaddrinfo",{"_index":521,"t":{"999":{"position":[[39,11]]}}}],["git",{"_index":201,"t":{"277":{"position":[[0,3]]},"278":{"position":[[0,3]]},"280":{"position":[[0,3]]},"647":{"position":[[0,3]]},"648":{"position":[[0,3]]},"650":{"position":[[0,3]]},"1022":{"position":[[0,3]]}}}],["github",{"_index":573,"t":{"1085":{"position":[[0,6]]},"1087":{"position":[[0,6]]}}}],["glanc",{"_index":612,"t":{"1173":{"position":[[0,7]]}}}],["gluetun",{"_index":604,"t":{"1158":{"position":[[0,7]]}}}],["granular",{"_index":18,"t":{"25":{"position":[[0,8]]}}}],["ground",{"_index":401,"t":{"737":{"position":[[0,6]]},"798":{"position":[[0,6]]}}}],["group",{"_index":42,"t":{"49":{"position":[[7,6]]}}}],["guest",{"_index":16,"t":{"23":{"position":[[9,5]]}}}],["guid",{"_index":214,"t":{"289":{"position":[[6,5]]},"1223":{"position":[[13,5]]}}}],["guidelin",{"_index":208,"t":{"284":{"position":[[3,10]]},"654":{"position":[[3,10]]}}}],["hackernew",{"_index":576,"t":{"1091":{"position":[[0,10]]}}}],["handl",{"_index":627,"t":{"1229":{"position":[[0,8]]}}}],["hard",{"_index":423,"t":{"855":{"position":[[0,4]]}}}],["hash",{"_index":11,"t":{"19":{"position":[[0,4]]}}}],["header",{"_index":46,"t":{"51":{"position":[[5,7]]},"57":{"position":[[0,6]]},"441":{"position":[[8,7]]},"443":{"position":[[8,7]]},"826":{"position":[[15,7]]},"988":{"position":[[13,6]]}}}],["headlin",{"_index":558,"t":{"1067":{"position":[[5,9]]}}}],["health",{"_index":583,"t":{"1108":{"position":[[16,7]]}}}],["healthcheck",{"_index":262,"t":{"393":{"position":[[0,12]]},"395":{"position":[[5,11]]},"980":{"position":[[0,11]]},"1089":{"position":[[0,12]]}}}],["heap",{"_index":487,"t":{"938":{"position":[[31,4]]}}}],["hide",{"_index":234,"t":{"329":{"position":[[0,6]]},"331":{"position":[[36,4]]},"333":{"position":[[54,4]]}}}],["high",{"_index":478,"t":{"936":{"position":[[0,4]]}}}],["histori",{"_index":544,"t":{"1047":{"position":[[19,7]]},"1071":{"position":[[12,7]]},"1110":{"position":[[4,7]]},"1112":{"position":[[7,7]]},"1114":{"position":[[5,7]]},"1181":{"position":[[10,7]]},"1187":{"position":[[13,7]]},"1197":{"position":[[12,7]]}}}],["hole",{"_index":585,"t":{"1116":{"position":[[3,4]]},"1118":{"position":[[3,4]]},"1120":{"position":[[3,4]]},"1122":{"position":[[3,4]]},"1124":{"position":[[3,4]]},"1126":{"position":[[3,4]]}}}],["holiday",{"_index":553,"t":{"1061":{"position":[[7,8]]}}}],["home",{"_index":176,"t":{"233":{"position":[[0,4]]},"364":{"position":[[0,4]]},"700":{"position":[[0,4]]},"707":{"position":[[4,4]]},"761":{"position":[[0,4]]},"768":{"position":[[4,4]]},"914":{"position":[[29,4]]},"1132":{"position":[[8,4]]},"1134":{"position":[[8,4]]},"1136":{"position":[[8,4]]},"1138":{"position":[[8,4]]}}}],["homelab",{"_index":387,"t":{"713":{"position":[[0,7]]},"735":{"position":[[0,7]]},"745":{"position":[[0,7]]},"749":{"position":[[12,7]]},"774":{"position":[[0,7]]},"796":{"position":[[0,7]]},"806":{"position":[[0,7]]},"810":{"position":[[12,7]]}}}],["host",{"_index":70,"t":{"85":{"position":[[6,7]]},"99":{"position":[[5,7]]},"251":{"position":[[0,7]]},"499":{"position":[[9,7]]},"717":{"position":[[19,7]]},"778":{"position":[[19,7]]},"887":{"position":[[52,5]]},"912":{"position":[[14,7]]},"918":{"position":[[6,6]]},"988":{"position":[[8,4]]},"1001":{"position":[[22,4]]},"1105":{"position":[[5,6]]}}}],["hotkey",{"_index":373,"t":{"678":{"position":[[7,7]]}}}],["html",{"_index":449,"t":{"889":{"position":[[36,4]]},"1214":{"position":[[0,4]]}}}],["http",{"_index":24,"t":{"31":{"position":[[7,4]]},"35":{"position":[[0,4]]},"395":{"position":[[0,4]]},"1001":{"position":[[55,7],[66,8]]}}}],["hugalafutro",{"_index":383,"t":{"705":{"position":[[0,11]]},"766":{"position":[[0,11]]}}}],["icon",{"_index":249,"t":{"356":{"position":[[7,5]]},"358":{"position":[[11,5]]},"360":{"position":[[6,5]]},"362":{"position":[[9,5]]},"364":{"position":[[9,5]]},"366":{"position":[[16,5]]},"368":{"position":[[0,5]]},"370":{"position":[[6,5]]},"372":{"position":[[8,4]]},"374":{"position":[[3,4]]},"376":{"position":[[0,4]]},"572":{"position":[[0,5]]},"1011":{"position":[[13,5]]}}}],["id",{"_index":500,"t":{"953":{"position":[[43,3]]}}}],["ifram",{"_index":623,"t":{"1212":{"position":[[0,6]]}}}],["ignor",{"_index":475,"t":{"925":{"position":[[9,7]]},"1235":{"position":[[0,8]]}}}],["imag",{"_index":307,"t":{"481":{"position":[[7,5]]},"971":{"position":[[9,5]]},"978":{"position":[[4,5]]},"1037":{"position":[[0,5]]}}}],["improv",{"_index":141,"t":{"183":{"position":[[0,7]]}}}],["inaccur",{"_index":528,"t":{"1007":{"position":[[18,10]]}}}],["includ",{"_index":407,"t":{"753":{"position":[[8,7]]},"814":{"position":[[8,7]]}}}],["incorrectli",{"_index":523,"t":{"1003":{"position":[[19,11]]}}}],["index.j",{"_index":83,"t":{"104":{"position":[[9,8]]}}}],["indic",{"_index":409,"t":{"820":{"position":[[16,10]]}}}],["ineffect",{"_index":483,"t":{"938":{"position":[[0,11]]}}}],["info",{"_index":212,"t":{"288":{"position":[[4,4]]},"1106":{"position":[[7,4]]},"1136":{"position":[[17,4]]},"1158":{"position":[[12,4]]}}}],["initi",{"_index":344,"t":{"584":{"position":[[0,14]]}}}],["insid",{"_index":334,"t":{"552":{"position":[[6,6]]},"934":{"position":[[17,6]]}}}],["instal",{"_index":357,"t":{"625":{"position":[[3,12]]}}}],["integr",{"_index":347,"t":{"595":{"position":[[12,9]]}}}],["intend",{"_index":351,"t":{"607":{"position":[[0,8]]}}}],["intention",{"_index":460,"t":{"899":{"position":[[0,13]]}}}],["inter",{"_index":300,"t":{"473":{"position":[[8,5]]}}}],["interfac",{"_index":618,"t":{"1199":{"position":[[8,10]]}}}],["invalid",{"_index":450,"t":{"891":{"position":[[0,8]]},"988":{"position":[[0,7]]}}}],["invalid_redirect_uri",{"_index":502,"t":{"957":{"position":[[0,20]]}}}],["io",{"_index":617,"t":{"1191":{"position":[[5,2]]}}}],["ip",{"_index":62,"t":{"79":{"position":[[0,2]]},"1009":{"position":[[7,2]]},"1039":{"position":[[7,2]]},"1041":{"position":[[0,2]]},"1205":{"position":[[0,2]]}}}],["ipinfo",{"_index":530,"t":{"1009":{"position":[[33,6]]}}}],["ipqueri",{"_index":531,"t":{"1009":{"position":[[43,7]]}}}],["issu",{"_index":356,"t":{"619":{"position":[[21,5]]},"662":{"position":[[0,5]]},"911":{"position":[[14,6]]},"971":{"position":[[15,6]]},"1251":{"position":[[11,5]]}}}],["item",{"_index":333,"t":{"552":{"position":[[0,5]]},"554":{"position":[[4,5]]},"855":{"position":[[23,4]]}}}],["item.displaydata",{"_index":107,"t":{"146":{"position":[[0,16]]}}}],["item.displaydata.hideforkeycloakus",{"_index":114,"t":{"154":{"position":[[84,37]]}}}],["item.displaydata.showforkeycloakus",{"_index":115,"t":{"154":{"position":[[126,37]]}}}],["join",{"_index":144,"t":{"187":{"position":[[0,4]]}}}],["joke",{"_index":563,"t":{"1075":{"position":[[0,4]]}}}],["keep",{"_index":289,"t":{"461":{"position":[[0,4]]}}}],["keycloak",{"_index":33,"t":{"41":{"position":[[0,8]]},"43":{"position":[[10,8]]},"45":{"position":[[9,8]]},"47":{"position":[[10,8]]},"55":{"position":[[16,8]]},"951":{"position":[[0,8]]},"953":{"position":[[8,8]]}}}],["known",{"_index":222,"t":{"299":{"position":[[0,5]]},"615":{"position":[[0,5]]}}}],["kubernet",{"_index":174,"t":{"229":{"position":[[0,10]]},"879":{"position":[[0,10]]}}}],["kuma",{"_index":609,"t":{"1166":{"position":[[7,4]]},"1168":{"position":[[7,4]]}}}],["lab",{"_index":252,"t":{"364":{"position":[[5,3]]},"700":{"position":[[5,3]]},"761":{"position":[[5,3]]}}}],["label",{"_index":630,"t":{"1233":{"position":[[7,6]]}}}],["lair",{"_index":386,"t":{"711":{"position":[[12,4]]},"772":{"position":[[12,4]]}}}],["lambda",{"_index":230,"t":{"325":{"position":[[12,6]]}}}],["languag",{"_index":227,"t":{"311":{"position":[[16,8]]},"509":{"position":[[8,8]]},"515":{"position":[[10,9]]},"517":{"position":[[13,8]]},"519":{"position":[[16,8]]},"1241":{"position":[[0,8]]}}}],["launch",{"_index":372,"t":{"674":{"position":[[0,9]]},"914":{"position":[[10,6]]}}}],["layout",{"_index":330,"t":{"548":{"position":[[0,6]]},"925":{"position":[[28,6]]}}}],["leav",{"_index":154,"t":{"193":{"position":[[16,5]]},"965":{"position":[[9,6]]}}}],["level",{"_index":92,"t":{"118":{"position":[[4,5]]},"479":{"position":[[16,5]]},"863":{"position":[[4,5]]}}}],["licens",{"_index":259,"t":{"381":{"position":[[4,7]]}}}],["lighthous",{"_index":220,"t":{"294":{"position":[[14,10]]}}}],["lighthttpd",{"_index":464,"t":{"909":{"position":[[0,10]]}}}],["limit",{"_index":295,"t":{"469":{"position":[[0,5]]},"615":{"position":[[6,11]]},"938":{"position":[[36,5]]}}}],["linkd",{"_index":607,"t":{"1164":{"position":[[0,8]]}}}],["list",{"_index":541,"t":{"1045":{"position":[[13,4]]},"1097":{"position":[[7,4]]},"1154":{"position":[[8,5]]}}}],["live",{"_index":395,"t":{"725":{"position":[[6,4]]},"786":{"position":[[6,4]]}}}],["load",{"_index":421,"t":{"853":{"position":[[0,7]]},"929":{"position":[[18,7]]},"1114":{"position":[[0,4]]},"1193":{"position":[[7,4]]},"1197":{"position":[[7,4]]}}}],["local",{"_index":256,"t":{"370":{"position":[[0,5]]},"540":{"position":[[6,5]]},"566":{"position":[[0,5]]},"1016":{"position":[[13,5]]}}}],["lockdown",{"_index":348,"t":{"601":{"position":[[14,8]]}}}],["log",{"_index":13,"t":{"21":{"position":[[0,7]]},"397":{"position":[[0,4]]},"398":{"position":[[10,4]]},"404":{"position":[[9,7]]},"479":{"position":[[8,7]]},"961":{"position":[[0,6]]}}}],["login",{"_index":501,"t":{"955":{"position":[[20,5]]},"959":{"position":[[0,5]]},"963":{"position":[[0,5]]},"982":{"position":[[7,5]]}}}],["loop",{"_index":457,"t":{"897":{"position":[[29,4]]},"955":{"position":[[9,4]]}}}],["make",{"_index":128,"t":{"173":{"position":[[0,4]]},"550":{"position":[[0,6]]},"1018":{"position":[[7,4]]}}}],["manag",{"_index":264,"t":{"402":{"position":[[0,10]]},"437":{"position":[[0,8]]},"662":{"position":[[6,10]]},"732":{"position":[[8,10]]},"793":{"position":[[8,10]]}}}],["manager/sav",{"_index":447,"t":{"889":{"position":[[8,12]]}}}],["mark",{"_index":484,"t":{"938":{"position":[[12,4]]}}}],["materi",{"_index":253,"t":{"366":{"position":[[0,8]]}}}],["memori",{"_index":476,"t":{"931":{"position":[[8,6]]},"1112":{"position":[[0,6]]},"1183":{"position":[[8,6]]},"1185":{"position":[[8,6]]},"1187":{"position":[[0,6]]}}}],["metadata",{"_index":326,"t":{"536":{"position":[[5,8]]}}}],["metal",{"_index":192,"t":{"258":{"position":[[5,5]]}}}],["method",{"_index":5,"t":{"11":{"position":[[8,7]]},"71":{"position":[[27,7]]},"161":{"position":[[14,7]]},"219":{"position":[[11,7]]},"635":{"position":[[23,6]]},"637":{"position":[[23,6]]},"639":{"position":[[23,6]]},"686":{"position":[[16,6]]}}}],["metric",{"_index":621,"t":{"1209":{"position":[[8,7]]}}}],["minecraft",{"_index":581,"t":{"1103":{"position":[[0,9]]}}}],["minim",{"_index":3,"t":{"9":{"position":[[0,7]]}}}],["minut",{"_index":124,"t":{"169":{"position":[[9,6]]}}}],["mismatch",{"_index":513,"t":{"974":{"position":[[11,8]]}}}],["miss",{"_index":442,"t":{"885":{"position":[[15,7]]},"923":{"position":[[9,7]]}}}],["mit",{"_index":258,"t":{"381":{"position":[[0,3]]}}}],["mndashboard",{"_index":378,"t":{"698":{"position":[[0,11]]},"759":{"position":[[0,11]]}}}],["mobil",{"_index":465,"t":{"914":{"position":[[22,6]]}}}],["modal",{"_index":463,"t":{"901":{"position":[[22,5]]}}}],["mode",{"_index":200,"t":{"275":{"position":[[12,5]]},"899":{"position":[[24,4]]}}}],["model",{"_index":350,"t":{"605":{"position":[[7,5]]}}}],["modifi",{"_index":318,"t":{"503":{"position":[[10,8]]},"847":{"position":[[0,9]]}}}],["modul",{"_index":313,"t":{"489":{"position":[[9,7]]},"946":{"position":[[19,6]]}}}],["monitor",{"_index":266,"t":{"404":{"position":[[21,10]]},"727":{"position":[[7,7]]},"788":{"position":[[7,7]]},"1043":{"position":[[7,7]]},"1108":{"position":[[5,10]]},"1172":{"position":[[16,10]]}}}],["more",{"_index":157,"t":{"195":{"position":[[11,4]]}}}],["morn",{"_index":405,"t":{"747":{"position":[[0,7]]},"808":{"position":[[0,7]]}}}],["mount",{"_index":436,"t":{"879":{"position":[[21,5]]},"974":{"position":[[0,5]]}}}],["mullvad",{"_index":548,"t":{"1053":{"position":[[0,7]]}}}],["multi",{"_index":327,"t":{"538":{"position":[[0,5]]},"916":{"position":[[7,5]]}}}],["mvg",{"_index":577,"t":{"1093":{"position":[[0,3]]},"1095":{"position":[[0,3]]}}}],["na",{"_index":179,"t":{"235":{"position":[[9,3]]},"707":{"position":[[0,3]]},"768":{"position":[[0,3]]}}}],["name",{"_index":205,"t":{"280":{"position":[[11,6]]},"305":{"position":[[13,4]]},"331":{"position":[[17,4]]},"650":{"position":[[11,6]]}}}],["nav",{"_index":472,"t":{"923":{"position":[[22,4]]}}}],["navig",{"_index":371,"t":{"672":{"position":[[0,10]]}}}],["near",{"_index":486,"t":{"938":{"position":[[26,4]]}}}],["netdata",{"_index":584,"t":{"1110":{"position":[[12,9]]},"1112":{"position":[[15,9]]},"1114":{"position":[[13,9]]}}}],["netlifi",{"_index":183,"t":{"241":{"position":[[0,7]]},"321":{"position":[[11,7]]},"323":{"position":[[7,7]]},"637":{"position":[[34,7]]},"887":{"position":[[28,7]]}}}],["network",{"_index":279,"t":{"435":{"position":[[0,7]]},"723":{"position":[[0,10]]},"784":{"position":[[0,10]]},"1199":{"position":[[0,7]]},"1201":{"position":[[0,7]]}}}],["new",{"_index":224,"t":{"303":{"position":[[11,3]]},"311":{"position":[[12,3]]},"317":{"position":[[9,3]]},"471":{"position":[[8,3]]},"517":{"position":[[9,3]]},"519":{"position":[[12,3]]},"527":{"position":[[7,3]]},"1067":{"position":[[0,4]]}}}],["newest",{"_index":163,"t":{"204":{"position":[[0,6]]}}}],["nextcloud",{"_index":596,"t":{"1140":{"position":[[0,9]]},"1142":{"position":[[0,9]]},"1144":{"position":[[0,9]]},"1146":{"position":[[0,9]]},"1148":{"position":[[0,9]]},"1150":{"position":[[0,9]]}}}],["nginx",{"_index":288,"t":{"457":{"position":[[6,5]]},"493":{"position":[[0,5]]},"903":{"position":[[0,5]]}}}],["ngrok",{"_index":517,"t":{"988":{"position":[[42,5]]}}}],["node",{"_index":232,"t":{"327":{"position":[[16,4]]},"942":{"position":[[0,4]]}}}],["non",{"_index":429,"t":{"867":{"position":[[0,3]]}}}],["norri",{"_index":565,"t":{"1077":{"position":[[6,6]]}}}],["note",{"_index":52,"t":{"63":{"position":[[0,5]]},"156":{"position":[[0,5]]},"298":{"position":[[0,5]]},"378":{"position":[[0,5]]},"633":{"position":[[9,4]]}}}],["notif",{"_index":597,"t":{"1144":{"position":[[10,13]]}}}],["ntfi",{"_index":600,"t":{"1152":{"position":[[0,4]]}}}],["numer",{"_index":498,"t":{"953":{"position":[[28,7]]},"969":{"position":[[0,7]]}}}],["oauth",{"_index":67,"t":{"83":{"position":[[6,5]]}}}],["object",{"_index":496,"t":{"949":{"position":[[34,7]]}}}],["oidc",{"_index":53,"t":{"65":{"position":[[0,4]]},"948":{"position":[[7,4]]},"953":{"position":[[0,4]]},"963":{"position":[[50,5]]}}}],["old",{"_index":454,"t":{"895":{"position":[[40,3]]},"978":{"position":[[0,3]]}}}],["on",{"_index":132,"t":{"175":{"position":[[20,3]]}}}],["opcach",{"_index":599,"t":{"1150":{"position":[[14,7]]}}}],["open",{"_index":4,"t":{"11":{"position":[[0,7]]},"686":{"position":[[8,7]]},"692":{"position":[[0,7]]},"923":{"position":[[36,4]]},"1020":{"position":[[7,4]]}}}],["option",{"_index":44,"t":{"49":{"position":[[24,10]]},"122":{"position":[[18,10]]},"124":{"position":[[8,10]]},"126":{"position":[[10,10]]},"128":{"position":[[15,10]]},"130":{"position":[[21,10]]},"132":{"position":[[24,10]]},"134":{"position":[[26,10]]},"136":{"position":[[20,10]]},"138":{"position":[[20,10]]},"140":{"position":[[25,10]]},"146":{"position":[[17,10]]},"148":{"position":[[16,10]]},"150":{"position":[[20,10]]},"317":{"position":[[13,6]]},"843":{"position":[[0,6]]},"845":{"position":[[0,6]]},"1243":{"position":[[10,7]]}}}],["out",{"_index":14,"t":{"21":{"position":[[15,3]]},"965":{"position":[[5,3]]}}}],["over",{"_index":160,"t":{"199":{"position":[[12,4]]}}}],["overview",{"_index":580,"t":{"1101":{"position":[[11,8]]}}}],["ownership",{"_index":261,"t":{"389":{"position":[[5,9]]}}}],["page",{"_index":96,"t":{"124":{"position":[[0,7]]},"247":{"position":[[8,5]]},"329":{"position":[[7,4]]},"536":{"position":[[0,4]]},"538":{"position":[[6,4]]},"540":{"position":[[16,5]]},"542":{"position":[[17,5]]},"584":{"position":[[15,4]]},"851":{"position":[[0,4]]},"891":{"position":[[37,4]]},"916":{"position":[[13,4]]},"920":{"position":[[4,5]]},"921":{"position":[[4,4]]},"923":{"position":[[4,4]]},"925":{"position":[[4,4]]},"1168":{"position":[[19,4]]}}}],["pageinfo",{"_index":94,"t":{"120":{"position":[[0,8]]}}}],["pageinfo.navlink",{"_index":95,"t":{"122":{"position":[[0,17]]}}}],["pass",{"_index":278,"t":{"431":{"position":[[0,7]]},"439":{"position":[[0,7]]}}}],["password",{"_index":12,"t":{"19":{"position":[[5,8]]},"29":{"position":[[32,9]]}}}],["patch",{"_index":355,"t":{"617":{"position":[[9,5]]}}}],["path",{"_index":467,"t":{"918":{"position":[[22,4]]}}}],["per",{"_index":614,"t":{"1179":{"position":[[10,3]]}}}],["perform",{"_index":219,"t":{"294":{"position":[[0,11]]},"397":{"position":[[9,11]]},"400":{"position":[[10,11]]}}}],["permiss",{"_index":19,"t":{"27":{"position":[[0,11]]},"348":{"position":[[18,11]]},"389":{"position":[[19,11]]},"877":{"position":[[0,10]]}}}],["php",{"_index":598,"t":{"1150":{"position":[[10,3]]}}}],["pi",{"_index":390,"t":{"715":{"position":[[10,2]]},"776":{"position":[[10,2]]},"1116":{"position":[[0,2]]},"1118":{"position":[[0,2]]},"1120":{"position":[[0,2]]},"1122":{"position":[[0,2]]},"1124":{"position":[[0,2]]},"1126":{"position":[[0,2]]}}}],["pictur",{"_index":571,"t":{"1083":{"position":[[10,7]]}}}],["ping",{"_index":590,"t":{"1128":{"position":[[5,4]]}}}],["pipelin",{"_index":370,"t":{"668":{"position":[[8,8]]}}}],["platform",{"_index":177,"t":{"233":{"position":[[12,9]]}}}],["play",{"_index":187,"t":{"249":{"position":[[0,4]]}}}],["podman",{"_index":172,"t":{"225":{"position":[[0,6]]}}}],["polici",{"_index":76,"t":{"97":{"position":[[9,6]]},"617":{"position":[[15,6]]}}}],["popul",{"_index":80,"t":{"102":{"position":[[0,8]]}}}],["portain",{"_index":173,"t":{"227":{"position":[[0,9]]}}}],["pr",{"_index":140,"t":{"181":{"position":[[9,2]]},"284":{"position":[[0,2]]},"654":{"position":[[0,2]]}}}],["prerequisit",{"_index":196,"t":{"267":{"position":[[0,13]]},"338":{"position":[[9,13]]},"623":{"position":[[3,13]]}}}],["prevent",{"_index":120,"t":{"163":{"position":[[0,10]]},"471":{"position":[[0,7]]}}}],["price",{"_index":543,"t":{"1047":{"position":[[13,5]]},"1071":{"position":[[6,5]]},"1073":{"position":[[8,6]]}}}],["privaci",{"_index":74,"t":{"95":{"position":[[0,7]]}}}],["privat",{"_index":394,"t":{"721":{"position":[[4,7]]},"782":{"position":[[4,7]]}}}],["privileg",{"_index":297,"t":{"471":{"position":[[12,10]]}}}],["process",{"_index":365,"t":{"645":{"position":[[11,7]]}}}],["profil",{"_index":575,"t":{"1087":{"position":[[7,7]]}}}],["project",{"_index":198,"t":{"269":{"position":[[12,7]]},"271":{"position":[[0,7]]},"732":{"position":[[0,7]]},"793":{"position":[[0,7]]}}}],["prometheu",{"_index":626,"t":{"1218":{"position":[[0,10]]}}}],["provid",{"_index":68,"t":{"83":{"position":[[12,9]]},"85":{"position":[[14,9]]},"387":{"position":[[0,9]]},"1009":{"position":[[51,9]]}}}],["proxi",{"_index":57,"t":{"73":{"position":[[8,5]]},"999":{"position":[[5,5]]},"1001":{"position":[[5,5]]},"1227":{"position":[[0,8]]}}}],["proxmox",{"_index":602,"t":{"1154":{"position":[[0,7]]}}}],["public",{"_index":529,"t":{"1009":{"position":[[0,6]]},"1039":{"position":[[0,6]]},"1061":{"position":[[0,6]]}}}],["pull",{"_index":309,"t":{"481":{"position":[[20,7]]},"978":{"position":[[23,4]]}}}],["queri",{"_index":587,"t":{"1120":{"position":[[8,7]]},"1122":{"position":[[8,7]]}}}],["quick",{"_index":78,"t":{"100":{"position":[[0,5]]},"217":{"position":[[0,5]]}}}],["quot",{"_index":566,"t":{"1077":{"position":[[13,6]]}}}],["quota",{"_index":291,"t":{"463":{"position":[[13,6]]}}}],["rais",{"_index":143,"t":{"185":{"position":[[0,5]]},"1251":{"position":[[0,7]]}}}],["ram",{"_index":480,"t":{"936":{"position":[[12,3]]}}}],["raspberri",{"_index":389,"t":{"715":{"position":[[0,9]]},"776":{"position":[[0,9]]}}}],["rate",{"_index":552,"t":{"1059":{"position":[[9,5]]}}}],["ratty222",{"_index":382,"t":{"703":{"position":[[0,8]]},"764":{"position":[[0,8]]}}}],["read",{"_index":305,"t":{"477":{"position":[[4,4]]},"877":{"position":[[21,4]]},"879":{"position":[[30,4]]},"899":{"position":[[14,4]]}}}],["recent",{"_index":588,"t":{"1124":{"position":[[8,6]]},"1126":{"position":[[8,6]]}}}],["recommend",{"_index":30,"t":{"37":{"position":[[24,13]]}}}],["redirect",{"_index":231,"t":{"327":{"position":[[3,8]]},"951":{"position":[[9,8]]},"955":{"position":[[0,8]]}}}],["refer",{"_index":534,"t":{"1015":{"position":[[9,9]]}}}],["refus",{"_index":461,"t":{"901":{"position":[[0,7]]},"959":{"position":[[45,7]]}}}],["regist",{"_index":243,"t":{"344":{"position":[[9,8]]}}}],["registri",{"_index":312,"t":{"487":{"position":[[0,8]]}}}],["releas",{"_index":364,"t":{"643":{"position":[[0,7]]},"660":{"position":[[0,8]]},"668":{"position":[[0,7]]}}}],["remot",{"_index":281,"t":{"445":{"position":[[0,6]]},"542":{"position":[[6,6]]},"929":{"position":[[0,6]]}}}],["report",{"_index":137,"t":{"177":{"position":[[21,7]]},"586":{"position":[[16,9]]},"619":{"position":[[0,9]]},"1018":{"position":[[18,6]]}}}],["request",{"_index":146,"t":{"189":{"position":[[0,7]]},"570":{"position":[[9,8]]},"1227":{"position":[[9,8]]},"1247":{"position":[[0,10]]}}}],["requir",{"_index":189,"t":{"253":{"position":[[0,12]]},"254":{"position":[[7,12]]}}}],["rescuetim",{"_index":579,"t":{"1101":{"position":[[0,10]]}}}],["reset",{"_index":535,"t":{"1016":{"position":[[7,5]]}}}],["resourc",{"_index":209,"t":{"286":{"position":[[0,9]]},"376":{"position":[[21,9]]},"463":{"position":[[4,8]]},"1172":{"position":[[7,8]]},"1203":{"position":[[0,8]]}}}],["respect",{"_index":244,"t":{"348":{"position":[[0,10]]}}}],["respons",{"_index":625,"t":{"1216":{"position":[[4,8]]}}}],["restart",{"_index":456,"t":{"897":{"position":[[21,7]]}}}],["restor",{"_index":73,"t":{"93":{"position":[[0,9]]}}}],["restrict",{"_index":329,"t":{"544":{"position":[[0,12]]}}}],["return",{"_index":443,"t":{"885":{"position":[[26,7]]},"889":{"position":[[21,7]]},"927":{"position":[[17,6]]},"963":{"position":[[71,8]]}}}],["revers",{"_index":56,"t":{"73":{"position":[[0,7]]},"449":{"position":[[0,7]]}}}],["review",{"_index":155,"t":{"193":{"position":[[24,6]]}}}],["right",{"_index":353,"t":{"613":{"position":[[22,5]]}}}],["rmm",{"_index":611,"t":{"1170":{"position":[[9,3]]}}}],["role",{"_index":43,"t":{"49":{"position":[[18,5]]}}}],["root",{"_index":293,"t":{"465":{"position":[[13,4]]}}}],["rout",{"_index":237,"t":{"329":{"position":[[33,6]]},"331":{"position":[[11,5]]},"911":{"position":[[6,7]]}}}],["rss",{"_index":537,"t":{"1035":{"position":[[0,3]]}}}],["run",{"_index":197,"t":{"269":{"position":[[0,7]]},"323":{"position":[[3,3]]},"391":{"position":[[0,7]]},"465":{"position":[[6,3]]},"503":{"position":[[0,7]]},"988":{"position":[[26,7]]}}}],["sabnzbd",{"_index":603,"t":{"1156":{"position":[[0,7]]}}}],["sass",{"_index":491,"t":{"942":{"position":[[5,4]]}}}],["save",{"_index":119,"t":{"161":{"position":[[7,6]]},"875":{"position":[[11,6]]},"883":{"position":[[21,4]]},"885":{"position":[[0,4]]},"887":{"position":[[0,4]]},"891":{"position":[[24,6]]},"893":{"position":[[8,4]]},"895":{"position":[[0,5]]},"897":{"position":[[40,6]]},"959":{"position":[[56,4]]}}}],["scan",{"_index":311,"t":{"485":{"position":[[19,8]]}}}],["schedul",{"_index":273,"t":{"423":{"position":[[0,10]]},"643":{"position":[[8,8]]}}}],["score",{"_index":557,"t":{"1065":{"position":[[7,6]]}}}],["screen",{"_index":466,"t":{"914":{"position":[[34,6]]}}}],["search",{"_index":343,"t":{"582":{"position":[[4,6]]},"670":{"position":[[0,9]]},"680":{"position":[[4,6]]},"682":{"position":[[8,6]]},"684":{"position":[[13,6]]},"690":{"position":[[14,6]]},"694":{"position":[[9,6]]},"1099":{"position":[[7,6]]}}}],["secret",{"_index":628,"t":{"1229":{"position":[[9,7]]}}}],["section",{"_index":105,"t":{"142":{"position":[[0,7]]},"550":{"position":[[9,7]]},"552":{"position":[[15,7]]},"855":{"position":[[12,7]]}}}],["section.displaydata",{"_index":109,"t":{"150":{"position":[[0,19]]}}}],["section.displaydata.hideforkeycloakus",{"_index":112,"t":{"154":{"position":[[0,41]]}}}],["section.displaydata.showforkeycloakus",{"_index":113,"t":{"154":{"position":[[42,41]]}}}],["section.icon",{"_index":110,"t":{"152":{"position":[[0,12]]}}}],["section.item",{"_index":106,"t":{"144":{"position":[[0,12]]}}}],["section.item.icon",{"_index":111,"t":{"152":{"position":[[17,17]]}}}],["section.widget",{"_index":108,"t":{"148":{"position":[[0,15]]}}}],["secur",{"_index":26,"t":{"33":{"position":[[0,8]]},"95":{"position":[[10,8]]},"459":{"position":[[10,8]]},"485":{"position":[[10,8]]},"487":{"position":[[9,8]]},"489":{"position":[[0,8]]},"592":{"position":[[0,8]]},"594":{"position":[[0,8]]},"619":{"position":[[12,8]]},"828":{"position":[[10,8]]}}}],["self",{"_index":77,"t":{"99":{"position":[[0,4]]},"429":{"position":[[10,4]]},"431":{"position":[[10,4]]},"717":{"position":[[14,4]]},"778":{"position":[[14,4]]},"1105":{"position":[[0,4]]}}}],["selfh.st",{"_index":251,"t":{"362":{"position":[[0,8]]}}}],["selinux",{"_index":437,"t":{"881":{"position":[[0,7]]}}}],["send",{"_index":131,"t":{"175":{"position":[[13,4]]}}}],["server",{"_index":47,"t":{"53":{"position":[[4,6]]},"67":{"position":[[4,6]]},"81":{"position":[[4,6]]},"99":{"position":[[24,6]]},"233":{"position":[[5,6]]},"323":{"position":[[19,6]]},"491":{"position":[[4,6]]},"1103":{"position":[[10,6]]}}}],["servic",{"_index":182,"t":{"239":{"position":[[16,7]]},"639":{"position":[[40,8]]},"723":{"position":[[11,8]]},"784":{"position":[[11,8]]},"1105":{"position":[[12,8]]}}}],["session",{"_index":337,"t":{"564":{"position":[[0,7]]}}}],["set",{"_index":8,"t":{"17":{"position":[[0,7]]},"266":{"position":[[0,7]]},"441":{"position":[[0,7]]},"463":{"position":[[0,3]]},"479":{"position":[[0,3]]},"509":{"position":[[0,7]]},"682":{"position":[[0,7]]},"686":{"position":[[0,7]]},"826":{"position":[[0,7]]},"849":{"position":[[0,7]]},"1016":{"position":[[19,8]]},"1231":{"position":[[0,7]]}}}],["setup",{"_index":37,"t":{"45":{"position":[[3,5]]}}}],["share",{"_index":126,"t":{"171":{"position":[[0,5]]}}}],["show",{"_index":453,"t":{"895":{"position":[[30,5]]},"921":{"position":[[9,5]]},"1003":{"position":[[7,5]]}}}],["side",{"_index":48,"t":{"53":{"position":[[11,4]]},"67":{"position":[[11,4]]}}}],["sigkil",{"_index":490,"t":{"940":{"position":[[27,9]]}}}],["sign",{"_index":277,"t":{"429":{"position":[[15,6]]},"431":{"position":[[15,6]]},"965":{"position":[[0,4]]}}}],["signal",{"_index":489,"t":{"940":{"position":[[20,6]]}}}],["signincallback",{"_index":506,"t":{"963":{"position":[[56,14]]}}}],["simpl",{"_index":248,"t":{"356":{"position":[[0,6]]}}}],["small",{"_index":129,"t":{"173":{"position":[[7,5]]},"175":{"position":[[28,5]]}}}],["socket",{"_index":304,"t":{"475":{"position":[[31,6]]}}}],["sourc",{"_index":181,"t":{"237":{"position":[[11,6]]},"414":{"position":[[20,6]]},"635":{"position":[[39,6]]}}}],["space",{"_index":616,"t":{"1189":{"position":[[5,5]]}}}],["specif",{"_index":272,"t":{"421":{"position":[[6,8]]},"851":{"position":[[5,8]]}}}],["specifi",{"_index":294,"t":{"467":{"position":[[0,7]]},"483":{"position":[[0,7]]}}}],["speedomet",{"_index":613,"t":{"1177":{"position":[[18,11]]},"1185":{"position":[[21,11]]}}}],["sponsor",{"_index":162,"t":{"202":{"position":[[0,8]]}}}],["sport",{"_index":556,"t":{"1065":{"position":[[0,6]]}}}],["spread",{"_index":150,"t":{"191":{"position":[[0,6]]}}}],["ssh",{"_index":283,"t":{"449":{"position":[[8,3]]}}}],["ssl",{"_index":274,"t":{"425":{"position":[[0,3]]},"427":{"position":[[5,3]]},"429":{"position":[[22,3]]},"597":{"position":[[0,3]]}}}],["sso",{"_index":65,"t":{"83":{"position":[[0,3]]}}}],["star",{"_index":152,"t":{"193":{"position":[[0,5]]},"199":{"position":[[0,4]]}}}],["stargaz",{"_index":164,"t":{"204":{"position":[[7,10]]}}}],["start",{"_index":79,"t":{"100":{"position":[[6,5]]},"217":{"position":[[6,5]]},"406":{"position":[[5,8]]},"972":{"position":[[8,8]]}}}],["startpag",{"_index":396,"t":{"729":{"position":[[8,9]]},"790":{"position":[[8,9]]}}}],["startup",{"_index":482,"t":{"936":{"position":[[25,7]]}}}],["stat",{"_index":547,"t":{"1051":{"position":[[5,5]]},"1087":{"position":[[15,5]]},"1116":{"position":[[8,5]]},"1118":{"position":[[8,5]]},"1128":{"position":[[0,4]]},"1132":{"position":[[19,5]]},"1148":{"position":[[10,5]]},"1150":{"position":[[22,5]]}}}],["static",{"_index":31,"t":{"39":{"position":[[6,6]]},"887":{"position":[[45,6]]},"912":{"position":[[7,6]]}}}],["station",{"_index":593,"t":{"1130":{"position":[[18,7]]}}}],["statu",{"_index":342,"t":{"576":{"position":[[0,6]]},"820":{"position":[[9,6]]},"830":{"position":[[21,6]]},"832":{"position":[[24,6]]},"992":{"position":[[0,6]]},"1053":{"position":[[8,6]]},"1063":{"position":[[9,6]]},"1069":{"position":[[4,6]]},"1089":{"position":[[13,6]]},"1168":{"position":[[12,6]]}}}],["status",{"_index":591,"t":{"1128":{"position":[[10,8]]},"1142":{"position":[[15,8]]}}}],["stefantigro",{"_index":404,"t":{"743":{"position":[[0,11]]},"804":{"position":[[0,11]]}}}],["step",{"_index":241,"t":{"338":{"position":[[0,4]]},"340":{"position":[[0,4]]},"342":{"position":[[0,4]]},"344":{"position":[[0,4]]},"346":{"position":[[0,4]]},"883":{"position":[[7,4]]}}}],["stock",{"_index":560,"t":{"1071":{"position":[[0,5]]}}}],["storag",{"_index":335,"t":{"560":{"position":[[8,7]]},"564":{"position":[[8,7]]},"566":{"position":[[6,7]]}}}],["store",{"_index":339,"t":{"568":{"position":[[9,6]]}}}],["strategi",{"_index":202,"t":{"277":{"position":[[4,8]]},"647":{"position":[[4,8]]}}}],["stream",{"_index":601,"t":{"1152":{"position":[[5,6]]}}}],["structur",{"_index":216,"t":{"291":{"position":[[12,9]]},"333":{"position":[[30,10]]},"546":{"position":[[4,9]]}}}],["stuck",{"_index":507,"t":{"965":{"position":[[20,5]]}}}],["style",{"_index":213,"t":{"289":{"position":[[0,5]]},"307":{"position":[[14,6]]},"851":{"position":[[14,6]]},"984":{"position":[[0,6]]},"1237":{"position":[[7,7]]}}}],["stylesheet",{"_index":422,"t":{"853":{"position":[[17,11]]}}}],["sub",{"_index":328,"t":{"540":{"position":[[12,3]]},"542":{"position":[[13,3]]},"554":{"position":[[0,3]]},"891":{"position":[[33,3]]},"918":{"position":[[18,3]]},"920":{"position":[[0,3]]},"921":{"position":[[0,3]]},"923":{"position":[[0,3]]},"925":{"position":[[0,3]]},"927":{"position":[[0,3]]}}}],["submit",{"_index":139,"t":{"181":{"position":[[0,6]]},"750":{"position":[[0,10]]},"751":{"position":[[7,6]]},"811":{"position":[[0,10]]},"812":{"position":[[7,6]]}}}],["subresourc",{"_index":346,"t":{"595":{"position":[[0,11]]}}}],["successfulli",{"_index":452,"t":{"895":{"position":[[6,12]]}}}],["support",{"_index":194,"t":{"262":{"position":[[8,7]]},"538":{"position":[[11,7]]},"942":{"position":[[23,7]]}}}],["survey",{"_index":125,"t":{"169":{"position":[[16,6]]}}}],["switch",{"_index":417,"t":{"840":{"position":[[10,9]]}}}],["synolog",{"_index":178,"t":{"235":{"position":[[0,8]]},"1130":{"position":[[0,8]]}}}],["system",{"_index":190,"t":{"254":{"position":[[0,6]]},"406":{"position":[[17,6]]},"727":{"position":[[0,6]]},"788":{"position":[[0,6]]},"1106":{"position":[[0,6]]},"1146":{"position":[[10,6]]},"1172":{"position":[[0,6]]},"1193":{"position":[[0,6]]},"1197":{"position":[[0,6]]}}}],["tab",{"_index":427,"t":{"859":{"position":[[31,3]]}}}],["tactic",{"_index":610,"t":{"1170":{"position":[[0,8]]}}}],["tag",{"_index":310,"t":{"483":{"position":[[12,3]]},"676":{"position":[[0,4]]},"978":{"position":[[10,4]]}}}],["take",{"_index":123,"t":{"169":{"position":[[0,4]]}}}],["taller",{"_index":332,"t":{"550":{"position":[[26,6]]}}}],["target",{"_index":428,"t":{"865":{"position":[[0,8]]},"1001":{"position":[[11,6]]}}}],["tcp",{"_index":284,"t":{"451":{"position":[[0,3]]}}}],["temp",{"_index":620,"t":{"1207":{"position":[[4,4]]}}}],["templat",{"_index":408,"t":{"755":{"position":[[0,8]]},"816":{"position":[[0,8]]}}}],["text",{"_index":323,"t":{"527":{"position":[[11,4]]},"529":{"position":[[18,4]]},"531":{"position":[[7,4]]}}}],["tfl",{"_index":559,"t":{"1069":{"position":[[0,3]]}}}],["theme",{"_index":225,"t":{"303":{"position":[[15,5]]},"305":{"position":[[7,5]]},"574":{"position":[[0,6]]},"840":{"position":[[4,5]]},"842":{"position":[[16,5]]},"847":{"position":[[10,5]]},"925":{"position":[[21,6]]}}}],["threat",{"_index":349,"t":{"605":{"position":[[0,6]]}}}],["three",{"_index":86,"t":{"112":{"position":[[10,5]]}}}],["through",{"_index":116,"t":{"157":{"position":[[15,7]]},"988":{"position":[[34,7]]}}}],["time",{"_index":161,"t":{"199":{"position":[[17,4]]}}}],["timeout",{"_index":629,"t":{"1231":{"position":[[8,7]]}}}],["tip",{"_index":89,"t":{"114":{"position":[[0,4]]}}}],["token",{"_index":542,"t":{"1047":{"position":[[7,5]]}}}],["tool",{"_index":218,"t":{"293":{"position":[[12,5]]}}}],["toolbox",{"_index":398,"t":{"730":{"position":[[4,7]]},"791":{"position":[[4,7]]}}}],["toomanyrequest",{"_index":515,"t":{"976":{"position":[[10,15]]}}}],["top",{"_index":91,"t":{"118":{"position":[[0,3]]},"863":{"position":[[0,3]]},"1138":{"position":[[13,3]]}}}],["traffic",{"_index":589,"t":{"1124":{"position":[[15,7]]},"1126":{"position":[[15,7]]},"1201":{"position":[[8,7]]}}}],["translat",{"_index":138,"t":{"179":{"position":[[4,12]]},"309":{"position":[[8,12]]},"313":{"position":[[3,9]]},"521":{"position":[[3,9]]},"525":{"position":[[9,11]]},"529":{"position":[[7,10]]},"1241":{"position":[[9,12]]}}}],["trend",{"_index":574,"t":{"1085":{"position":[[7,8]]},"1091":{"position":[[11,8]]}}}],["troubleshoot",{"_index":51,"t":{"55":{"position":[[0,15]]},"832":{"position":[[0,15]]},"1249":{"position":[[0,15]]}}}],["truncat",{"_index":510,"t":{"969":{"position":[[26,9]]}}}],["trust",{"_index":59,"t":{"75":{"position":[[5,5]]},"609":{"position":[[0,5]]}}}],["tunnel",{"_index":60,"t":{"75":{"position":[[11,7]]},"449":{"position":[[12,6]]},"451":{"position":[[4,6]]}}}],["type",{"_index":512,"t":{"974":{"position":[[6,4]]}}}],["typographi",{"_index":424,"t":{"857":{"position":[[0,10]]}}}],["ubuntu",{"_index":516,"t":{"982":{"position":[[22,6]]}}}],["ui",{"_index":117,"t":{"157":{"position":[[27,2]]},"511":{"position":[[7,2]]},"849":{"position":[[26,2]]},"895":{"position":[[27,2]]},"1243":{"position":[[7,2]]}}}],["unabl",{"_index":470,"t":{"921":{"position":[[15,7]]}}}],["unavail",{"_index":446,"t":{"887":{"position":[[5,11]]}}}],["unraid",{"_index":175,"t":{"231":{"position":[[0,6]]}}}],["unreach",{"_index":493,"t":{"944":{"position":[[0,11]]}}}],["untrust",{"_index":508,"t":{"967":{"position":[[0,9]]}}}],["up",{"_index":9,"t":{"17":{"position":[[8,2]]},"266":{"position":[[8,2]]},"416":{"position":[[8,2]]},"417":{"position":[[8,2]]},"419":{"position":[[8,2]]},"461":{"position":[[12,2]]}}}],["updat",{"_index":228,"t":{"319":{"position":[[0,8]]},"408":{"position":[[0,8]]},"410":{"position":[[0,8]]},"412":{"position":[[17,7]]},"414":{"position":[[0,8]]},"578":{"position":[[0,6]]},"617":{"position":[[0,6]]},"972":{"position":[[23,6]]},"984":{"position":[[22,8]]},"1225":{"position":[[11,7]]}}}],["uptim",{"_index":608,"t":{"1166":{"position":[[0,6]]},"1168":{"position":[[0,6]]},"1195":{"position":[[0,6]]}}}],["upvot",{"_index":153,"t":{"193":{"position":[[6,6]]}}}],["url",{"_index":255,"t":{"368":{"position":[[9,3]]},"546":{"position":[[0,3]]},"692":{"position":[[8,4]]},"893":{"position":[[28,4]]},"1001":{"position":[[18,3]]}}}],["us",{"_index":20,"t":{"29":{"position":[[0,5]]},"37":{"position":[[0,5]]},"39":{"position":[[0,5]]},"97":{"position":[[5,3]]},"175":{"position":[[47,5]]},"223":{"position":[[0,5]]},"335":{"position":[[9,5]]},"455":{"position":[[0,5]]},"457":{"position":[[0,5]]},"477":{"position":[[0,3]]},"531":{"position":[[3,3]]},"540":{"position":[[0,5]]},"542":{"position":[[0,5]]},"684":{"position":[[0,5]]},"688":{"position":[[0,5]]},"824":{"position":[[0,5]]},"1001":{"position":[[51,3]]}}}],["usag",{"_index":481,"t":{"936":{"position":[[16,5]]},"1175":{"position":[[12,5]]},"1177":{"position":[[12,5]]},"1179":{"position":[[4,5]]},"1181":{"position":[[4,5]]},"1183":{"position":[[15,5]]},"1185":{"position":[[15,5]]},"1187":{"position":[[7,5]]},"1203":{"position":[[9,5]]},"1222":{"position":[[0,5]]},"1223":{"position":[[7,5]]}}}],["user",{"_index":29,"t":{"37":{"position":[[18,5]]},"45":{"position":[[18,5]]},"467":{"position":[[10,4]]},"627":{"position":[[3,4]]},"963":{"position":[[83,5]]},"1140":{"position":[[10,4]]},"1142":{"position":[[10,4]]}}}],["util",{"_index":167,"t":{"209":{"position":[[0,9]]}}}],["v6",{"_index":586,"t":{"1118":{"position":[[14,2]]},"1122":{"position":[[16,2]]},"1126":{"position":[[23,2]]}}}],["valid",{"_index":495,"t":{"949":{"position":[[5,10]]},"986":{"position":[[7,10]]}}}],["variabl",{"_index":22,"t":{"29":{"position":[[18,9]]},"273":{"position":[[14,9]]},"335":{"position":[[29,9]]},"439":{"position":[[25,9]]},"861":{"position":[[4,9]]},"863":{"position":[[10,9]]},"865":{"position":[[15,9]]},"867":{"position":[[10,9]]}}}],["vercel",{"_index":184,"t":{"243":{"position":[[0,6]]},"887":{"position":[[20,7]]}}}],["verifi",{"_index":306,"t":{"481":{"position":[[0,6]]}}}],["version",{"_index":319,"t":{"503":{"position":[[19,7]]}}}],["via",{"_index":148,"t":{"189":{"position":[[18,3]]}}}],["view",{"_index":0,"t":{"3":{"position":[[0,5]]},"9":{"position":[[8,4]]},"901":{"position":[[41,4]]}}}],["volum",{"_index":271,"t":{"419":{"position":[[11,7]]},"477":{"position":[[14,7]]}}}],["vp",{"_index":388,"t":{"713":{"position":[[10,3]]},"774":{"position":[[10,3]]}}}],["vpn",{"_index":61,"t":{"77":{"position":[[0,3]]},"1158":{"position":[[8,3]]}}}],["vulner",{"_index":550,"t":{"1057":{"position":[[0,13]]}}}],["wallet",{"_index":545,"t":{"1049":{"position":[[0,6]]}}}],["warn",{"_index":223,"t":{"299":{"position":[[6,8]]},"990":{"position":[[0,8]]}}}],["watch",{"_index":540,"t":{"1045":{"position":[[7,5]]}}}],["way",{"_index":87,"t":{"112":{"position":[[16,4]]}}}],["weather",{"_index":524,"t":{"1005":{"position":[[0,7]]},"1031":{"position":[[0,7]]},"1033":{"position":[[0,7]]}}}],["web",{"_index":64,"t":{"81":{"position":[[0,3]]},"491":{"position":[[0,3]]},"582":{"position":[[0,3]]},"680":{"position":[[0,3]]},"690":{"position":[[10,3]]}}}],["week",{"_index":392,"t":{"717":{"position":[[6,4]]},"778":{"position":[[6,4]]}}}],["wider",{"_index":331,"t":{"550":{"position":[[17,5]]}}}],["widget",{"_index":240,"t":{"337":{"position":[[11,6]]},"340":{"position":[[16,6]]},"588":{"position":[[0,7]]},"994":{"position":[[0,7]]},"995":{"position":[[0,6]]},"997":{"position":[[0,6]]},"1003":{"position":[[0,6]]},"1005":{"position":[[17,6]]},"1007":{"position":[[0,6]]},"1009":{"position":[[10,6]]},"1028":{"position":[[8,7]]},"1105":{"position":[[21,7]]},"1211":{"position":[[8,7]]},"1212":{"position":[[7,6]]},"1214":{"position":[[14,6]]},"1223":{"position":[[0,6]]},"1237":{"position":[[0,6]]},"1243":{"position":[[0,6]]},"1245":{"position":[[15,6]]},"1247":{"position":[[13,6]]},"1249":{"position":[[16,6]]}}}],["wireguard",{"_index":282,"t":{"447":{"position":[[0,9]]}}}],["within",{"_index":324,"t":{"531":{"position":[[12,6]]}}}],["won't",{"_index":473,"t":{"923":{"position":[[30,5]]}}}],["word",{"_index":151,"t":{"191":{"position":[[11,4]]}}}],["work",{"_index":50,"t":{"53":{"position":[[28,5]]},"61":{"position":[[7,5]]},"67":{"position":[[28,5]]},"89":{"position":[[7,5]]},"836":{"position":[[7,5]]},"840":{"position":[[20,5]]},"959":{"position":[[6,5]]},"963":{"position":[[6,5]]},"1009":{"position":[[21,7]]},"1013":{"position":[[22,7]]}}}],["workflow",{"_index":367,"t":{"656":{"position":[[10,9]]}}}],["workspac",{"_index":2,"t":{"7":{"position":[[0,9]]},"901":{"position":[[31,9]]}}}],["wrangler.toml",{"_index":81,"t":{"102":{"position":[[9,13]]}}}],["write",{"_index":226,"t":{"307":{"position":[[3,5]]},"309":{"position":[[0,7]]},"881":{"position":[[31,5]]}}}],["xkcd",{"_index":567,"t":{"1079":{"position":[[0,4]]}}}],["yaml",{"_index":118,"t":{"159":{"position":[[6,4]]}}}],["yarn",{"_index":477,"t":{"932":{"position":[[0,4]]},"934":{"position":[[0,4]]}}}],["zero",{"_index":58,"t":{"75":{"position":[[0,4]]}}}]],"pipeline":["stemmer"]}},{"documents":[{"i":1,"t":"Documentation | Dashy, a self-hosted dashboard for your homelab","s":"","u":"/docs/","p":1},{"i":2,"t":"Views","s":"Alternate Views & Opening Methods","u":"/docs/alternate-views","p":2},{"i":13,"t":"- Basic Auth","s":"Authentication","u":"/docs/authentication","p":13},{"i":87,"t":"Beyond the cloud backup/restore service, there are several other self-hosted options you can use to backup Dashy, and any other Docker container data. These are outlined in the Management docs, at: Docker Backup Options.","s":"Cloud Backup and Restore","u":"/docs/backup-restore","p":87},{"i":110,"t":"All app configuration is specified in /user-data/conf.yml which is in YAML Format format. If you're using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done through the UI. From the UI you can also export, backup, reset, validate and download your configuration file.","s":"Configuring","u":"/docs/configuring","p":110},{"i":167,"t":"First off, thank you for considering contributing towards Dashy! 🙌","s":"Contributing","u":"/docs/contributing","p":167},{"i":201,"t":"Sponsors","s":"Credits","u":"/docs/credits","p":201},{"i":215,"t":"Welcome to Dashy, so glad you're here :) Deployment is super easy, and there are several methods available depending on what type of system you're using. If you're self-hosting, then deploying with Docker (or similar container engine) is the recommended approach.","s":"Deployment","u":"/docs/deployment","p":215},{"i":264,"t":"This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture.","s":"Developing","u":"/docs/developing","p":264},{"i":301,"t":"A series of short tutorials, to guide you through the most common development tasks.","s":"Development Guides","u":"/docs/development-guides","p":301},{"i":350,"t":"Both sections and items can have an icon, which is specified using the icon attribute. Using icons improves the aesthetics of your UI and makes the app more intuitive to use. Dashy supports multiple different icon providers, usage instructions for which are explained here.","s":"Icons","u":"/docs/icons","p":350},{"i":380,"t":"MIT License","s":"License","u":"/docs/license","p":380},{"i":383,"t":"The following article is a primer on managing self-hosted apps. It covers everything from keeping the Dashy (or any other app) up-to-date, secure, backed up, to other topics like auto-starting, monitoring, log management, web server configuration and using custom domains.","s":"App Management","u":"/docs/management","p":383},{"i":507,"t":"Internationalization is the process of making an application available in other languages. This is important, as not everyone is a native English speaker. This page explains how you can switch languages, how to add a new language, and how to make text translatable when writing a new component.","s":"Internationalization","u":"/docs/multi-language-support","p":507},{"i":535,"t":"Page Metadata","s":"Pages and Sections","u":"/docs/pages-and-sections","p":535},{"i":556,"t":"Dashy was built with privacy in mind.","s":"Privacy & Security","u":"/docs/privacy","p":556},{"i":621,"t":"Welcome to Dashy! So glad you're here 😊 In a couple of minutes, you'll have your new dashboard up and running 🚀","s":"Quick Start","u":"/docs/quick-start","p":621},{"i":641,"t":"- Release Schedule","s":"Releases and Workflows","u":"/docs/release-workflow","p":641},{"i":669,"t":"Searching","s":"Keyboard Shortcuts","u":"/docs/searching","p":669},{"i":696,"t":"| 💗 Got a sweet dashboard? Submit it to the showcase! 👉 See How |","s":"Dashy Showcase 🌟","u":"/docs/showcase","p":696},{"i":757,"t":"| 💗 Got a sweet dashboard? Submit it to the showcase! 👉 See How |","s":"Dashy Showcase 🌟","u":"/docs/showcase/","p":757},{"i":818,"t":"Dashy has an optional feature that can display a small icon next to each of your running services, indicating it's current status. This can be useful if you are using Dashy as your homelab's start page, as it gives you an overview of the health of each of your running services. The status feature will show response time, response code, online/ offline check and if applicable, a relevant error message.","s":"Status Indicators","u":"/docs/status-indicators","p":818},{"i":838,"t":"By default Dashy comes with 40+ built-in themes, which can be applied from the dropdown menu in the UI.","s":"Theming","u":"/docs/theming","p":838},{"i":871,"t":"**This document contains common problems and their solutions.**","s":"Troubleshooting","u":"/docs/troubleshooting","p":871},{"i":1024,"t":"Dashy has support for displaying dynamic content in the form of widgets. There are several built-in widgets available out-of-the-box as well as support for custom widgets to display stats from almost any service with an API.","s":"Widgets","u":"/docs/widgets","p":1024}],"index":{"version":"2.3.9","fields":["t"],"fieldVectors":[["t/1",[0,2.852,1,1.688,2,0.892,3,2.136,4,2.136,5,2.136,6,3.474]],["t/2",[7,4.573]],["t/13",[8,3.566,9,4.344]],["t/87",[2,0.587,3,1.405,4,1.405,10,2.285,11,2.285,12,2.285,13,1.606,14,1.606,15,2.75,16,1.11,17,2.75,18,2.355,19,1.606,20,2.285,21,1.876,22,1.876,23,2.285]],["t/110",[16,0.898,17,1.516,18,1.298,24,1.298,25,2.343,26,1.516,27,1.847,28,1.847,29,1.847,30,2.854,31,1.298,32,3.488,33,1.847,34,1.847,35,1.847,36,1.847,37,1.847,38,1.847,39,1.516,40,2.006,41,1.847,42,1.847,43,1.847,44,1.847]],["t/167",[1,1.688,2,0.892,45,3.474,46,3.474,47,3.474,48,3.474,49,3.474]],["t/201",[50,4.573]],["t/215",[1,0.937,2,0.495,3,1.186,4,1.186,14,1.356,16,0.937,18,1.356,19,1.356,31,2.519,51,1.584,52,1.584,53,1.356,54,2.951,55,1.929,56,1.929,57,1.929,58,1.356,59,1.929,60,1.929,61,1.929,62,1.929,63,1.929,64,1.929,65,1.929]],["t/264",[2,0.826,8,2.641,21,3.492,66,2.641,67,2.261,68,2.641,69,3.216,70,3.216]],["t/301",[39,2.742,68,2.742,71,3.34,72,3.34,73,3.34,74,3.34,75,2.742,76,3.34]],["t/350",[2,0.484,16,1.718,24,1.327,26,1.549,40,1.327,53,1.327,77,1.887,78,1.887,79,1.887,80,3.258,81,1.887,82,1.887,83,1.887,84,1.549,85,1.887,86,1.887,87,1.549,88,1.887,89,1.887,90,1.887,91,1.887,92,1.887,93,1.549]],["t/380",[94,4.344,95,4.344]],["t/383",[2,0.474,3,1.136,4,1.136,16,0.898,22,2.343,24,2.006,25,1.516,66,1.516,96,1.847,97,1.847,98,1.847,99,1.847,100,1.847,101,2.343,102,1.847,103,1.847,104,1.847,105,1.847,106,1.847,107,1.516,108,1.847,109,1.847,110,1.847,111,1.847,112,1.516,113,1.847]],["t/507",[58,1.453,84,2.553,93,1.697,114,2.067,115,2.067,116,1.697,117,3.738,118,2.067,119,2.067,120,2.067,121,2.067,122,2.067,123,1.453,124,2.067,125,2.067,126,2.553,127,2.067,128,2.067,129,2.067,130,2.067]],["t/535",[123,3.054,131,4.344]],["t/556",[2,1.014,132,2.776,133,3.949,134,3.949]],["t/621",[1,1.841,2,0.697,5,1.668,31,1.907,51,2.228,52,2.228,53,1.907,67,1.907,101,2.228,126,2.228,135,2.713,136,2.713,137,2.713]],["t/641",[138,4.344,139,4.344]],["t/669",[140,4.573]],["t/696",[1,2.464,5,1.978,141,2.641,142,2.641,143,2.641,144,2.641]],["t/757",[1,2.464,5,1.978,141,2.641,142,2.641,143,2.641,144,2.641]],["t/818",[2,0.62,13,1.699,15,1.228,16,1.174,67,1.699,80,1.228,107,1.228,116,1.228,123,1.052,145,2.417,146,1.228,147,1.496,148,1.496,149,2.417,150,1.496,151,1.496,152,1.496,153,2.417,154,1.496,155,1.496,156,1.496,157,1.496,158,1.496,159,2.417,160,1.496,161,1.496,162,1.496,163,1.496,164,1.496,165,1.496,166,1.496,167,1.496]],["t/838",[2,0.796,40,2.18,132,2.18,168,3.101,169,3.101,170,3.101,171,3.101,172,3.101,173,3.101,174,3.101]],["t/871",[0,3.1,19,2.655,75,3.1,175,3.777,176,3.777]],["t/1024",[2,0.571,13,1.565,14,1.565,58,1.565,87,2.698,112,1.827,132,1.565,146,2.698,177,2.226,178,2.226,179,2.226,180,3.907,181,2.226,182,2.226,183,2.226,184,2.226,185,2.226]]],"invertedIndex":[["",{"_index":1,"t":{"1":{"position":[[14,1]]},"167":{"position":[[65,2]]},"215":{"position":[[38,2]]},"621":{"position":[[38,2],[111,2]]},"696":{"position":[[0,1],[2,2],[55,2],[66,1]]},"757":{"position":[[0,1],[2,2],[55,2],[66,1]]}}}],["40",{"_index":170,"t":{"838":{"position":[[28,3]]}}}],["add",{"_index":125,"t":{"507":{"position":[[211,3]]}}}],["aesthet",{"_index":83,"t":{"350":{"position":[[112,10]]}}}],["api",{"_index":185,"t":{"1024":{"position":[[220,4]]}}}],["app",{"_index":24,"t":{"110":{"position":[[4,3]]},"350":{"position":[[148,3]]},"383":{"position":[[58,5],[122,4]]}}}],["appli",{"_index":172,"t":{"838":{"position":[[62,7]]}}}],["applic",{"_index":116,"t":{"507":{"position":[[49,11]]},"818":{"position":[[367,11]]}}}],["approach",{"_index":65,"t":{"215":{"position":[[254,9]]}}}],["architectur",{"_index":70,"t":{"264":{"position":[[108,13]]}}}],["articl",{"_index":66,"t":{"264":{"position":[[5,7]]},"383":{"position":[[14,7]]}}}],["attribut",{"_index":81,"t":{"350":{"position":[[76,10]]}}}],["auth",{"_index":9,"t":{"13":{"position":[[8,4]]}}}],["auto",{"_index":106,"t":{"383":{"position":[[179,4]]}}}],["avail",{"_index":58,"t":{"215":{"position":[[97,9]]},"507":{"position":[[61,9]]},"1024":{"position":[[108,9]]}}}],["back",{"_index":104,"t":{"383":{"position":[[147,6]]}}}],["backup",{"_index":17,"t":{"87":{"position":[[100,6],[205,6]]},"110":{"position":[[261,7]]}}}],["backup/restor",{"_index":12,"t":{"87":{"position":[[17,14]]}}}],["basic",{"_index":8,"t":{"13":{"position":[[2,5]]},"264":{"position":[[94,6]]}}}],["beyond",{"_index":10,"t":{"87":{"position":[[0,6]]}}}],["both",{"_index":77,"t":{"350":{"position":[[0,4]]}}}],["box",{"_index":182,"t":{"1024":{"position":[[129,3]]}}}],["built",{"_index":132,"t":{"556":{"position":[[10,5]]},"838":{"position":[[32,5]]},"1024":{"position":[[91,5]]}}}],["chang",{"_index":35,"t":{"110":{"position":[[154,7]]}}}],["check",{"_index":164,"t":{"818":{"position":[[354,5]]}}}],["cloud",{"_index":11,"t":{"87":{"position":[[11,5]]}}}],["code",{"_index":161,"t":{"818":{"position":[[332,5]]}}}],["come",{"_index":169,"t":{"838":{"position":[[17,5]]}}}],["common",{"_index":75,"t":{"301":{"position":[[59,6]]},"871":{"position":[[25,6]]}}}],["compon",{"_index":130,"t":{"507":{"position":[[284,10]]}}}],["configur",{"_index":25,"t":{"110":{"position":[[8,13],[303,13]]},"383":{"position":[[233,13]]}}}],["consid",{"_index":47,"t":{"167":{"position":[[25,11]]}}}],["contain",{"_index":19,"t":{"87":{"position":[[135,9]]},"215":{"position":[[217,9]]},"871":{"position":[[16,8]]}}}],["content",{"_index":178,"t":{"1024":{"position":[[41,7]]}}}],["contribut",{"_index":48,"t":{"167":{"position":[[37,12]]}}}],["coupl",{"_index":135,"t":{"621":{"position":[[46,6]]}}}],["cover",{"_index":98,"t":{"383":{"position":[[67,6]]}}}],["current",{"_index":152,"t":{"818":{"position":[[115,7]]}}}],["custom",{"_index":112,"t":{"383":{"position":[[257,6]]},"1024":{"position":[[156,6]]}}}],["dashboard",{"_index":5,"t":{"1":{"position":[[37,9]]},"621":{"position":[[86,9]]},"696":{"position":[[17,10]]},"757":{"position":[[17,10]]}}}],["dashi",{"_index":2,"t":{"1":{"position":[[16,6]]},"87":{"position":[[107,6]]},"167":{"position":[[58,6]]},"215":{"position":[[11,6]]},"264":{"position":[[33,5]]},"350":{"position":[[175,5]]},"383":{"position":[[102,5]]},"556":{"position":[[0,5]]},"621":{"position":[[11,6]]},"818":{"position":[[0,5],[167,5]]},"838":{"position":[[11,5]]},"1024":{"position":[[0,5]]}}}],["data",{"_index":20,"t":{"87":{"position":[[145,5]]}}}],["data/conf.yml",{"_index":28,"t":{"110":{"position":[[44,13]]}}}],["date",{"_index":102,"t":{"383":{"position":[[133,5]]}}}],["default",{"_index":168,"t":{"838":{"position":[[3,7]]}}}],["depend",{"_index":59,"t":{"215":{"position":[[107,9]]}}}],["deploy",{"_index":54,"t":{"215":{"position":[[41,10],[183,9]]}}}],["develop",{"_index":68,"t":{"264":{"position":[[52,11]]},"301":{"position":[[66,11]]}}}],["differ",{"_index":89,"t":{"350":{"position":[[199,9]]}}}],["directli",{"_index":37,"t":{"110":{"position":[[181,8]]}}}],["display",{"_index":146,"t":{"818":{"position":[[39,7]]},"1024":{"position":[[22,10],[174,7]]}}}],["doc",{"_index":23,"t":{"87":{"position":[[188,5]]}}}],["docker",{"_index":18,"t":{"87":{"position":[[128,6],[198,6]]},"110":{"position":[[106,7]]},"215":{"position":[[198,6]]}}}],["document",{"_index":0,"t":{"1":{"position":[[0,13]]},"871":{"position":[[7,8]]}}}],["domain",{"_index":113,"t":{"383":{"position":[[264,8]]}}}],["done",{"_index":38,"t":{"110":{"position":[[207,4]]}}}],["download",{"_index":44,"t":{"110":{"position":[[289,8]]}}}],["dropdown",{"_index":173,"t":{"838":{"position":[[79,8]]}}}],["dynam",{"_index":177,"t":{"1024":{"position":[[33,7]]}}}],["each",{"_index":149,"t":{"818":{"position":[[68,4],[248,4]]}}}],["easi",{"_index":56,"t":{"215":{"position":[[61,5]]}}}],["engin",{"_index":63,"t":{"215":{"position":[[227,7]]}}}],["english",{"_index":121,"t":{"507":{"position":[[138,7]]}}}],["environ",{"_index":69,"t":{"264":{"position":[[64,12]]}}}],["error",{"_index":166,"t":{"818":{"position":[[390,5]]}}}],["everyon",{"_index":119,"t":{"507":{"position":[[117,8]]}}}],["everyth",{"_index":99,"t":{"383":{"position":[[74,10]]}}}],["explain",{"_index":93,"t":{"350":{"position":[[258,9]]},"507":{"position":[[165,8]]}}}],["export",{"_index":41,"t":{"110":{"position":[[253,7]]}}}],["featur",{"_index":145,"t":{"818":{"position":[[22,7],[290,7]]}}}],["file",{"_index":32,"t":{"110":{"position":[[119,4],[198,5],[317,5]]}}}],["first",{"_index":45,"t":{"167":{"position":[[0,5]]}}}],["follow",{"_index":96,"t":{"383":{"position":[[4,9]]}}}],["form",{"_index":179,"t":{"1024":{"position":[[56,4]]}}}],["format",{"_index":30,"t":{"110":{"position":[[75,6],[82,7]]}}}],["give",{"_index":155,"t":{"818":{"position":[[209,5]]}}}],["glad",{"_index":52,"t":{"215":{"position":[[21,4]]},"621":{"position":[[21,4]]}}}],["guid",{"_index":74,"t":{"301":{"position":[[32,5]]}}}],["health",{"_index":157,"t":{"818":{"position":[[238,6]]}}}],["here",{"_index":53,"t":{"215":{"position":[[33,4]]},"350":{"position":[[268,5]]},"621":{"position":[[33,4]]}}}],["homelab",{"_index":6,"t":{"1":{"position":[[56,7]]}}}],["homelab'",{"_index":154,"t":{"818":{"position":[[181,9]]}}}],["host",{"_index":4,"t":{"1":{"position":[[30,6]]},"87":{"position":[[70,6]]},"215":{"position":[[169,8]]},"383":{"position":[[51,6]]}}}],["icon",{"_index":80,"t":{"350":{"position":[[36,5],[71,4],[93,5],[209,4]]},"818":{"position":[[55,4]]}}}],["import",{"_index":118,"t":{"507":{"position":[[99,10]]}}}],["improv",{"_index":82,"t":{"350":{"position":[[99,8]]}}}],["indic",{"_index":150,"t":{"818":{"position":[[99,10]]}}}],["instruct",{"_index":92,"t":{"350":{"position":[[231,12]]}}}],["internation",{"_index":114,"t":{"507":{"position":[[0,20]]}}}],["intuit",{"_index":86,"t":{"350":{"position":[[157,9]]}}}],["it'",{"_index":151,"t":{"818":{"position":[[110,4]]}}}],["item",{"_index":79,"t":{"350":{"position":[[18,5]]}}}],["keep",{"_index":100,"t":{"383":{"position":[[90,7]]}}}],["languag",{"_index":117,"t":{"507":{"position":[[80,10],[193,10],[221,9]]}}}],["licens",{"_index":95,"t":{"380":{"position":[[4,7]]}}}],["log",{"_index":109,"t":{"383":{"position":[[206,3]]}}}],["made",{"_index":36,"t":{"110":{"position":[[176,4]]}}}],["make",{"_index":84,"t":{"350":{"position":[[138,5]]},"507":{"position":[[39,6],[242,4]]}}}],["manag",{"_index":22,"t":{"87":{"position":[[177,10]]},"383":{"position":[[37,8],[210,11]]}}}],["menu",{"_index":174,"t":{"838":{"position":[[88,4]]}}}],["messag",{"_index":167,"t":{"818":{"position":[[396,8]]}}}],["metadata",{"_index":131,"t":{"535":{"position":[[5,8]]}}}],["method",{"_index":57,"t":{"215":{"position":[[89,7]]}}}],["mind",{"_index":134,"t":{"556":{"position":[[32,5]]}}}],["minut",{"_index":136,"t":{"621":{"position":[[56,8]]}}}],["mit",{"_index":94,"t":{"380":{"position":[[0,3]]}}}],["monitor",{"_index":108,"t":{"383":{"position":[[194,11]]}}}],["more",{"_index":85,"t":{"350":{"position":[[152,4]]}}}],["multipl",{"_index":88,"t":{"350":{"position":[[190,8]]}}}],["nativ",{"_index":120,"t":{"507":{"position":[[131,6]]}}}],["new",{"_index":126,"t":{"507":{"position":[[217,3],[280,3]]},"621":{"position":[[82,3]]}}}],["next",{"_index":148,"t":{"818":{"position":[[60,4]]}}}],["offlin",{"_index":163,"t":{"818":{"position":[[346,7]]}}}],["onlin",{"_index":162,"t":{"818":{"position":[[338,7]]}}}],["option",{"_index":15,"t":{"87":{"position":[[77,7],[212,8]]},"818":{"position":[[13,8]]}}}],["out",{"_index":181,"t":{"1024":{"position":[[118,3]]}}}],["outlin",{"_index":21,"t":{"87":{"position":[[161,8]]},"264":{"position":[[13,8],[81,8]]}}}],["overview",{"_index":156,"t":{"818":{"position":[[222,8]]}}}],["page",{"_index":123,"t":{"507":{"position":[[160,4]]},"535":{"position":[[0,4]]},"818":{"position":[[197,5]]}}}],["pass",{"_index":33,"t":{"110":{"position":[[131,6]]}}}],["primer",{"_index":97,"t":{"383":{"position":[[27,6]]}}}],["privaci",{"_index":133,"t":{"556":{"position":[[21,7]]}}}],["problem",{"_index":175,"t":{"871":{"position":[[32,8]]}}}],["process",{"_index":115,"t":{"507":{"position":[[28,7]]}}}],["provid",{"_index":90,"t":{"350":{"position":[[214,10]]}}}],["recommend",{"_index":64,"t":{"215":{"position":[[242,11]]}}}],["releas",{"_index":138,"t":{"641":{"position":[[2,7]]}}}],["relev",{"_index":165,"t":{"818":{"position":[[381,8]]}}}],["reset",{"_index":42,"t":{"110":{"position":[[269,6]]}}}],["respons",{"_index":159,"t":{"818":{"position":[[308,8],[323,8]]}}}],["run",{"_index":67,"t":{"264":{"position":[[39,7]]},"621":{"position":[[103,7]]},"818":{"position":[[81,7],[261,7]]}}}],["schedul",{"_index":139,"t":{"641":{"position":[[10,8]]}}}],["search",{"_index":140,"t":{"669":{"position":[[0,9]]}}}],["section",{"_index":78,"t":{"350":{"position":[[5,8]]}}}],["secur",{"_index":103,"t":{"383":{"position":[[139,7]]}}}],["see",{"_index":144,"t":{"696":{"position":[[58,3]]},"757":{"position":[[58,3]]}}}],["self",{"_index":3,"t":{"1":{"position":[[25,4]]},"87":{"position":[[65,4]]},"215":{"position":[[164,4]]},"383":{"position":[[46,4]]}}}],["seri",{"_index":71,"t":{"301":{"position":[[2,6]]}}}],["server",{"_index":111,"t":{"383":{"position":[[226,6]]}}}],["servic",{"_index":13,"t":{"87":{"position":[[32,8]]},"818":{"position":[[89,9],[269,9]]},"1024":{"position":[[204,7]]}}}],["sever",{"_index":14,"t":{"87":{"position":[[51,7]]},"215":{"position":[[81,7]]},"1024":{"position":[[83,7]]}}}],["short",{"_index":72,"t":{"301":{"position":[[12,5]]}}}],["show",{"_index":158,"t":{"818":{"position":[[303,4]]}}}],["showcas",{"_index":143,"t":{"696":{"position":[[45,9]]},"757":{"position":[[45,9]]}}}],["similar",{"_index":62,"t":{"215":{"position":[[209,7]]}}}],["small",{"_index":147,"t":{"818":{"position":[[49,5]]}}}],["solut",{"_index":176,"t":{"871":{"position":[[51,12]]}}}],["speaker",{"_index":122,"t":{"507":{"position":[[146,8]]}}}],["specifi",{"_index":26,"t":{"110":{"position":[[25,9]]},"350":{"position":[[51,9]]}}}],["sponsor",{"_index":50,"t":{"201":{"position":[[0,8]]}}}],["start",{"_index":107,"t":{"383":{"position":[[184,9]]},"818":{"position":[[191,5]]}}}],["stat",{"_index":184,"t":{"1024":{"position":[[182,5]]}}}],["statu",{"_index":153,"t":{"818":{"position":[[123,7],[283,6]]}}}],["submit",{"_index":142,"t":{"696":{"position":[[28,6]]},"757":{"position":[[28,6]]}}}],["super",{"_index":55,"t":{"215":{"position":[[55,5]]}}}],["support",{"_index":87,"t":{"350":{"position":[[181,8]]},"1024":{"position":[[10,7],[144,7]]}}}],["sweet",{"_index":141,"t":{"696":{"position":[[11,5]]},"757":{"position":[[11,5]]}}}],["switch",{"_index":124,"t":{"507":{"position":[[186,6]]}}}],["system",{"_index":61,"t":{"215":{"position":[[133,6]]}}}],["task",{"_index":76,"t":{"301":{"position":[[78,6]]}}}],["text",{"_index":127,"t":{"507":{"position":[[247,4]]}}}],["thank",{"_index":46,"t":{"167":{"position":[[11,5]]}}}],["theme",{"_index":171,"t":{"838":{"position":[[41,7]]}}}],["through",{"_index":39,"t":{"110":{"position":[[212,7]]},"301":{"position":[[42,7]]}}}],["time",{"_index":160,"t":{"818":{"position":[[317,5]]}}}],["topic",{"_index":105,"t":{"383":{"position":[[167,6]]}}}],["toward",{"_index":49,"t":{"167":{"position":[[50,7]]}}}],["translat",{"_index":128,"t":{"507":{"position":[[252,12]]}}}],["tutori",{"_index":73,"t":{"301":{"position":[[18,10]]}}}],["type",{"_index":60,"t":{"215":{"position":[[125,4]]}}}],["ui",{"_index":40,"t":{"110":{"position":[[224,3],[237,2]]},"350":{"position":[[131,2]]},"838":{"position":[[100,3]]}}}],["up",{"_index":101,"t":{"383":{"position":[[127,2],[154,3]]},"621":{"position":[[96,2]]}}}],["us",{"_index":16,"t":{"87":{"position":[[93,3]]},"110":{"position":[[100,5]]},"215":{"position":[[147,6]]},"350":{"position":[[61,5],[87,5],[170,4]]},"383":{"position":[[251,5]]},"818":{"position":[[143,6],[161,5]]}}}],["usag",{"_index":91,"t":{"350":{"position":[[225,5]]}}}],["user",{"_index":27,"t":{"110":{"position":[[38,5]]}}}],["valid",{"_index":43,"t":{"110":{"position":[[276,8]]}}}],["view",{"_index":7,"t":{"2":{"position":[[0,5]]}}}],["volum",{"_index":34,"t":{"110":{"position":[[146,7]]}}}],["web",{"_index":110,"t":{"383":{"position":[[222,3]]}}}],["welcom",{"_index":51,"t":{"215":{"position":[[0,7]]},"621":{"position":[[0,7]]}}}],["well",{"_index":183,"t":{"1024":{"position":[[136,4]]}}}],["widget",{"_index":180,"t":{"1024":{"position":[[64,8],[100,7],[163,7]]}}}],["write",{"_index":129,"t":{"507":{"position":[[270,7]]}}}],["yaml",{"_index":29,"t":{"110":{"position":[[70,4]]}}}],["you'll",{"_index":137,"t":{"621":{"position":[[65,6]]}}}],["you'r",{"_index":31,"t":{"110":{"position":[[93,6]]},"215":{"position":[[26,6],[140,6],[157,6]]},"621":{"position":[[26,6]]}}}]],"pipeline":["stemmer"]}},{"documents":[{"i":1,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"","u":"/docs/","p":1},{"i":2,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Alternate Views & Opening Methods","u":"/docs/alternate-views","p":2},{"i":13,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Authentication","u":"/docs/authentication","p":13},{"i":87,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Cloud Backup and Restore","u":"/docs/backup-restore","p":87},{"i":110,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Configuring","u":"/docs/configuring","p":110},{"i":167,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Contributing","u":"/docs/contributing","p":167},{"i":201,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Credits","u":"/docs/credits","p":201},{"i":215,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Deployment","u":"/docs/deployment","p":215},{"i":264,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Developing","u":"/docs/developing","p":264},{"i":301,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Development Guides","u":"/docs/development-guides","p":301},{"i":350,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Icons","u":"/docs/icons","p":350},{"i":380,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"License","u":"/docs/license","p":380},{"i":383,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"App Management","u":"/docs/management","p":383},{"i":507,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Internationalization","u":"/docs/multi-language-support","p":507},{"i":535,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Pages and Sections","u":"/docs/pages-and-sections","p":535},{"i":556,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Privacy & Security","u":"/docs/privacy","p":556},{"i":621,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Quick Start","u":"/docs/quick-start","p":621},{"i":641,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Releases and Workflows","u":"/docs/release-workflow","p":641},{"i":669,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Keyboard Shortcuts","u":"/docs/searching","p":669},{"i":696,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Dashy Showcase 🌟","u":"/docs/showcase","p":696},{"i":757,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Dashy Showcase 🌟","u":"/docs/showcase/","p":757},{"i":818,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Status Indicators","u":"/docs/status-indicators","p":818},{"i":838,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Theming","u":"/docs/theming","p":838},{"i":871,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Troubleshooting","u":"/docs/troubleshooting","p":871},{"i":1024,"t":"dashy, dashboard, homelab, self-hosted, docker, homepage","s":"Widgets","u":"/docs/widgets","p":1024}],"index":{"version":"2.3.9","fields":["t"],"fieldVectors":[["t/1",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/2",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/13",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/87",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/110",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/167",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/201",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/215",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/264",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/301",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/350",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/380",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/383",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/507",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/535",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/556",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/621",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/641",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/669",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/696",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/757",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/818",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/838",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/871",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]],["t/1024",[0,0.019,1,0.019,2,0.019,3,0.019,4,0.019,5,0.019,6,0.019]]],"invertedIndex":[["dashboard",{"_index":1,"t":{"1":{"position":[[8,10]]},"2":{"position":[[8,10]]},"13":{"position":[[8,10]]},"87":{"position":[[8,10]]},"110":{"position":[[8,10]]},"167":{"position":[[8,10]]},"201":{"position":[[8,10]]},"215":{"position":[[8,10]]},"264":{"position":[[8,10]]},"301":{"position":[[8,10]]},"350":{"position":[[8,10]]},"380":{"position":[[8,10]]},"383":{"position":[[8,10]]},"507":{"position":[[8,10]]},"535":{"position":[[8,10]]},"556":{"position":[[8,10]]},"621":{"position":[[8,10]]},"641":{"position":[[8,10]]},"669":{"position":[[8,10]]},"696":{"position":[[8,10]]},"757":{"position":[[8,10]]},"818":{"position":[[8,10]]},"838":{"position":[[8,10]]},"871":{"position":[[8,10]]},"1024":{"position":[[8,10]]}}}],["dashi",{"_index":0,"t":{"1":{"position":[[0,6]]},"2":{"position":[[0,6]]},"13":{"position":[[0,6]]},"87":{"position":[[0,6]]},"110":{"position":[[0,6]]},"167":{"position":[[0,6]]},"201":{"position":[[0,6]]},"215":{"position":[[0,6]]},"264":{"position":[[0,6]]},"301":{"position":[[0,6]]},"350":{"position":[[0,6]]},"380":{"position":[[0,6]]},"383":{"position":[[0,6]]},"507":{"position":[[0,6]]},"535":{"position":[[0,6]]},"556":{"position":[[0,6]]},"621":{"position":[[0,6]]},"641":{"position":[[0,6]]},"669":{"position":[[0,6]]},"696":{"position":[[0,6]]},"757":{"position":[[0,6]]},"818":{"position":[[0,6]]},"838":{"position":[[0,6]]},"871":{"position":[[0,6]]},"1024":{"position":[[0,6]]}}}],["docker",{"_index":5,"t":{"1":{"position":[[44,7]]},"2":{"position":[[44,7]]},"13":{"position":[[44,7]]},"87":{"position":[[44,7]]},"110":{"position":[[44,7]]},"167":{"position":[[44,7]]},"201":{"position":[[44,7]]},"215":{"position":[[44,7]]},"264":{"position":[[44,7]]},"301":{"position":[[44,7]]},"350":{"position":[[44,7]]},"380":{"position":[[44,7]]},"383":{"position":[[44,7]]},"507":{"position":[[44,7]]},"535":{"position":[[44,7]]},"556":{"position":[[44,7]]},"621":{"position":[[44,7]]},"641":{"position":[[44,7]]},"669":{"position":[[44,7]]},"696":{"position":[[44,7]]},"757":{"position":[[44,7]]},"818":{"position":[[44,7]]},"838":{"position":[[44,7]]},"871":{"position":[[44,7]]},"1024":{"position":[[44,7]]}}}],["homelab",{"_index":2,"t":{"1":{"position":[[20,8]]},"2":{"position":[[20,8]]},"13":{"position":[[20,8]]},"87":{"position":[[20,8]]},"110":{"position":[[20,8]]},"167":{"position":[[20,8]]},"201":{"position":[[20,8]]},"215":{"position":[[20,8]]},"264":{"position":[[20,8]]},"301":{"position":[[20,8]]},"350":{"position":[[20,8]]},"380":{"position":[[20,8]]},"383":{"position":[[20,8]]},"507":{"position":[[20,8]]},"535":{"position":[[20,8]]},"556":{"position":[[20,8]]},"621":{"position":[[20,8]]},"641":{"position":[[20,8]]},"669":{"position":[[20,8]]},"696":{"position":[[20,8]]},"757":{"position":[[20,8]]},"818":{"position":[[20,8]]},"838":{"position":[[20,8]]},"871":{"position":[[20,8]]},"1024":{"position":[[20,8]]}}}],["homepag",{"_index":6,"t":{"1":{"position":[[53,8]]},"2":{"position":[[53,8]]},"13":{"position":[[53,8]]},"87":{"position":[[53,8]]},"110":{"position":[[53,8]]},"167":{"position":[[53,8]]},"201":{"position":[[53,8]]},"215":{"position":[[53,8]]},"264":{"position":[[53,8]]},"301":{"position":[[53,8]]},"350":{"position":[[53,8]]},"380":{"position":[[53,8]]},"383":{"position":[[53,8]]},"507":{"position":[[53,8]]},"535":{"position":[[53,8]]},"556":{"position":[[53,8]]},"621":{"position":[[53,8]]},"641":{"position":[[53,8]]},"669":{"position":[[53,8]]},"696":{"position":[[53,8]]},"757":{"position":[[53,8]]},"818":{"position":[[53,8]]},"838":{"position":[[53,8]]},"871":{"position":[[53,8]]},"1024":{"position":[[53,8]]}}}],["host",{"_index":4,"t":{"1":{"position":[[35,7]]},"2":{"position":[[35,7]]},"13":{"position":[[35,7]]},"87":{"position":[[35,7]]},"110":{"position":[[35,7]]},"167":{"position":[[35,7]]},"201":{"position":[[35,7]]},"215":{"position":[[35,7]]},"264":{"position":[[35,7]]},"301":{"position":[[35,7]]},"350":{"position":[[35,7]]},"380":{"position":[[35,7]]},"383":{"position":[[35,7]]},"507":{"position":[[35,7]]},"535":{"position":[[35,7]]},"556":{"position":[[35,7]]},"621":{"position":[[35,7]]},"641":{"position":[[35,7]]},"669":{"position":[[35,7]]},"696":{"position":[[35,7]]},"757":{"position":[[35,7]]},"818":{"position":[[35,7]]},"838":{"position":[[35,7]]},"871":{"position":[[35,7]]},"1024":{"position":[[35,7]]}}}],["self",{"_index":3,"t":{"1":{"position":[[30,4]]},"2":{"position":[[30,4]]},"13":{"position":[[30,4]]},"87":{"position":[[30,4]]},"110":{"position":[[30,4]]},"167":{"position":[[30,4]]},"201":{"position":[[30,4]]},"215":{"position":[[30,4]]},"264":{"position":[[30,4]]},"301":{"position":[[30,4]]},"350":{"position":[[30,4]]},"380":{"position":[[30,4]]},"383":{"position":[[30,4]]},"507":{"position":[[30,4]]},"535":{"position":[[30,4]]},"556":{"position":[[30,4]]},"621":{"position":[[30,4]]},"641":{"position":[[30,4]]},"669":{"position":[[30,4]]},"696":{"position":[[30,4]]},"757":{"position":[[30,4]]},"818":{"position":[[30,4]]},"838":{"position":[[30,4]]},"871":{"position":[[30,4]]},"1024":{"position":[[30,4]]}}}]],"pipeline":["stemmer"]}},{"documents":[{"i":4,"t":"Dashy has three different views: Default View - This is the main homepage with sections in a grid layout Workspace View - Items displayed on the side, and are launched within Dashy Minimal View - A clean + simple tabbed view You can switch between views using the dropdown in the top-right corner. Set your chosen Starting View with appConfig.startingView. Click the page title at any time to go back to your selected starting view.","s":"Views","u":"/docs/alternate-views","h":"#views","p":2},{"i":6,"t":"This is the main page that you will land on when you first launch the application. Here all of your sections (with items + widgets) are visible in a grid layout. Example of Default View","s":"Default","u":"/docs/alternate-views","h":"#default","p":2},{"i":8,"t":"The workspace view displays your links in a sidebar on the left-hand side, and apps are launched inside an iframe without having to leave Dashy. This enables you to use all of your self-hosted apps from one place, and makes multi-tasking easy. You can specify a default app to be opened when you land on the workspace, by setting appConfig.workspaceLandingUrl: https://app-to-open/. If this app exists within your sections.items, then the corresponding section will also be expanded. You can also opt to keep previously opened websites/ apps open in the background, by setting appConfig.enableMultiTasking: true. This comes at the cost of performance, but does mean that your session with each app is preserved, enabling you to quickly switch between them. Example of Workspace View","s":"Workspace","u":"/docs/alternate-views","h":"#workspace","p":2},{"i":10,"t":"The minimal view aims to be super fast and simple, and can be used as a browser startpage. Items are grouped into a tab view, and the last opened tab will be remembered. Similar to the main view, you can search and launch items just by typing, and right-clicking will show more options (like open in modal, workspace or new tab). Example of Minimal View","s":"Minimal View","u":"/docs/alternate-views","h":"#minimal-view","p":2},{"i":12,"t":"Dashy supports several different ways to launch your apps. The primary opening method for each app can be specified using the target attribute, with a value of one of the following: sametab - The app will be launched in the current tab newtab - The app will be launched in a new tab top - Opens in the top-most browsing context, useful if you're accessing Dashy through an iframe modal - Launch app in a resizable/ movable popup modal on the current page workspace - Changes to Workspace view, and launches app You can also set a default opening method, which will be applied to all items that don't have a specified target, using appConfig.defaultOpeningMethod, to one of the above values. Even if the target is not set (or is set to sametab), you can still launch any given app in an alternative method. Either right-click to see all options, or use one of the keyboard shortcuts: Alt + Click will open the modal, and Ctrl + Click will open in a new tab. If you don't like the custom context menu, it can be disabled by setting appConfig.disableContextMenu: true. If you get a 'Refused to Connect' error in the modal or workspace views, then the target app has it's X-Frame-Options HTTP set to block requests from embedded content. You can easily fix this by setting this header to ALLOW, for instructions on how to do so, see the Troubleshooting Docs.","s":"Opening Methods","u":"/docs/alternate-views","h":"#opening-methods","p":2},{"i":14,"t":"Basic Auth Setting Up Authentication Hash Password Logging In and Out Guest Access Granular Access Permissions Using Environment Variables for Passwords Adding HTTP Auth to Configuration Security HTTP Auth Using Config-File Users Using Static Credentials Keycloak Auth Header Authentication OIDC Auth authentik Alternative Authentication Methods Reverse Proxy Auth Zero-Trust Tunnels VPN IP-Based Access Web Server Authentication SSO / OAuth Providers Cloud Hosting Providers important Dashy's built-in auth is not intended to protect a publicly hosted instance against unauthorized access. Instead you should use an auth provider compatible with your reverse proxy, or access Dashy via your VPN, or implement your own SSO logic. If Dashy is only accessible within your home network and you just want a login page, then the built-in auth may be sufficient. To also protect server-side endpoints and config files: with built-in auth set ENABLE_HTTP_AUTH=true (details). (Or, consider setting upOIDC, Keycloak, or Header Auth, where the server-side enforcement is on automatically).","s":"Authentication","u":"/docs/authentication","h":"","p":13},{"i":16,"t":"Dashy has a basic login page included, and frontend authentication. You can enable this by adding users to the auth section under appConfig in your conf.yml. If this section is not specified, then no authentication will be required to access the app, and the homepage will resolve to your dashboard. To also enable HTTP Authorization, set the ENABLE_HTTP_AUTH env var to true.","s":"Built-In Auth","u":"/docs/authentication","h":"#built-in-auth","p":13},{"i":18,"t":"The auth property takes an array of users. Each user needs to include a username, hash and optional user type (admin or normal). The hash property is a SHA-256 Hash of your desired password. For example: appConfig: auth: users: - user: alicia hash: 4D1E58C90B3B94BCAD9848ECCACD6D2A8C9FBC5CA913304BBA5CDEAB36FEEFA3 type: admin - user: bob hash: 5E884898DA28047151D0E56F8DC6292773603D0D6AABBDD62A11EF721D1542D8","s":"Setting Up Authentication","u":"/docs/authentication","h":"#setting-up-authentication","p":13},{"i":20,"t":"Dashy uses SHA-256 Hash, a 64-character string, which you can generate by running echo -n \"my-super-secure-password\" | sha256sum, or using an online tool, such as this one or CyberChef (which can be self-hosted/ ran locally). A hash is a one-way cryptographic function, meaning that it is easy to generate a hash for a given password, but very hard to determine the original password for a given hash. This means, that so long as your password is long, strong and unique, it is safe to store its hash in the clear. Having said that, you should never reuse passwords, hashes can be cracked by iterating over known password lists, generating a hash of each.","s":"Hash Password","u":"/docs/authentication","h":"#hash-password","p":13},{"i":22,"t":"Once authentication is enabled, so long as there is no valid token in cookie storage, the application will redirect the user to the login page. When the user enters credentials in the login page, they will be checked, and if valid, then a token will be generated, and they can be redirected to the home page. If credentials are invalid, then an error message will be shown, and they will remain on the login page. Once in the application, to log out: the user can click the logout button (in the top-right), which will clear cookie storage, causing them to be redirected back to the login page.","s":"Logging In and Out","u":"/docs/authentication","h":"#logging-in-and-out","p":13},{"i":24,"t":"With authentication set up, by default no access is allowed to your dashboard without first logging in with valid credentials. Guest mode can be enabled to allow for read-only access to a secured dashboard by any user, without the need to log in. A guest user cannot write any changes to the config file, but can apply modifications locally (stored in their browser). You can enable guest access, by setting appConfig.auth.enableGuestAccess: true.","s":"Enabling Guest Access","u":"/docs/authentication","h":"#enabling-guest-access","p":13},{"i":26,"t":"You can use the following properties to make certain pages, sections or items only visible to some users, or hide pages, sections and items from guests. hideForUsers - Page, Section or Item will be visible to all users, except for those specified in this list showForUsers - Page, Section or Item will be hidden from all users, except for those specified in this list hideForGuests - Page, Section or Item will be visible for logged in users, but not for guests For Example: pages: - name: Home Lab path: home-lab.yml displayData: showForUsers: [admin] - name: Intranet path: intranet.yml displayData: hideForGuests: true hideForUsers: [alicia, bob] - name: Code Analysis & Monitoring icon: fas fa-code displayData: cols: 2 hideForUsers: [alicia, bob] items: ... - name: Deployment Pipelines icon: fas fa-rocket displayData: hideForGuests: true items: - title: Hide Me displayData: hideForUsers: [alicia, bob]","s":"Granular Access","u":"/docs/authentication","h":"#granular-access","p":13},{"i":28,"t":"Any user who is not an admin (with type: admin) will not be able to write changes to disk. You can also prevent any user from writing changes to disk, using preventWriteToDisk. Or prevent any changes from being saved locally in browser storage, using preventLocalSave. Both properties can be found under appConfig. To disable all UI config features, including View Config, set disableConfiguration. Alternatively you can disable UI config features for all non admin users by setting disableConfigurationForNonAdmin to true.","s":"Permissions","u":"/docs/authentication","h":"#permissions","p":13},{"i":30,"t":"If you don't want to hash your password, you can instead leave out the hash attribute, and replace it with password which should have the value of an environmental variable name you wish to use. Note that env var must begin with VITE_APP_, and you must set this variable before building the app. For example: auth: users: - user: bob password: VITE_APP_BOB Just be sure to set VITE_APP_BOB='my super secret password' before build-time.","s":"Using Environment Variables for Passwords","u":"/docs/authentication","h":"#using-environment-variables-for-passwords","p":13},{"i":32,"t":"Without this, the built-in auth is just a client-side login page — your config and API endpoints can still be accessed directly. Set ENABLE_HTTP_AUTH=true to protect them. note HTTP Auth and guest access (enableGuestAccess) are incompatible. Guests have no credentials, so they can't fetch the config file when HTTP auth is active. This uses the same users you've already defined in appConfig.auth.users to authenticate all server-side requests (config files, status checks, system info, CORS proxy, etc.) via HTTP Basic Auth. How it works: When a user logs in through the Dashy UI, a session token is stored in a cookie. The frontend automatically includes this token in requests to local API endpoints. On the server side, the token is validated against your configured users. If someone tries to access an endpoint directly (e.g. with curl), the server will respond with a 401 and a Basic Auth challenge — they'll need to provide a valid username and password. Setup: Make sure you have users configured in your conf.yml (see Setting Up Authentication above) Set the ENABLE_HTTP_AUTH=true environment variable (e.g. in your docker-compose.yml or .env file) Restart the container - the auth mode is determined at startup, so env var changes need a restart Adding or removing users in conf.yml takes effect immediately without a restart, since the user list is read from disk on each request. For full protection, you'll want both the client-side login page (via appConfig.auth.users) and server-side auth (via ENABLE_HTTP_AUTH=true).","s":"Adding HTTP Auth to Configuration","u":"/docs/authentication","h":"#adding-http-auth-to-configuration","p":13},{"i":34,"t":"With basic auth (and without HTTP auth), the login logic runs on the client-side. A technical user could inspect the code and view parts of your configuration, including password hashes. If the SHA-256 hash is of a common password, it may be possible to determine it using a lookup table, and then use that to generate a valid auth token. Therefore, you should always use a long, strong and unique password. If your instance is exposed to the internet, the built-in auth alone is not sufficient - use a reverse proxy with its own authentication layer (see Alternative Authentication Methods), or access Dashy over a VPN. See the Network Exposure section in the management docs for more on this. The built-in login page prevents casual unauthorized access on a private network. It's not a security perimeter. ⬆️ Back to Top","s":"Security","u":"/docs/authentication","h":"#security","p":13},{"i":36,"t":"If you'd like to protect server-side endpoints with HTTP Basic Auth, there are two approaches. They protect the same endpoints but use different credential sources, so pick one - don't combine them.","s":"HTTP Auth","u":"/docs/authentication","h":"#http-auth","p":13},{"i":38,"t":"This is the approach described in Adding HTTP Auth to Configuration above. Set ENABLE_HTTP_AUTH=true and it uses the same appConfig.auth.users from your conf.yml. The frontend handles authentication automatically using the session token from the login page, so no extra setup is needed. This is the recommended approach because it keeps credentials in one place and works together with the client-side login page. But the drawback is that your credentials will be stored in your config file.","s":"Using config-file users (recommended)","u":"/docs/authentication","h":"#using-config-file-users-recommended","p":13},{"i":40,"t":"If you don't have users in your conf.yml (e.g. you handle user management externally, or just want a single shared password for server-side access), you can set the BASIC_AUTH_USERNAME and BASIC_AUTH_PASSWORD environmental variables instead. With this approach, there is no Dashy login page. When the browser first requests the config file, the server responds with a 401 and the browser shows its native HTTP auth prompt. Once the user enters the correct credentials, the browser caches them for the session and all subsequent requests work. To skip the browser prompt and have the frontend authenticate automatically, also set VITE_APP_BASIC_AUTH_USERNAME and VITE_APP_BASIC_AUTH_PASSWORD to the same values. These are baked in at build time, so a rebuild is required, and you should only do this on a trusted network. warning Do not combine BASIC_AUTH_USERNAME/BASIC_AUTH_PASSWORD with conf.yml users. If both are present, the server will log a warning at startup. With ENABLE_HTTP_AUTH set, config-file users take priority and the static credentials are ignored. Without it, the static credentials protect the server but the Dashy login page will use conf.yml credentials, and the frontend will send the wrong credentials to server endpoints. Pick one approach or the other. ⬆️ Back to Top","s":"Using static credentials","u":"/docs/authentication","h":"#using-static-credentials","p":13},{"i":42,"t":"Dashy also supports using a Keycloak authentication server. The setup for this is a bit more involved, but it gives you greater security overall, useful for if your instance is exposed to the internet. Keycloak is a Java-based open source, high-performance, secure authentication system, supported by RedHat. It is easy to setup (with Docker), and enables you to secure multiple self-hosted applications with single-sign-on using standard protocols (OpenID Connect, OAuth 2.0, SAML 2.0 and social login). It's also very customizable, you can write or use custom themes, plugins, password policies and more. The following guide will walk you through setting up Keycloak with Dashy. If you already have a Keycloak instance configured, then skip to Step 3.","s":"Keycloak","u":"/docs/authentication","h":"#keycloak","p":13},{"i":44,"t":"First thing to do is to spin up a new instance of Keycloak. You will need Docker installed, and can then choose a tag, and pull the container from quay.io/keycloak/keycloak Use the following run command, replacing the attributes (default credentials, port and name), or incorporate this into your docker-compose file. docker run -d \\ -p 8081:8080 \\ --name auth-server \\ -e KEYCLOAK_ADMIN=admin \\ -e KEYCLOAK_ADMIN_PASSWORD=admin \\ quay.io/keycloak/keycloak:25.0 start-dev (The KEYCLOAK_USER / KEYCLOAK_PASSWORD env vars and the /auth URL prefix from Keycloak 16 and older have been replaced. If you are still on 17 or older, set legacySupport: true in your Dashy config later on.) If you need to pull from DockerHub, a non-official image is available here. Or if you would prefer not to use Docker, you can also directly install Keycloak from source, following this guide. You should now be able to access the Keycloak web interface, using the port specified above (e.g. http://127.0.0.1:8081), login with the default credentials, and when prompted create a new password.","s":"1. Deploy Keycloak","u":"/docs/authentication","h":"#1-deploy-keycloak","p":13},{"i":46,"t":"Before we can use Keycloak, we must first set it up with some users. Keycloak uses Realms (similar to tenants) to create isolated groups of users. You must create a Realm before you will be able to add your first user. Head over to the admin console In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm' Give your realm a name, and hit 'Create' You can now create your first user. In the left-hand menu, click 'Users', then 'Add User' Fill in the form. On Keycloak 25 and newer, First name and Last name are required by the default user-profile schema. If you skip them the user can sign in but login will then fail with \"Account is not fully set up\" Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login Next, create a new client for Dashy. Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right Choose a 'Client ID' (e.g. dashy), set 'Client Protocol' to 'openid-connect' Turn Client authentication OFF and leave Standard flow enabled. Dashy is a SPA, so it acts as an OAuth public client with PKCE. A confidential client requires a client_secret that a browser app can't safely hold For 'Valid Redirect URIs' put the URL where you host Dashy (e.g. https://dashy.example.com/*, or just * while testing locally). Do the same for the 'Web Origins' field Make note of your client-id, and click 'Save' For the adminRole check to work, the role must appear in the id_token (Keycloak's default mapper only adds it to the access token): Open your dashy client, go to the Client scopes tab, click the dedicated scope row (dashy-dedicated) Add a new mapper of type User Realm Role, name it (e.g. realm_roles), claim name realm_access.roles, multivalued ON, Add to ID token ON, Add to access token ON (Optional, for adminGroup instead of adminRole) Add a second mapper of type Group Membership, claim name groups To create the admin role itself and grant it to a user: Realm roles in the left-hand menu, Create role, name it (e.g. dashy-admin) Users → pick your admin user → Role mapping → Assign role → select dashy-admin","s":"2. Setup Keycloak Users","u":"/docs/authentication","h":"#2-setup-keycloak-users","p":13},{"i":48,"t":"Now that your Keycloak instance is up and running, all that's left to do is to configure Dashy to use it. Under appConfig, set auth.enableKeycloak: true, then fill in the details in auth.keycloak, including: serverUrl - the URL where your Keycloak instance is hosted, realm - the name you gave your Realm, and clientId - the Client ID you chose. For example: appConfig: # ... disableConfigurationForNonAdmin: true auth: enableKeycloak: true keycloak: serverUrl: 'http://localhost:8081' realm: 'alicia-homelab' clientId: 'dashy' adminRole: 'dashy-admin' # role name that grants admin privileges Note that if you are using Keycloak V 17 or older, you will also need to set legacySupport: true (also under appConfig.auth.keycloak). This is because the API endpoint was updated in later versions. If you use Keycloak with an external Identity Provier, you can set the idpHint: 'alias-of-kc-idp' option to allow the IdP Hint to be passed to Keycloak. This will cause Keycloak to skip its login page and redirect the user directly to the specified IdP's login page. Set to the value of the 'Alias' field of the desired IdP as defined in Keycloak under 'Identity Providers'.","s":"3. Enable Keycloak in Dashy Config File","u":"/docs/authentication","h":"#3-enable-keycloak-in-dashy-config-file","p":13},{"i":50,"t":"Keycloak allows you to assign users roles and groups. You can use these values to configure who can access various sections or items in Dashy. Keycloak server administration and configuration is a deep topic; please refer to the server admin guide to see details about creating and assigning roles and groups. Once you have groups or roles assigned to users you can configure access under each section or item displayData.showForKeycloakUser and displayData.hideForKeycloakUser. Both show and hide configurations accept a list of groups and roles that limit access. If a users data matches one or more items in these lists they will be allowed or excluded as defined. sections: - name: DeveloperResources displayData: showForKeycloakUsers: roles: ['canViewDevResources'] hideForKeycloakUsers: groups: ['ProductTeam'] items: - title: Not Visible for developers displayData: hideForKeycloakUsers: groups: ['DevelopmentTeam'] Your app is now secured :) When you load Dashy, it will redirect to your Keycloak login page, and any user without valid credentials will be prevented from accessing your dashboard. From within the Keycloak console, you can then configure things like time-outs, password policies, etc. You can also backup your full Keycloak config, and it is recommended to do this, along with your Dashy config. You can spin up both Dashy and Keycloak simultaneously and restore both applications configs using a docker-compose.yml file, and this is recommended.","s":"4. Add groups and roles (Optional)","u":"/docs/authentication","h":"#4-add-groups-and-roles-optional","p":13},{"i":52,"t":"If Dashy and Keycloak run on different origins (typical when testing locally on different localhost: ports), Keycloak's default Content-Security-Policy: frame-ancestors 'self' and X-Frame-Options: SAMEORIGIN block the hidden iframe keycloak-js uses to check your session. Symptom: a generic \"Authentication failed (Keycloak)\" toast on first load. To allow the iframe, open Realm settings → Security defenses → Browser headers, clear X-Frame-Options, and change Content-Security-Policy to frame-src 'self' ; frame-ancestors 'self' ; object-src 'none';. Same-origin production deployments don't hit this.","s":"CORS Headers","u":"/docs/authentication","h":"#cors-headers","p":13},{"i":54,"t":"Dashy's server reads auth.keycloak from conf.yml at boot, lazily fetches your Keycloak realm's OIDC discovery doc + JWKS, then verifies the id_token the SPA attaches to every API call as Authorization: Bearer . Tokens that fail signature / issuer / audience / expiry verification are rejected with 401. Write endpoints (POST /config-manager/save) additionally require the adminRole (or adminGroup) to be present in the token claims, and non-admins receive 403. Note that the /conf.yml remains anonymously readable still (so the SPA can bootstrap before login). The admin check reads the role / group claim from the id_token, so the client mapper from Step 2 above (the one with Add to ID token on) is what makes adminRole / adminGroup work. Without it the server gets a token with no roles claim and treats everyone as non-admin.","s":"How server-side enforcement works","u":"/docs/authentication","h":"#how-server-side-enforcement-works","p":13},{"i":56,"t":"If you encounter issues with your Keycloak setup, follow these steps to troubleshoot and resolve common problems. Client Authentication Issue Problem: Redirect loop, if client authentication is enabled. Solution: Switch off \"client authentication\" in \"TC clients\" -> \"Advanced\" settings. Double URL Problem: If you get redirected to \"https://dashy.my.domain/#iss=https://keycloak.my.domain/realms/my-realm\" Solution: Make sure to turn on \"Exclude Issuer From Authentication Response\" in \"TC clients\" -> \"Advanced\" -> \"OpenID Connect Compatibility Modes\" Problems with mutiple Dashy Pages Problem: Refreshing or logging out of dashy results in an \"invalid_redirect_uri\" error. Solution: In \"TC clients\" -> \"Access settings\" -> \"Root URL\" https://dashy.my.domain/, valid redirect URIs must be /*","s":"Troubleshooting Keycloak","u":"/docs/authentication","h":"#troubleshooting-keycloak","p":13},{"i":58,"t":"Header authentication allows Dashy to trust an upstream reverse proxy to handle authentication. The proxy authenticates users and forwards their identity to Dashy via a configurable HTTP header (e.g. REMOTE_USER). This is the standard pattern used by Authelia, Authentik, Traefik's forwardAuth, Caddy's forward_auth, and Nginx's auth_request. This is useful when you already have a central authentication layer in front of your self-hosted services and want Dashy to automatically pick up the authenticated user without requiring a separate login.","s":"Header Authentication","u":"/docs/authentication","h":"#header-authentication","p":13},{"i":60,"t":"appConfig: auth: enableHeaderAuth: true users: - user: alice hash: 0a7b1d4c2e... # SHA-256 hash of password type: admin - user: bob hash: 3f8e2b1a9d... type: normal headerAuth: userHeader: Remote-User proxyWhitelist: - 172.18.0.2 - 127.0.0.1 userHeader - The HTTP header name containing the authenticated username. Defaults to Remote-User if not specified. Common values: Remote-User (Authelia), X-authentik-username (Authentik), or whatever your proxy forwards. Header matching is case-insensitive. proxyWhitelist - Required. An array of IP addresses that Dashy will accept the header from. Only requests originating from these IPs will be trusted. This prevents clients from spoofing the header directly. users - Required. The header username is matched against this list to determine the user's role (admin or normal) and to generate the session token. Users must be defined here even though authentication is handled externally.","s":"Configuration","u":"/docs/authentication","h":"#configuration","p":13},{"i":62,"t":"User visits Dashy, which is behind a reverse proxy (e.g. Authelia) The proxy authenticates the user and forwards the request with a header like Remote-User: alice Dashy's server checks that the request comes from a whitelisted proxy IP, then returns the username via the /get-user endpoint The client matches the username against the configured users, generates a session token, and sets the auth cookie From this point, standard Dashy auth applies - isLoggedIn(), admin checks, and granular access controls all work as normal","s":"How it Works","u":"/docs/authentication","h":"#how-it-works","p":13},{"i":64,"t":"The proxyWhitelist checks req.socket.remoteAddress, which is the direct connection source. If your proxy connects through Docker networking, use the container's internal IP (e.g. 172.18.0.2), not the external IP Logout clears Dashy's session cookie, but the user remains authenticated at the proxy level. Revisiting the page will re-authenticate automatically When header auth is enabled, server-side API endpoints are also protected by the proxy whitelist. Requests not from a whitelisted IP will be rejected. Admin enforcement applies - only users with type: admin can access write endpoints (config save)","s":"Notes","u":"/docs/authentication","h":"#notes","p":13},{"i":66,"t":"Dashy also supports using a general OIDC compatible authentication server. In order to use it, the authentication section needs to be configured: appConfig: disableConfigurationForNonAdmin: true # Prevent authenticated non-admins using editor auth: enableOidc: true oidc: clientId: 'registered-client-id' endpoint: 'https://your-oidc-provider.example.com' scope: 'openid profile email' adminGroup: admin Because Dashy is a SPA, a public client registration with PKCE is needed. If you set adminGroup, include groups in scope (e.g. scope: 'openid profile email groups') so your IdP actually returns the claim in the id_token. Same goes for adminRole and a roles scope if your IdP needs one. Note, that if your clientId is numeric, you must place it in quotes. Otherwise it will be interpreted as a number and truncated to 64 chars! An example for Authelia is shared below, but other OIDC systems can be used: identity_providers: oidc: clients: - client_id: dashy client_name: dashy public: true authorization_policy: 'one_factor' require_pkce: true pkce_challenge_method: 'S256' redirect_uris: - https://dashy.local # should point to your dashy endpoint grant_types: - authorization_code scopes: - 'openid' - 'profile' - 'roles' - 'email' - 'groups' Groups and roles will be populated and available for controlling display similar to Keycloak above.","s":"OIDC","u":"/docs/authentication","h":"#oidc","p":13},{"i":68,"t":"Dashy's server reads auth.oidc from conf.yml at boot, lazily fetches the OIDC discovery doc + JWKS from your endpoint, then verifies the id_token the SPA attaches to every API call as Authorization: Bearer . Tokens that fail signature / issuer / audience / expiry verification are rejected with 401. Write endpoints (POST /config-manager/save) additionally require the adminGroup (or adminRole) to be present in the token's groups / roles claims, and non-admins receive 403. Note that the /conf.yml remains anonymously readable still Your IdP must include groups / roles in the id_token, not only the access token, for the admin check to work (most IdPs do this when the groups scope is requested).","s":"How server-side enforcement works","u":"/docs/authentication","h":"#how-server-side-enforcement-works-1","p":13},{"i":70,"t":"This documentation is specific to authentik, however it may be useful in getting other idP's working with Dashy. This guide will only walk through the following: Creating and configuring an OIDC provider Creating and configuring an application Assigning groups Configuring Dashy to use the OIDC client Show quick examples of how to hide/show pages, items, and sections using OIDC groups This guide assumes the following: You have a working instance of authentik terminated with SSL You have a working instance of Dashy terminated with SSL Users and groups are provisioned You are familiar with how authentik works in case you need to do further troubleshooting that is outside the scope of this guide. tip It it recommended that you create groups specific for Dashy. Groups will allow you to display content based on group membership as well as limiting user access to Dashy. If you do not need this functionality, then you can forgo creating specific groups. tip You can use the application wizard to create the provider and application at one time. This is the recommended route, but only the manual process will be outlined in this guide. 1. Create an OIDC provider​ Login to the admin console for authentik. Go to Applications > Providers. Click Create. A dialog box will pop-up, select the OAuth2/OpenID Provider. Click Next. On the next page of the wizard, set the Name, Authentication flow, Authorization flow, and Invalidation flow. See example below. Using the default-provider-authorization-implicit-consent authorization flow on internal services and default-provider-authorization-explicit-consent on external services is a common practice. However, it is fully up to you on how you would like to configure this option. Implicit will login directly without user consent, explicit will ask if the user approves the service being logged into with their user credentials. For the invalidation flow (required on Authentik 2023.10 and later) the built-in default-provider-invalidation-flow is fine. Scroll down and configure the Protocol settings. Set the Client type to Public. Add the Redirect URIs/Origins (RegEx). If the site is hosted at dashy.lan.domain.com, then you would enter as the example below. note If you have an internal and external domain for Dashy, enter both URI's. Enter each URI on a new line. Scroll down to set the Signing Key. It is recommended to use the built in authentik Self-signed Certificate here unless you have special needs for your own custom cert. If you plan to use adminGroup in your Dashy config, you need a groups scope mapping first. Authentik does not ship one by default. Open Customisation > Property Mappings in a new tab, click Create > Scope Mapping, set Name to groups, Scope name to groups, and Expression to: return {\"groups\": [g.name for g in request.user.ak_groups.all()]} Save it, then come back to the provider wizard. Expand Advanced protocol settings then verify the Scopes are set to what is highlighted in white below (including the groups mapping you just created, if you want adminGroup to work). Set the Subject mode to Based on the Users's Email. Lastly, toggle Include claims in id_token to on. Click Finish to complete creating the provider. Grab the generated Client ID and OpenID Configuration Issuer URL by clicking the newly created provider as this will use this later when Dashy is configured to use the OIDC auth mechanism. In this tutorial, what was generated is used below. Obviously adjust the Client ID that was generated and use your domain here for the issuer. Client ID: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15 OpenID Configuration Issuer: https://auth.domain.com/application/o/dashy/ 2. Create an application​ Make sure you are still in the authentik admin console then go to Applications > Applications. Click Create. Next, it is required to give a user facing Name, Slug and assign the newly created provider. Use the example below if you have been following the guide. If you have used your own naming, then adjust accordingly. Click Create once you are done. tip Open the application in a new tab from the authentik user portal and upload a custom icon. You can also enter a user facing Description that the user would see. 3. (Optional) Limiting access via authentik with groups​ If you would like to deny Dashy access from specific users who are not within authentik based groups, you bind them to the application you just created now. authentik will deny access to those who are not members of this group or groups. If you want to allow everyone access from your authentik instance, skip this step. Make sure you are still in the authentik admin console then go to Applications > Applications. Click the newly created Dashy application. Click the Policy/Group/User Bindings tab at the top, then click Bind existing policy. This assumes you have already created the groups you want to use for Dashy and populated users in those groups. Click Group for the binding type. Under Group select the appropriate group you would like to bind. Make sure Enabled is toggeled on. Click Create. Dashy will now be scoped only to users within the assigned groups you have bound the application to. Keep adding groups if you would like to adjust the dashboard visibilty based on group membership. 4. Configure Dashy to use OIDC client​ important It is highly recommended to edit your conf.yml directly for this step. caution Do not make the same mistake many have made here by including the fully qualified address for the OpenID Configuration URL. Dashy will append the .well-known configuration automatically. If the .well-known URI is included the app will get redirect loops and 400 errors. Enter the Client ID in the clientId field and OpenID Configuration Issuer in the endpoint field. Below is how to configure the auth section in the yaml syntax. Once this is enabled, when an attempt to access Dashy is made it will now redirect you to the authentik login page moving forward. appConfig: theme: glass layout: auto iconSize: medium disableConfigurationForNonAdmin: true # Prevent logged-in, non-admins using the view/edit config features auth: enableOidc: true oidc: clientId: pzN9DCMLqHTTatgtYFg50cl0jn1NmCyBC3wreX15 endpoint: https://auth.domain.com/application/o/dashy/ 5. (OPTIONAL) Example snippets for dashboard visibility​ Using the hideForKeycloakUsers configuration option is needed to use the authentik groups that were created previously. Adjusting pages visibility: pages: - name: App Management path: appmgmt.yml displayData: hideForKeycloakUsers: groups: - Dashy Users - name: Network Management path: network.yml displayData: hideForKeycloakUsers: groups: - Dashy Users Adjusting items visibility: items: - title: Authentik Admin icon: authentik.svg url: https://auth.domain.com/if/admin/ target: newtab id: 0_1472_authentikadmin displayData: hideForKeycloakUsers: groups: - Dashy Users - title: Authentik User icon: authentik-light.png url: https://auth.domain.com/if/user/ target: newtab id: 1_1472_authentikuser Adjusting sections visibility: sections: - name: Authentication displayData: sortBy: default cols: 1 collapsed: false hideForGuests: false hideForKeycloakUsers: groups: - Dashy Users","s":"authentik","u":"/docs/authentication","h":"#authentik","p":13},{"i":72,"t":"These are alternatives to Dashy's built-in auth, Keycloak, and OIDC. Most of them sit in front of Dashy at the network or reverse proxy level, which is generally the better approach for anything internet-facing. Reverse Proxy Auth - Authelia, Authentik, or similar sitting in front of Dashy Zero-Trust Tunnels - Cloudflare Tunnel, Tailscale Funnel VPN - Keep Dashy off the internet entirely IP-Based Access - Restrict by source IP in your web server Web Server Authentication - HTTP basic auth at the proxy level SSO / OAuth Providers - Cloud-hosted identity providers Cloud Hosting Providers - Built-in auth on hosting platforms","s":"Alternative Authentication Methods","u":"/docs/authentication","h":"#alternative-authentication-methods","p":13},{"i":74,"t":"The most common setup for self-hosters running multiple services. You put an auth server in front of your reverse proxy, and it handles login, 2FA, and sessions for everything behind it. You configure it once, and all your apps get protected. Dashy has Header Authentication support, so when your proxy authenticates a user and forwards their identity via a header, Dashy picks up the username and maps it to a configured user automatically. No separate Dashy login needed. Authelia is lightweight and Docker-friendly. It supports 2FA, per-path access rules, and multiple user backends. To get started quickly: git clone https://github.com/authelia/authelia.git cd authelia/examples/compose/lite Edit users_database.yml, configuration.yml, and docker-compose.yml for your domain and users docker compose up -d See the Authelia docs for the full setup guide. Authentik is heavier but gives you a proper admin UI, built-in OIDC/SAML support, and user self-service (password resets, enrollment flows, etc). Good if you want a single identity provider across many apps. See the authentik Docker Compose install to get started, and the authentik section above for Dashy-specific OIDC config.","s":"Reverse proxy auth","u":"/docs/authentication","h":"#reverse-proxy-auth","p":13},{"i":76,"t":"These let you expose Dashy to the internet without opening inbound ports or configuring port forwarding. Auth is handled by the tunnel provider before traffic ever reaches your server. Cloudflare Tunnel connects Dashy to Cloudflare's edge network via an outbound-only cloudflared daemon (runs nicely as a Docker sidecar). Cloudflare handles DNS, TLS, and DDoS protection. Pair it with Cloudflare Access to require identity provider login before anyone reaches Dashy. The free tier covers most home setups. See the Cloudflare Tunnel docs. Tailscale Funnel exposes Dashy through your Tailscale mesh to the public internet, with automatic TLS. Simpler to set up than Cloudflare but you get less control over access policies. See the Funnel docs.","s":"Zero-trust tunnels","u":"/docs/authentication","h":"#zero-trust-tunnels","p":13},{"i":78,"t":"A VPN keeps Dashy off the public internet entirely. You connect to your home network remotely and access Dashy like you're on the LAN. No auth to configure, no attack surface to worry about. The downside: you need the VPN running to see anything, and some networks (corporate WiFi, hotels) block VPN traffic. WireGuard is fast and minimal. Most self-hosters run it through a UI like wg-easy, which gives you a web interface for managing peers and generating QR codes for mobile. Tailscale wraps WireGuard and takes care of NAT traversal, key exchange, and device management. No port forwarding needed, works across networks with zero config. There's a generous free tier. Headscale is a self-hosted coordination server if you want to keep everything on your own infrastructure. OpenVPN still works fine if you already have it running, but for a new setup WireGuard or Tailscale are easier to get going.","s":"VPN","u":"/docs/authentication","h":"#vpn","p":13},{"i":80,"t":"If you have a static IP or are already on a VPN, you can restrict access to Dashy by source IP at the web server level. This works well as an extra layer on top of other auth methods. NGINX: location / { proxy_pass http://dashy:8080; allow 192.168.1.0/24; allow 203.0.113.50; deny all; } Caddy (request matchers docs): dashy.example.com { @blocked not remote_ip 192.168.1.0/24 203.0.113.50 respond @blocked \"Access denied\" 403 reverse_proxy dashy:8080 } Apache (2.4+): Require ip 192.168.1.0/24 Require ip 203.0.113.50 ","s":"IP-based access","u":"/docs/authentication","h":"#ip-based-access","p":13},{"i":82,"t":"Your reverse proxy can handle HTTP basic auth directly, no extra services needed. This gives you a browser login prompt in front of Dashy. Make sure you're using HTTPS, as basic auth sends credentials base64-encoded (not encrypted) with every request. NGINX (auth module docs): location / { auth_basic \"Dashy\"; auth_basic_user_file /etc/nginx/conf.d/.htpasswd; proxy_pass http://dashy:8080; } Generate the password file with htpasswd -c /etc/nginx/conf.d/.htpasswd alicia. Caddy (basicauth directive): dashy.example.com { basicauth { alicia $2a$14$... # generate with: caddy hash-password } reverse_proxy dashy:8080 } Apache: AuthType Basic AuthName \"Dashy\" AuthUserFile /path/to/.htpasswd Require valid-user Generate the password file with htpasswd -c /path/to/.htpasswd alicia.","s":"Web server authentication","u":"/docs/authentication","h":"#web-server-authentication","p":13},{"i":84,"t":"Cloud identity providers like Auth0, Okta, Ory, and Google Cloud Identity can work with Dashy through its OIDC support. If your provider speaks OIDC (most do), just configure it as described in the OIDC section and you're set. For providers that only support OAuth2 or SAML without an OIDC layer, you'll need something in between to translate. Authentik, Keycloak, and Authelia can all bridge from SAML/OAuth2 to OIDC.","s":"SSO / OAuth providers","u":"/docs/authentication","h":"#sso--oauth-providers","p":13},{"i":86,"t":"If you're running Dashy on a cloud platform, most have their own auth options you can enable without touching Dashy's config. See your provider's docs: Cloudflare Access, Netlify Password Protection, AWS Cognito, Azure App Service Authentication, and Vercel Password Protection. ⬆️ Back to Top","s":"Cloud hosting providers","u":"/docs/authentication","h":"#cloud-hosting-providers","p":13},{"i":88,"t":"Beyond the cloud backup/restore service, there are several other self-hosted options you can use to backup Dashy, and any other Docker container data. These are outlined in the Management docs, at: Docker Backup Options. Dashy has a built-in feature for securely backing up your config to a hosted cloud service, and then restoring it on another instance. This feature is totally optional, and if you do not enable it, then Dashy will not make any external network requests. This is useful not only for backing up your configuration off-site, but it also enables Dashy to be used without having write a YAML config file, and makes it possible to use a public hosted instance, without the need to self-host.","s":"Cloud Backup and Restore","u":"/docs/backup-restore","h":"","p":87},{"i":90,"t":"All data is encrypted before being sent to the backend. In Dashy, this is done in CloudBackup.js, using crypto.js's AES method, using the users chosen password as the key. The data is then sent to a Cloudflare worker (a platform for running serverless functions), and stored in a KV data store.","s":"How it Works","u":"/docs/backup-restore","h":"#how-it-works","p":87},{"i":92,"t":"Once you've got Dashy configured to your preference, open the Backup & Restore menu (click the Cloud icon in the top-right corner). Here you will be prompted to choose a password, which will be used to encrypt your data. If you forget this password, there will be no way to recover your config. After clicking 'Backup' your data will be encrypted, compressed and sent to the hosted cloud service. A backup ID will be returned (in the format of xxxx-xxxx-xxxx-xxxx), this is what you use, along with your password to restore the backup on another system, so take note of it. To update a backup, return to this menu, enter your password, and click 'Update Backup'.","s":"Creating a Backup","u":"/docs/backup-restore","h":"#creating-a-backup","p":87},{"i":94,"t":"To restore a backup, navigate to the Backup & Restore menu, and under restore, enter your backup ID, and the password you chose. Your config file will be downloaded, decrypted and applied to local storage.","s":"Restoring a Backup","u":"/docs/backup-restore","h":"#restoring-a-backup","p":87},{"i":96,"t":"Data is only ever sent to the cloud when the user actively triggers a backup. All transmitted data is first encrypted using AES. Your selected password never leaves your device, and is hashed before being compared. It is only possible to restore a configuration if you have both the backup ID and decryption password. Because the data is encrypted on the client-side (before being sent to the cloud), it is not possible for a man-in-the-middle, government entity, website owner, or even Cloudflare to be able read any of your data. The biggest risk to your data, would be a weak password, or a compromised system. Having said that, although the code uses robust security libraries and is open source- it was never intended to be a security product, and has not been audited, and therefore cannot be considered totally secure - please keep that in mind.","s":"Privacy & Security","u":"/docs/backup-restore","h":"#privacy--security","p":87},{"i":98,"t":"Maximum of 24mb of storage per user. Please do not repeatedly hit the endpoint, as if the quota is exceeded the service may become less available to other users. Abuse may result in your IP being temporarily banned by Cloudflare.","s":"Fair Use Policy","u":"/docs/backup-restore","h":"#fair-use-policy","p":87},{"i":101,"t":"Install Wrangler CLI Tool: npm i -g @cloudflare/wrangler Log into Cloudflare account: wrangler login Create a new project: wrangler generate my-project Install dependencies: cd my-project && npm i","s":"Quick Start","u":"/docs/backup-restore","h":"#quick-start","p":87},{"i":103,"t":"Add your account_id (found on the right sidebar of the Workers or Overview Dashboard) Add your zone_id (found in the Overview tab of your desired domain on Cloudflare) Add your route, which should be a domain or host, supporting a wildcard name = \"dashy-worker\" type = \"javascript\" workers_dev = true route = \"example.com/*\" zone_id = \"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\" account_id = \"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\" kv_namespaces = [ { binding = \"DASHY_CLOUD_BACKUP\", id = \"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\" } ]","s":"Populate wrangler.toml","u":"/docs/backup-restore","h":"#populate-wranglertoml","p":87},{"i":105,"t":"Write code to handle your requests, and interact with any other data sources in this file Generally, this is done within an event listener for 'fetch', and returns a promise For Example: addEventListener('fetch', event => { event.respondWith(handleRequest(event.request)) }) async function handleRequest(request) { return new Response('Hello World!', { headers: { 'content-type': 'text/plain' }, }) } For the code used for Dashy's cloud service, see here","s":"Complete index.js","u":"/docs/backup-restore","h":"#complete-indexjs","p":87},{"i":107,"t":"wrangler dev - To start the wrangler development server wrangler publish - To publish to your cloudflare account (first run wrangler login)","s":"Commands","u":"/docs/backup-restore","h":"#commands","p":87},{"i":109,"t":"There are four endpoints, and to keep things simple, they all use the same base URL/ route. GET - Get config for a given user backupId - The ID of the desired encrypted object subHash - The latter half of the password hash, to verify ownership POST - Save a new config object, and returns backupId userData - The encrypted, compressed and stringified user config subHash - The latter half of the password hash, to verify ownership PUT - Update an existing config object backupId - The ID of the object to update subHash - Part of the hash, to verify ownership of said object userData - The new data to store DELETE - Delete a specified config object backupId - The ID of the object to be deleted subHash - Part of the password hash, to verify ownership of the object For more info, see the API Docs. If you are using Postman, you may find this pre-made collection helpful in getting things setup.","s":"API","u":"/docs/backup-restore","h":"#api","p":87},{"i":111,"t":"All app configuration is specified in /user-data/conf.yml which is in YAML Format format. If you're using Docker, this file can be passed in as a volume. Changes can either be made directly to this file, or done through the UI. From the UI you can also export, backup, reset, validate and download your configuration file.","s":"Configuring","u":"/docs/configuring","h":"","p":110},{"i":113,"t":"Directly in the YAML file (5/5 reliability, 3/5 usability) Write changes directly to the conf.yml file, optionally using one of the templates provided. This can be done in your favorite editor and uploading to your server, or directly editing the file via SSH, but the easiest method would be to use Code Server UI JSON Editor (4/5 reliability, 4/5 usability) From the UI, under the config menu there is a JSON editor, with built-in validation, documentation and advanced options UI Visual Editor (3/5 reliability, 5/5 usability) From the UI, enter the Interactive Edit Mode, then click any part of the page to edit. Changes are previewed live, and then saved to disk REST API (Coming soon) Programmatically edit config either through the command line, using a script or a third-party application","s":"There are three ways to edit the config","u":"/docs/configuring","h":"#there-are-three-ways-to-edit-the-config","p":110},{"i":115,"t":"You may find it helpful to look at some sample config files to get you started, a collection of which can be found here You can check that your config file fits the schema, by running yarn validate-config After modifying your config, the app needs to be recompiled, by running yarn build - this happens automatically if you're using Docker It is recommended to keep a backup of your config file. You can download it under Config menu, or use the Cloud Backup feature. You can make use of YAML features, like anchors, comments, multi-line strings, etc to reuse attributes and keep your config file readable Once you have finished configuring your dashboard, you can choose to disable UI config if you wish All fields are optional, unless otherwise stated. The following file provides a reference of all supported configuration options.","s":"Tips","u":"/docs/configuring","h":"#tips","p":110},{"i":117,"t":"pageInfo - Header text, footer, title, navigation, etc navLinks - Links to display in the navigation bar pages - List of additional config files, for multi-page dashboards appConfig - Main application settings webSearch - Configure web search engine options hideComponents - Show/ hide page components auth - Built-in authentication setup users - List or users (for simple auth) keycloak - Auth config for Keycloak headerAuth - Auth config for HeaderAuth sections - List of sections displayData - Section display settings show/hideForKeycloakUsers - Set user controls icon - Icon for a section items - List of items icon - Icon for an item displayData - Item display settings show/hideForKeycloakUsers - Set user controls widgets - List of widgets Notes Editing Config through the UI About YAML Config Saving Methods Preventing Changes Example","s":"Contents","u":"/docs/configuring","h":"#contents","p":110},{"i":119,"t":"Field Type Required Description pageInfo object Required Basic meta data like title, description, nav bar links, footer text. See pageInfo appConfig object Optional Settings related to how the app functions, including API keys and global styles. See appConfig sections array Required An array of sections, each containing an array of items, which will be displayed as links. See section pages array Optional An array additional config files, used for multi-page dashboards. See pages ⬆️ Back to Top","s":"Top-Level Fields","u":"/docs/configuring","h":"#top-level-fields","p":110},{"i":121,"t":"Field Type Required Description title string Required Your dashboard title, displayed in the header and browser tab description string Optional Description of your dashboard, also displayed as a subtitle navLinks array Optional Optional list of a maximum of 6 links, which will be displayed in the navigation bar. See navLinks footer string Optional Text to display in the footer. When omitted, no footer is rendered. Supports inline HTML (sanitized before render) logo string Optional The path to an image to display in the header (to the right of the title). This can be either local, where / is the root of ./public, or any remote image, such as https://i.ibb.co/yhbt6CY/dashy.png. It's recommended to scale your image down, so that it doesn't impact load times favicon string Optional URL or path to a custom favicon shown in the browser tab. Can be absolute (https://...), root-relative (/icons/x.png), or a data: URI color string Optional Theme colour applied to the browser chrome (mobile address bar). Any valid CSS color (e.g. #ff00a7) is accepted ⬆️ Back to Top","s":"PageInfo","u":"/docs/configuring","h":"#pageinfo","p":110},{"i":123,"t":"Field Type Required Description title string Required The text to display on the link button path string Required The URL to navigate to when clicked. Can be relative (e.g. /about) or absolute (e.g. https://example.com or http://192.168.1.1) target string Optional The opening method (external links only). Can be either newtab, sametab, top or parent. Defaults to newtab ⬆️ Back to Top","s":"pageInfo.navLinks (optional)","u":"/docs/configuring","h":"#pageinfonavlinks-optional","p":110},{"i":125,"t":"Field Type Required Description name string Required A unique name for that page path string Required The path (local or remote) to the config file to use. For files located within /user-data, you only need to specify filename, for externally hosted files you must include the full URL For more info, see theMulti-Page docs ⬆️ Back to Top","s":"pages[] (optional)","u":"/docs/configuring","h":"#pages-optional","p":110},{"i":127,"t":"Field Type Required Description language string Optional The 2 (or 4-digit) ISO 639-1 code for your language, e.g. en or en-GB. This must be a language that the app has already been translated into. If your language is unavailable, Dashy will fallback to English. By default Dashy will attempt to auto-detect your language, although this may not work on some privacy browsers. startingView enum Optional Which view to land on when visiting /. One of home, minimal or workspace. Defaults to home. Applied at runtime, so no rebuild is needed. You can always switch views from the UI. (Legacy value default is accepted as an alias for home.) defaultOpeningMethod enum Optional The default opening method for items, if no target is specified for a given item. Can be either newtab, sametab, modal, workspace, clipboard, top or parent. Defaults to newtab statusCheck boolean Optional When set to true, Dashy will ping each of your services and display their status as a dot next to each item. This can be overridden by setting statusCheck under each item. Defaults to false statusCheckInterval number Optional The number of seconds between checks. If set to 0 then service will only be checked on initial page load, which is usually the desired functionality. If value is less than 10 you may experience a hit in performance. Defaults to 0 statusCheckAccessibility boolean Optional When set to true, status indicators will use distinct shapes to indicate status for color-blind users. Defaults to false webSearch object Optional Configuration options for the web search feature, set your default search engine, opening method or disable web search. See webSearch backgroundImg string Optional Path to an optional full-screen app background image. This can be either remote (http) or local (relative to /app/public/item-icons/ inside the container). Note that this will slow down initial load enableFontAwesome boolean Optional If set to true font-awesome will be loaded, if set to false they will not be. if left blank font-awesome will be enabled only if required by 1 or more icons enableMaterialDesignIcons boolean Optional If set to true mdi icons will be loaded, if set to false they will not be. Where true is enabled, if left blank material design icons will be enabled only if required by 1 or more icons fontAwesomeKey string Optional If you have a font-awesome key, then you can use it here and make use of premium icons. It is a 10-digit alpha-numeric string from you're FA kit URL (e.g. 13014ae648) faviconApi enum Optional Only applicable if you are using favicons for item icons. Specifies which service to use to resolve favicons. Set to local to do this locally, without using an API. Services running locally will use this option always. Available options are: local, allesedv, iconhorse, faviconkit, duckduckgo, yandex, google, besticon, webmasterapi and mcapi. Defaults to allesedv. See Icons for more info auth object Optional All settings relating to user authentication. See auth defaultIcon string Optional An icon to be applied to any items, which don't already have an icon set. See Icon Docs for more info layout enum Optional Layout for homepage, one of auto, horizontal, vertical or masonry. Defaults to auto. Specifies the layout and direction of how sections are positioned on the home screen. auto uses a responsive CSS grid where each section's footprint is controlled by displayData.cols and displayData.rows. masonry uses a responsive grid where heights follow content, so shorter sections flow into the gaps left by taller neighbours (rows is ignored, cols still controls width). This can also be modified and overridden from the UI. iconSize enum Optional The size of link items / icons. Can be either small, medium, or large. Defaults to medium. This can also be set directly from the UI. colCount number Optional The number of columns of sections displayed on the homepage, using the default view. Should be in integer between 1 and 8. Note that by default this is applied responsively, based on current screen size, and specifying a value here will override this behavior, which may not be desirable. contentMaxWidth string or number Optional Sets the max width of the main sections area on the homepage, overriding the responsive default. Can be a percentage, or any CSS unit theme string Optional The default theme for first load (you can change this later from the UI) dayTheme string Optional Theme to apply when the OS is set to light mode. Overrides theme on initial load nightTheme string Optional Theme to apply when the OS is set to dark mode. Overrides theme on initial load cssThemes string[] Optional An array of custom theme names which can be used in the theme switcher dropdown customColors object Optional Enables you to apply a custom color palette to any given theme. Use the theme name (lowercase) as the key, for an object including key-value-pairs, with the color variable name as keys, and 6-digit hex code as value. See Theming for more info externalStyleSheet string or string[] Optional Either a URL to an external stylesheet or an array or URLs, which can be applied as themes within the UI customCss string Optional Raw CSS that will be applied to the page. This can also be set from the UI. Please minify it first. hideComponents object Optional A list of key page components (header, nav, search, settings) that are present by default, but can be removed using this option. See appConfig.hideComponents enableMultiTasking boolean Optional If set to true, will keep apps open in the background when in the workspace view. Useful for quickly switching between multiple sites, and preserving their state, but comes at the cost of performance. workspaceLandingUrl string Optional The URL or an app, service or website to launch when the workspace view is opened, before another service has been launched preventWriteToDisk boolean Optional If set to true, users will be prevented from saving config changes to disk through the UI preventLocalSave boolean Optional If set to true, users will be prevented from applying config changes to local storage disableConfiguration boolean Optional If set to true, no users will be able to view or edit the config through the UI disableConfigurationForNonAdmin boolean Optional If set to true, only admin users will be able to view or edit the config through the UI. disableConfiguration must not be set to true. widgetsAlwaysUseProxy boolean Optional If set to true, requests made by widgets will always be proxied, same as setting useProxy: true on each widget. Note that this may break some widgets. showSplashScreen boolean Optional If set to true, a loading screen will be shown. Defaults to false. enableErrorReporting boolean Optional Enable reporting of unexpected errors and crashes. This is off by default, and no data will ever be captured unless you explicitly enable it. Turning on error reporting helps previously unknown bugs get discovered and fixed. Dashy uses Sentry for error reporting. Defaults to false. sentryDsn boolean Optional If you need to monitor errors in your instance, then you can use Sentry to collect and process bug reports. Sentry can be self-hosted, or used as SaaS, once your instance is setup, then all you need to do is pass in the DSN here, and enable error reporting. You can learn more on the Sentry DSN Docs. Note that this will only ever be used if enableErrorReporting is explicitly enabled. disableSmartSort boolean Optional For the most-used and last-used app sort functions to work, a basic open-count is stored in local storage. If you do not want this to happen, then disable smart sort here, but you wil no longer be able to use these sort options. Defaults to false. disableUpdateChecks boolean Optional If set to true, Dashy will not check for updates. Defaults to false. enableServiceWorker boolean Optional Service workers cache web applications to improve load times and offer basic offline functionality, and are disabled by default in Dashy. The service worker can sometimes cause older content to be cached, requiring the app to be hard-refreshed. If you do not want SW functionality, or are having issues with caching, set this property to false to disable all service workers. disableContextMenu boolean Optional If set to true, the custom right-click context menu will be disabled. Defaults to false. ⬆️ Back to Top","s":"appConfig (optional)","u":"/docs/configuring","h":"#appconfig-optional","p":110},{"i":129,"t":"note Since the auth is initiated in the main app entry point (for security), a rebuild is required to apply changes to the auth configuration. Run yarn build in the root directory, then restart the server. warning Built-in auth should not be used for security-critical applications, or if your Dashy instance is publicly accessible. For these, it is recommended to use an alternate authentication method. Field Type Required Description users array Optional An array of objects containing usernames and hashed passwords. If this is not provided, then authentication will be off by default, and you will not need any credentials to access the app. See appConfig.auth.users. Note this method of authentication is handled on the client side, so for security critical situations, it is recommended to use an alternate authentication method. enableKeycloak boolean Optional If set to true, then authentication using Keycloak will be enabled. Note that you need to have an instance running, and have also configured auth.keycloak. Defaults to false keycloak object Optional Config options to point Dashy to your Keycloak server. Requires enableKeycloak: true. See auth.keycloak for more info enableHeaderAuth boolean Optional If set to true, then authentication using HeaderAuth will be enabled. Note that you need to have your web server/reverse proxy running, and have also configured auth.headerAuth. Defaults to false headerAuth object Optional Config options to point Dashy to your headers for authentication. Requires enableHeaderAuth: true. See auth.headerAuth for more info enableOidc boolean Optional If set to true, then authentication using OIDC will be enabled. Note that you need to have a configured OIDC server and configure it with auth.oidc. Defaults to false oidc object Optional Config options to point Dash to your OIDC configuration. Request enableOidc: true. See auth.oidc for more info enableGuestAccess boolean Optional When set to true, an unauthenticated user will be able to access the dashboard, with read-only access, without having to login. Requires auth.users to be configured. Defaults to false. For more info, see the Authentication Docs ⬆️ Back to Top","s":"appConfig.auth (optional)","u":"/docs/configuring","h":"#appconfigauth-optional","p":110},{"i":131,"t":"Field Type Required Description user string Required Username to log in with hash string Required A SHA-256 hashed password type string Optional The user type, either admin or normal ⬆️ Back to Top","s":"appConfig.auth.users (optional)","u":"/docs/configuring","h":"#appconfigauthusers-optional","p":110},{"i":133,"t":"Field Type Required Description serverUrl string Required The URL (or URL/ IP + Port) where your keycloak server is running realm string Required The name of the realm (must already be created) that you want to use clientId string Required The Client ID of the client you created for use with Dashy idpHint string Optional The alias of an Identity Provider configured in your realm. If set, Keycloak will skip its login page and redirect straight to that external IdP legacySupport boolean Optional If using Keycloak 17 or older, then set this to true ⬆️ Back to Top","s":"appConfig.auth.keycloak (optional)","u":"/docs/configuring","h":"#appconfigauthkeycloak-optional","p":110},{"i":135,"t":"Field Type Required Description userHeader string Optional The HTTP header name containing the authenticated username (default: Remote-User). Case insensitive proxyWhitelist array Required An array of upstream proxy server IPs to accept authenticated requests from ⬆️ Back to Top","s":"appConfig.auth.headerAuth (optional)","u":"/docs/configuring","h":"#appconfigauthheaderauth-optional","p":110},{"i":137,"t":"Field Type Required Description clientId string Required The client id registered in the OIDC server endpoint string Required The URL of the OIDC server that should be used. adminRole string Optional The role that will be considered as admin. adminGroup string Optional The group that will be considered as admin. scope string Required The scope(s) to request from the OIDC provider ⬆️ Back to Top","s":"appConfig.auth.oidc (optional)","u":"/docs/configuring","h":"#appconfigauthoidc-optional","p":110},{"i":139,"t":"Field Type Required Description disableWebSearch boolean Optional Web search is enabled by default, but can be disabled by setting this property to true. Defaults to false searchEngine string Optional Set the key name for your search engine. Can also use a custom engine by setting this property to custom. Currently supported: duckduckgo, google, whoogle, qwant, startpage, searx-bar and searx-info. Defaults to duckduckgo customSearchEngine string Optional You can also use a custom search engine, or your own self-hosted instance. This requires searchEngine: custom to be set. Then add the URL of your service, with GET query string included here openingMethod string Optional Set your preferred opening method for search results: newtab, sametab, workspace. Defaults to newtab searchBangs object Optional A key-value-pair set of custom search bangs for redirecting query to a specific app or search engine. The key of each should be the bang you will type (typically starting with /, ! or :), and value is the destination, either as a search engine key (e.g. reddit) or a URL with search parameters (e.g. https://en.wikipedia.org/w/?search=) openUrlsDirectly boolean Optional If true, queries that look like URLs will be opened directly instead of searched. Defaults to false ⬆️ Back to Top","s":"appConfig.webSearch (optional)","u":"/docs/configuring","h":"#appconfigwebsearch-optional","p":110},{"i":141,"t":"Field Type Required Description hideHeading boolean Optional If set to true, the page title & sub-title will not be visible. Defaults to false hideNav boolean Optional If set to true, the navigation menu will not be visible. Defaults to false hideSearch boolean Optional If set to true, the search bar will not be visible. Defaults to false hideSettings boolean Optional If set to true, the settings menu will be initially collapsed. Defaults to false ⬆️ Back to Top","s":"appConfig.hideComponents (optional)","u":"/docs/configuring","h":"#appconfighidecomponents-optional","p":110},{"i":143,"t":"Field Type Required Description name string Required The title for the section icon string Optional An single icon to be displayed next to the title. See section.icon items array Optional An array of items to be displayed within the section. See item. Sections must include either 1 or more items, or 1 or more widgets. widgets array Optional An array of widgets to be displayed within the section. See widget displayData object Optional Meta-data to optionally override display settings for a given section. See displayData ⬆️ Back to Top","s":"section","u":"/docs/configuring","h":"#section","p":110},{"i":145,"t":"Field Type Required Description title string Required The text to display/ title of a given item. Max length 18 description string Optional Additional info about an item, which is shown in the tooltip on hover, or visible on large tiles url string Optional The URL / location of web address for when the item is clicked icon string Optional The icon for a given item. Can be a font-awesome icon, favicon, remote URL or local URL. See item.icon target string Optional The opening method for when the item is clicked, either newtab, sametab, modal, workspace, clipboard, top or parent. Where newtab will open the link in a new tab, sametab will open it in the current tab, and modal will open a pop-up modal, workspace will open in the Workspace view and clipboard will copy the URL to system clipboard (but not launch app). Defaults to newtab hotkey number Optional Give frequently opened applications a numeric hotkey, between 0 - 9. You can then just press that key to launch that application. tags string[] Optional A list of tags, which can be used for improved search statusCheck boolean Optional When set to true, Dashy will ping the URL associated with the current service, and display its status as a dot next to the item. The value here will override appConfig.statusCheck so you can turn off or on checks for a given service. Defaults to appConfig.statusCheck, falls back to false statusCheckUrl string Optional If you've enabled statusCheck, and want to use a different URL to what is defined under the item, then specify it here statusCheckHeaders object Optional If you're endpoint requires any specific headers for the status checking, then define them here statusCheckAllowInsecure boolean Optional By default, any request to insecure content will be blocked. Setting this option to true will disable the rejectUnauthorized option, enabling you to ping non-HTTPS services for the current item. Defaults to false statusCheckAcceptCodes string Optional If your service's response code is anything other than 2xx, then you can opt to specify an alternative success code. E.g. if you expect your server to return 403, but still want the status indicator to be green, set this value to 403 statusCheckMaxRedirects number Optional If your service redirects to another page, and you would like status checks to follow redirects, then specify the maximum number of redirects here. Defaults to 0 / will not follow redirects color string Optional An optional color for the text and font-awesome icon to be displayed in. Note that this will override the current theme and so may not display well rel string Optional The value of the rel attribute for the link. Useful for specifying the relationship between the target link/document and Dashy. Defaults to noopener noreferrer backgroundColor string Optional An optional background fill color for the that given item. Again, this will override the current theme and so might not display well against the background provider string Optional The name of the provider for a given service, useful for when including hosted apps. In some themes, this is visible under the item name displayData object Optional Meta-data to optionally override display settings for a given item. See displayData subItems array Optional An optional list of nested sub-items, rendered as smaller icons within the parent. Each sub-item supports title, url, icon, target, color and backgroundColor ⬆️ Back to Top","s":"section.item","u":"/docs/configuring","h":"#sectionitem","p":110},{"i":147,"t":"Field Type Required Description hideForUsers string[] Optional Current item will be visible to all users, except for those specified in this list showForUsers string[] Optional Current item will be hidden from all users, except for those specified in this list hideForGuests boolean Optional Current item will be visible for logged in users, but not for guests (see appConfig.enableGuestAccess). Defaults to false hideForKeycloakUsers object Optional Current item will be visible to all keycloak users, except for those configured via these groups and roles. See hideForKeycloakUsers showForKeycloakUsers object Optional Current item will be hidden from all keycloak users, except for those configured via these groups and roles. See showForKeycloakUsers hideFromWorkspace boolean Optional Current item will be visible in the default view but not in the Workspace view sidebar. Defaults to false ⬆️ Back to Top","s":"item.displayData (optional)","u":"/docs/configuring","h":"#itemdisplaydata-optional","p":110},{"i":149,"t":"Field Type Required Description type string Required The widget type. See Widget Docs for full list of supported widgets options object Optional Some widgets accept either optional or required additional options. Again, see the Widget Docs for full list of options updateInterval number Optional You can keep a widget constantly updated by specifying an update interval, in seconds. See Continuous Updates Docs for more info useProxy boolean Optional Some widgets make API requests to services that are not CORS-enabled. For these instances, you will need to route requests through a proxy, Dashy has a built in CORS-proxy, which you can use by setting this option to true. Defaults to false. See the Proxying Requests Docs for more info timeout number Optional Request timeout in milliseconds, defaults to ½ a second (500) ignoreErrors boolean Optional Prevent an error message being displayed, if a network request or something else fails. Useful for false-positives label string Optional Add custom label to a given widget. Useful for identification, if there are multiple of the same type of widget in a single section ⬆️ Back to Top","s":"section.widgets (optional)","u":"/docs/configuring","h":"#sectionwidgets-optional","p":110},{"i":151,"t":"Field Type Required Description sortBy string Optional The sort order for items within the current section. By default items are displayed in the order in which they are listed in within the config. The following sort options are supported: most-used (most opened apps first), last-used (the most recently used apps), alphabetical, reverse-alphabetical, random and default collapsed boolean Optional If true, the section will be collapsed initially, and will need to be clicked to open. Useful for less regularly used, or very long sections. Defaults to false cutToHeight boolean Optional By default, sections will fill available space. Set this option to true to match section height with content height rows number Optional Height of the section, specified as the number of rows it should span vertically, e.g. 2. Defaults to 1. Max is 5. Applies to the default auto layout; ignored by masonry (where heights follow content) and the flex-based horizontal/vertical layouts. cols number Optional Width of the section, specified as the number of columns the section should span horizontally, e.g. 2. Defaults to 1. Max is 5. Will be clamped to the page's active column count so that a section never exceeds the available grid width. itemSize string Optional Specify the size for items within this group, either small, medium or large. Note that this will override any settings specified through the UI color string Optional A custom accent color for the section, as a hex code or HTML color (e.g. #fff) customStyles string Optional Custom CSS properties that should be applied to that section, e.g. border: 2px dashed #ff0000; sectionLayout string Optional Specify which CSS layout will be used to responsively place items. Can be either auto (which uses flex layout), or grid. Defaults to auto. Setting itemCountX or itemCountY below will also switch the section to grid layout automatically itemCountX number Optional Number of items per row / horizontally. If not set, it will be calculated automatically based on available space. Setting this switches the section to grid layout. Must be a whole number between 1 and 12; values above 8 rely on the grid's responsive column sizing (no explicit minimum-width rule) itemCountY number Optional Number of explicit rows before items flow into implicit rows. Setting this switches the section to grid layout. Must be a whole number between 1 and 12. Row heights size to their content (section heights follow content in the masonry layout) hideForUsers string[] Optional Current section will be visible to all users, except for those specified in this list showForUsers string[] Optional Current section will be hidden from all users, except for those specified in this list hideForGuests boolean Optional Current section will be visible for logged in users, but not for guests (see appConfig.enableGuestAccess). Defaults to false hideForKeycloakUsers object Optional Current section will be visible to all keycloak users, except for those configured via these groups and roles. See hideForKeycloakUsers showForKeycloakUsers object Optional Current section will be hidden from all keycloak users, except for those configured via these groups and roles. See showForKeycloakUsers hideFromWorkspace boolean Optional Current section will be visible in the default view but not in the Workspace view sidebar. Defaults to false ⬆️ Back to Top","s":"section.displayData (optional)","u":"/docs/configuring","h":"#sectiondisplaydata-optional","p":110},{"i":153,"t":"Field Type Required Description icon string Optional The icon for a given item or section. See Icon Docs for all available supported icon types, including: auto-fetched favicons, generative icons, emoji icons, home-lab service logos, font-awesome, simple-icons, material icons, selfh.st icons, and icons specified by URL ⬆️ Back to Top","s":"section.icon and section.item.icon","u":"/docs/configuring","h":"#sectionicon-and-sectionitemicon","p":110},{"i":155,"t":"Field Type Required Description groups string[] Optional Current Section or Item will be hidden or shown based on the user having any of the groups in this list roles string[] Optional Current Section or Item will be hidden or shown based on the user having any of the roles in this list ⬆️ Back to Top","s":"section.displayData.hideForKeycloakUsers, section.displayData.showForKeycloakUsers, item.displayData.hideForKeycloakUsers and item.displayData.showForKeycloakUsers","u":"/docs/configuring","h":"#sectiondisplaydatahideforkeycloakusers-sectiondisplaydatashowforkeycloakusers-itemdisplaydatahideforkeycloakusers-and-itemdisplaydatashowforkeycloakusers","p":110},{"i":158,"t":"Config can be modified directly through the UI, and then written to disk, or applied locally. This can be done wither with the raw config editor (introduced in V 0.6.5 / #3), or the interactive editor (introduced in V 1.8.9 / #298). Interactive Editor JSON Editor","s":"Editing Config through the UI","u":"/docs/configuring","h":"#editing-config-through-the-ui","p":110},{"i":160,"t":"If you're new to YAML, it's pretty straight-forward. The format is exactly the same as that of JSON, but instead of using curly braces, structure is denoted using whitespace. This quick guide should get you up to speed in a few minutes, for more advanced topics take a look at this Wikipedia article.","s":"About YAML","u":"/docs/configuring","h":"#about-yaml","p":110},{"i":162,"t":"When updating the config through the JSON editor in the UI, you have two save options: Local or Write to Disk. Changes saved locally will only be applied to the current user through the browser, and will not apply to other instances - you either need to use the cloud sync feature, or manually update the conf.yml file. On the other-hand, if you choose to write changes to disk, then your main conf.yml file will be updated, and changes will be applied to all users, and visible across all devices. For this functionality to work, you must be running Dashy with using the Docker container, or the Node server. A backup of your current configuration will also be saved in the same directory.","s":"Config Saving Methods","u":"/docs/configuring","h":"#config-saving-methods","p":110},{"i":164,"t":"If you have authentication set up, then any user who is not an admin (with type: admin) will not be able to write changes to disk. You can also prevent changes from any user being written to disk, using preventWriteToDisk. Or prevent any changes from being saved locally in browser storage, using preventLocalSave. To disable all UI config features, set disableConfiguration. Alternatively you can disable UI config features for all non Admin users by setting disableConfigurationForNonAdmin to true.","s":"Preventing Changes","u":"/docs/configuring","h":"#preventing-changes","p":110},{"i":166,"t":"--- pageInfo: title: Home Lab sections: # An array of sections - name: Section 1 - Getting Started items: # An array of items - title: GitHub description: Source code and documentation on GitHub icon: fab fa-github url: https://github.com/Lissy93/dashy - title: Issues description: View currently open issues, or raise a new one icon: fas fa-bug url: https://github.com/Lissy93/dashy/issues - title: Demo description: A live demo icon: far fa-rocket url: https://dashy-demo-1.netlify.app - name: Section 2 - Local Services items: - title: Firewall icon: favicon url: http://192.168.1.1/ - title: Game Server icon: https://i.ibb.co/710B3Yc/space-invader-x256.png url: http://192.168.130.1/ For more example config files, see: this gist If you need any help, feel free to Raise an Issue or Start a Discussion Happy Configuring 🤓🔧 ⬆️ Back to Top","s":"Example","u":"/docs/configuring","h":"#example","p":110},{"i":168,"t":"First off, thank you for considering contributing towards Dashy! 🙌 There are several ways that you can help out, and any contributions, however small will always be very much appreciated. You will be appropriately credited in the readme - huge thank you to everyone who has helped so far 💞","s":"Contributing","u":"/docs/contributing","h":"","p":167},{"i":170,"t":"Help improve Dashy by taking a very short, 6-question survey. This will give me a better understanding of what is important to you, so that I can make Dashy better in the future :)","s":"Take a 2-minute survey","u":"/docs/contributing","h":"#take-a-2-minute-survey","p":167},{"i":172,"t":"Dashy now has a Showcase where you can show off a screenshot of your dashboard, and get inspiration from other users (and I really love seeing how people are using Dashy). To submit your dashboard, either open a PR or raise an issue.","s":"Share your dashboard","u":"/docs/contributing","h":"#share-your-dashboard","p":167},{"i":174,"t":"Donations help to cover server costs, development time and caffeine ;) Don't feel any pressure to donate anything, as Dashy and my other projects will always be 100% free, for everyone, for ever. Sponsoring will give you several perks - for $1 / £0.75 per month, you'll get a sponsor badge on your profile, be credited on the Dashy's readme, with a link to your website/ profile/ socials, get priority support, have your feature ideas implemented, plus lots more. For more info, see @Lissy93's Sponsor Page.","s":"Make a small donation","u":"/docs/contributing","h":"#make-a-small-donation","p":167},{"i":176,"t":"BTC: 3853bSxupMjvxEYfwGDGAaLZhTKxB2vEVC ETH: 0x0fc98cBf8bea932B4470C46C0FbE1ed1f6765017 / aliciasykes.eth XMR: 471KZdxb6N63aABR4WYwMRjTVkc1p1x7wGsUTEF7AMYzL8L94A5pCuYWkosgJQ5Ze8Y2PscVCGZFJa3hDPg6MaDq47GUm8r LTC: MAuck6Ea1qaNihwKfXutkR1R6BorMth86H ZEC: t1bw1SefijsXRDQVxC9w64XsRK8hBhtQohQ","s":"You can also send a one-off small contribution using crypto","u":"/docs/contributing","h":"#you-can-also-send-a-one-off-small-contribution-using-crypto","p":167},{"i":178,"t":"Bug reports helps me to discover bugs I was unaware of, and then fix them, in order to make Dashy more reliable long term. This is a simple, yet really helpful step you can take to help improve Dashy. Sentry is an open source error tracking and performance monitoring tool, which enables the identification any errors which occur in the production app (only if you enable it). To enable error reporting: appConfig: enableErrorReporting: true All reporting is disabled by default, and no data will ever be sent to any external endpoint without your explicit consent. All statistics are anonymized and stored securely. For more about privacy and security, see the Sentry Security Docs.","s":"Enable Anonymous Bug Reports","u":"/docs/contributing","h":"#enable-anonymous-bug-reports","p":167},{"i":180,"t":"If you speak another language, then adding translations will help make Dashy available to non-native English speakers. This is a very quick and easy task, as all application text is located in locales/en.json, so adding a new language is as simple as copying this file and translating the values. You don't have to translate it all, as any missing attributes will just fallback to English. For a full tutorial, see the Multi-Language Support Docs.","s":"Add Translations","u":"/docs/contributing","h":"#add-translations","p":167},{"i":182,"t":"Contributing to the code or docs is super helpful. You can fix a bug, add a new feature or improve an existing one. If you've built your own custom widget, theme or view, consider sharing it in a PR. I've written several guides to help you get started, and the steps for setting up the development environment are outlined in the Development Docs. Feel free to ask if you have any questions.","s":"Submit a PR","u":"/docs/contributing","h":"#submit-a-pr","p":167},{"i":184,"t":"Found a typo, or something that isn't as clear as it could be? Maybe I've missed something off altogether, or you hit a roadblock that took you a while to figure out. Submitting a pull request to add to or improve the documentation will help future users get Dashy up and running more easily. All content is located either in the ./README.md or /docs/ directory, and synced to the Wiki and website using a GH action.","s":"Improve the Docs","u":"/docs/contributing","h":"#improve-the-docs","p":167},{"i":186,"t":"If you've found a bug, then please do raise it as an issue. This will help me know if there's something that needs fixing. Try and include as much detail as possible, such as your environment, steps to reproduce, any console output and maybe an example screenshot or recording if necessary.","s":"Raise a bug","u":"/docs/contributing","h":"#raise-a-bug","p":167},{"i":188,"t":"I've enabled the discussion feature on GitHub, here you can share tips and tricks, useful information, or your dashboard. You can also ask questions, and offer basic support to other users.","s":"Join the discussion","u":"/docs/contributing","h":"#join-the-discussion","p":167},{"i":190,"t":"BountySource is a platform for sponsoring the development of certain features on open source projects. If there is a feature you'd like implemented into Dashy, but either isn't high enough priority or is deemed to be more work than it's worth, then you can instead contribute a bounty towards it's development. You won't pay a penny until your proposal is fully built, and you are satisfied with the result. This helps support the developers, and makes Dashy better for everyone.","s":"Request a feature via BountySource","u":"/docs/contributing","h":"#request-a-feature-via-bountysource","p":167},{"i":192,"t":"Dashy is still a relatively young project, and as such not many people know of it. It would be great to see more users, and so it would be awesome if you could consider sharing with your friends or on social platforms.","s":"Spread the word","u":"/docs/contributing","h":"#spread-the-word","p":167},{"i":194,"t":"Dashy is on the following platforms, and if you could spare a few seconds to give it an upvote or review, this will also help new users discover Dashy","s":"Star, Upvote or Leave a Review","u":"/docs/contributing","h":"#star-upvote-or-leave-a-review","p":167},{"i":196,"t":"If you've enjoyed Dashy, you can follow the me to get updates about other projects that I am working on. If you like, you could also consider subscribing to my mailing list for occasional blog post updates.","s":"Follow for More","u":"/docs/contributing","h":"#follow-for-more","p":167},{"i":198,"t":"For a full list of Dashy's contributors, see the Credits Page","s":"Contributors","u":"/docs/contributing","h":"#contributors","p":167},{"i":200,"t":"](https://seladb.github.io/StarTrack-js/#/preload?r=lissy93,dashy)","s":"Star-Gazers Over Time","u":"/docs/contributing","h":"#star-gazers-over-time","p":167},{"i":206,"t":"This app definitely wouldn't have been quite so possible without the making use of the following package and components. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them. Full credit and big kudos to their respective authors, who've done an amazing job in building and maintaining them","s":"Dependencies","u":"/docs/credits","h":"#dependencies","p":201},{"i":208,"t":"At it's core, the application uses Vue.js, as well as it's services with VueX for state management. Styling is done with SCSS, JavaScript is currently Babel, (but I am in the process of converting to TypeScript). Linting is done with ESLint and Prettier, both following the AirBnB Styleguide. The config is defined in YAML, with a simple Node.js server to serve up the static app and the optional API endpoints, and container deployment is done with Docker.","s":"Core","u":"/docs/credits","h":"#core","p":201},{"i":210,"t":"crypto-js - Encryption implementations by @evanvosberg and community MIT axios - Promise based HTTP client by @mzabriskie and community MIT ajv - JSON schema Validator by @epoberezkin and community MIT i18n - Internationalization plugin by @kazupon and community MIT frappe-charts - Lightweight charting library by @frappe MIT","s":"Utilities","u":"/docs/credits","h":"#utilities","p":201},{"i":212,"t":"vue-select - Dropdown component by @sagalbot MIT vue-js-modal - Modal component by @euvl MIT v-tooltip - Tooltip component by @Akryum MIT vue-material-tabs - Tab view component by @jairoblatt MIT VJsoneditor - Interactive JSON editor component by @yansenlei MIT Forked from JsonEditor by @josdejong Apache-2.0 License vue-swatches - Color palete picker by @saintplay MIT","s":"Frontend Components","u":"/docs/credits","h":"#frontend-components","p":201},{"i":214,"t":"Would you like to be listed here? Whatever your skill set, Dashy needs people like you to help support future development. Check out the Contributing Page for ways that you can get involved. Huge thank you to everyone who has already contributed! 💖","s":"You","u":"/docs/credits","h":"#you","p":201},{"i":216,"t":"Welcome to Dashy, so glad you're here :) Deployment is super easy, and there are several methods available depending on what type of system you're using. If you're self-hosting, then deploying with Docker (or similar container engine) is the recommended approach.","s":"Deployment","u":"/docs/deployment","h":"","p":215},{"i":218,"t":"If you want to skip the fuss, and get straight down to it, then you can spin up a new instance of Dashy by running: docker run -p 8080:8080 lissy93/dashy See Management Docs for info about securing, monitoring, updating, health checks, auto starting, web server configuration, etc Once you've got Dashy up and running, you'll want to configure it with your own content, for this you can reference the configuring docs.","s":"Quick Start","u":"/docs/deployment","h":"#quick-start","p":215},{"i":220,"t":"Deployment Quick Start Deployment Methods Deploy with Docker Using Docker Compose Podman Portainer Kubernetes Unraid Home Server Platforms Synology NAS Build from Source Deploy to Cloud Service Netlify Vercel Easypanel EdgeOne Pages Play-with-Docker Hosting with CDN Requirements System Requirements Docker Bare Metal CDN / Cloud Deploy Browser Support","s":"Deployment Methods","u":"/docs/deployment","h":"#deployment-methods","p":215},{"i":222,"t":"Container Info: Status: Dashy has a built container image hosted on Docker Hub. You will need Docker installed on your system. docker run -d \\ -p 8080:8080 \\ -v /path/to/your/user-data:/app/user-data \\ --name my-dashboard \\ --restart=always \\ lissy93/dashy:latest The user-data directory you mount must contain a conf.yml file. It can also contain any sub-config files, item icons, fonts, custom CSS, or other assets you want served from the web root. Anything you put in there is available at / in the browser. Explanation of the above options: -d Detached mode (not running in the foreground of your terminal) -p The port that should be exposed, and the port it should be mapped to in your host system [host-port]:[container-port], leave the container port as 8080 -v Mounts the host directory containing your conf.yml (and any other assets) into the container at /app/user-data --name Give your container a human-readable name --restart=always Spin up the container when the daemon starts, or after it has been stopped lissy93/dashy:latest The image to run. Replace :latest with a specific version from the tags if needed For all available options, and to learn more, see the Docker Run Docs Dashy is also available through GHCR: docker pull ghcr.io/lissy93/dashy:latest The latest image is multi-arch, so the same tag works on amd64, arm64, and arm/v7 (Raspberry Pi 2+). Docker selects the right variant for your host automatically. The image defaults to :latest, but you can instead specify a specific version, e.g. docker pull lissy93/dashy:4.0.0","s":"Deploy with Docker","u":"/docs/deployment","h":"#deploy-with-docker","p":215},{"i":224,"t":"Using Docker Compose can be useful for saving your specific config in files, without having to type out a long run command each time. Save compose config as a YAML file, and then run docker compose up -d (optionally use the -f flag to specify file location, if it isn't located at ./docker-compose.yml), -d is detached mode (not running in the foreground of your terminal). Compose is also useful if you are using clusters, as the format is very similar to stack files, used with Docker Swarm. The following is a complete example of a docker-compose.yml for Dashy. Run it as is, or uncomment the additional options you need. services: dashy: # The image to pull + version. Can use `ghcr.io/lissy93/dashy` instead image: lissy93/dashy:latest # Optional container name container_name: dashy # Port to serve on (keep container port (second one) as 8080) ports: - 8080:8080 # Mount a directory containing your conf.yml and any other assets volumes: - ./user-data:/app/user-data # Add any env vars for server here, if needed environment: - NODE_ENV=production # Auto-start the container on boot restart: unless-stopped # Healthcheck to determine when container healthy healthcheck: test: ['CMD', 'node', '/app/services/healthcheck.js'] interval: 1m30s timeout: 10s retries: 3 start_period: 30s To pull from GHCR instead of Docker Hub, set image: ghcr.io/lissy93/dashy:latest.","s":"Using Docker Compose","u":"/docs/deployment","h":"#using-docker-compose","p":215},{"i":226,"t":"Podman is a drop-in replacement for Docker that runs containers without a daemon and doesn't require root. If you're on Fedora, RHEL, or just prefer daemonless containers, Podman works with the same images and mostly the same CLI. podman run -d \\ -p 8080:8080 \\ -v /path/to/your/user-data:/app/user-data:Z \\ --name dashy \\ --restart=always \\ docker.io/lissy93/dashy:latest The :Z suffix on the volume mount handles SELinux relabeling, which you'll need on Fedora/RHEL. If you're not using SELinux, you can leave it off. Podman also supports podman-compose or podman compose (with the compose plugin) using the same docker-compose.yml file shown above.","s":"Podman","u":"/docs/deployment","h":"#podman","p":215},{"i":228,"t":"If you manage your Docker host through Portainer, you can deploy Dashy from its UI: Go to Stacks > Add stack Paste the docker-compose.yml contents, or point to the URL Adjust the port and volume mappings as needed Deploy the stack Alternatively, go to Containers > Add container and use the image lissy93/dashy:latest with port 8080 mapped.","s":"Portainer","u":"/docs/deployment","h":"#portainer","p":215},{"i":230,"t":"@vyrtualsynthese has written a Helm Chart for deploying with Kubernetes, available here","s":"Kubernetes","u":"/docs/deployment","h":"#kubernetes","p":215},{"i":232,"t":"Dashy is available through the Community Applications plugin. Search for \"Dashy\" in the Apps tab and install from there. The template pre-fills the Docker image, port mapping, and volume paths for you. If you'd prefer to set it up manually, go to Docker > Add Container and use lissy93/dashy:latest as the repository. Map port 8080, and add a path mapping for the host directory containing your conf.yml to /app/user-data.","s":"Unraid","u":"/docs/deployment","h":"#unraid","p":215},{"i":234,"t":"Several self-hosting platforms include Dashy in their app stores, giving you a one-click install with a management UI: CasaOS - Has Dashy in its built-in app store Cosmos Cloud - Install Dashy from the marketplace Umbrel - Available in the Umbrel App Store Runtipi - Available in the Runtipi App Store These all run Dashy as a Docker container under the hood, so configuration works the same way. You'll find your conf.yml in whichever directory the platform maps to /app/user-data/.","s":"Home Server Platforms","u":"/docs/deployment","h":"#home-server-platforms","p":215},{"i":236,"t":"Installing dashy is really simply and fast: Install Docker via Synology Package Center. Go to File Station and open the docker folder. Inside the docker folder, create one new folder and name it dashy. Note: Be careful to enter only lowercase, not uppercase letters. Go to Control Panel / Task Scheduler / Create / Scheduled Task / User-defined script. Once you click on User-defined script a new window will open. Follow the instructions below: General: In the Task field type in Install dashy. Uncheck \"Enabled\" option. Select root User. Schedule: Select Run on the following date then select \"Do not repeat\". Task Settings: Check \"Send run details by email\", add your email then copy paste the code below in the Run command area. After that click OK. docker run -d \\ -p 4000:8080 \\ -v /volume1/docker/dashy:/app/user-data \\ --name dashy \\ --restart=always \\ lissy93/dashy:latest (Place your conf.yml and any sub-configs / icons / assets inside /volume1/docker/dashy on the host.) dashy should be up within 1-2min after you've started the install task procedure","s":"Synology NAS","u":"/docs/deployment","h":"#synology-nas","p":215},{"i":238,"t":"If you do not want to use Docker, you can run Dashy directly on your host system. For this, you will need both git and the latest or LTS version of Node.js installed, and optionally yarn Get Code: git clone https://github.com/Lissy93/dashy.git and cd dashy Configuration: Fill in your settings in ./user-data/conf.yml Install dependencies: yarn Build: yarn build Run: yarn start","s":"Build from Source","u":"/docs/deployment","h":"#build-from-source","p":215},{"i":240,"t":"Dashy can be deployed to most cloud providers. The Docker guides above work on any VPS, but these providers offer quicker setup for static or containerized deployments. note Static hosting providers (Netlify, Vercel, EdgeOne) won't have status checks or config writing to disk, since those features need Dashy's Node server. Everything else works fine.","s":"Deploy to Cloud Service","u":"/docs/deployment","h":"#deploy-to-cloud-service","p":215},{"i":242,"t":"Dashy includes a netlify.toml so deployment works out of the box. Netlify is free for personal use, supports custom domains, and deploys automatically from your Git repo. Deploy link: https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy","s":"Netlify","u":"/docs/deployment","h":"#netlify","p":215},{"i":244,"t":"Vercel hosts static frontends with a generous free tier, custom domains, and built-in analytics. Deploy link: https://vercel.com/new/project?template=https://github.com/lissy93/dashy","s":"Vercel","u":"/docs/deployment","h":"#vercel","p":215},{"i":246,"t":"Easypanel is a self-hosted server control panel with a Dashy template. It runs the full Docker image, so all features including the Node server work. Template: https://easypanel.io/docs/templates/dashy","s":"Easypanel","u":"/docs/deployment","h":"#easypanel","p":215},{"i":248,"t":"EdgeOne Pages is Tencent's edge hosting platform. Static deploy from your Git repo. Deploy link: https://edgeone.ai/pages/new?repository-url=https://github.com/lissy93/dashy","s":"EdgeOne Pages","u":"/docs/deployment","h":"#edgeone-pages","p":215},{"i":250,"t":"Play with Docker gives you a free, temporary Docker environment in the browser. Good for trying Dashy without installing anything. Sessions last 4 hours. URL: https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml","s":"Play-with-Docker","u":"/docs/deployment","h":"#play-with-docker","p":215},{"i":252,"t":"Once Dashy has been built, it is effectively just a static web app. This means that it can be served up with pretty much any static host, CDN or web server. To host Dashy through a CDN, the steps are very similar to building from source: clone the project, cd into it, install dependencies, write your config file and build the app. Once build is complete you will have a ./dist directory within Dashy's root, and this is the build application which is ready to be served up. However without Dashy's node server, there are a couple of features that will be unavailable to you, including: writing config changes to disk through the UI, and application status checks. Everything else will work fine.","s":"Hosting with CDN","u":"/docs/deployment","h":"#hosting-with-cdn","p":215},{"i":255,"t":"Dashy works well on a Raspberry Pi (tested on Pi 3 and later), but should also run well on any system.","s":"System Requirements","u":"/docs/deployment","h":"#system-requirements","p":215},{"i":257,"t":"The initial build causes a spike in resource usage, but once running it's fairly steady. Minimum 1GB memory and 1GB disk space.","s":"Docker","u":"/docs/deployment","h":"#docker","p":215},{"i":259,"t":"Requires Node.js and Yarn. The engines field in package.json specifies >=18.0.0, but the Docker image is built and tested on Node 24, so that's the recommended version for bare-metal too. Minimum 512MB memory, 2GB disk space.","s":"Bare Metal","u":"/docs/deployment","h":"#bare-metal","p":215},{"i":261,"t":"No specific requirements. The built app (without the Node server) is very lightweight and can be served by any static host or CDN. If you're using custom icons or other assets, additional disk space will be needed.","s":"CDN / Cloud Deploy","u":"/docs/deployment","h":"#cdn--cloud-deploy","p":215},{"i":263,"t":"JavaScript is required. Dashy targets browsers with >1% global usage and the last 2 versions of each (via browserslist). In practice, any modern browser works fine. Internet Explorer is not supported. Browser Minimum Version Status Chrome / Chromium 90+ Fully supported Firefox 90+ Fully supported Edge 90+ Fully supported Safari 14+ Supported Opera 76+ Supported Samsung Internet 15+ Supported Firefox ESR Latest Supported Internet Explorer - Not supported","s":"Browser Support","u":"/docs/deployment","h":"#browser-support","p":215},{"i":265,"t":"This article outlines how to get Dashy running in a development environment, and outlines the basics of the architecture. If you're adding new features, you may want to check out the Development Guides docs, for tutorials covering basic tasks. Setting up the Development Environment Prerequisites Running the App Project Commands Environmental Variables Git Strategy Flow Branches Commit emojis PR Guidelines Resources for Beginners App Info Code Style Guide Application Structure Development Tools Misc / Notes","s":"Developing","u":"/docs/developing","h":"","p":264},{"i":268,"t":"You will need either the latest or LTS version of Node.js to build and serve the application and Git to easily fetch the code, and push any changes. If you plan on running or deploying the container, you'll also need Docker. To avoid any unexpected issues, ensure you've got at least NPM V 7.5 or Yarn 1.22 (you may find NVM helpful for switching/ managing versions).","s":"Prerequisites","u":"/docs/developing","h":"#prerequisites","p":264},{"i":270,"t":"Get Code: git clone https://github.com/Lissy93/dashy.git Navigate into the directory: cd dashy Install dependencies: yarn Start dev server: yarn dev Dashy should now be being served on http://localhost:8080/. Hot reload is enabled, so making changes to any of the files will trigger them to be rebuilt and the page refreshed.","s":"Running the Project","u":"/docs/developing","h":"#running-the-project","p":264},{"i":272,"t":"Basics​ yarn build - Builds the production bundle into ./dist. The Vite dev/runtime server serves user-data/conf.yml on each request, so a rebuild is only needed when source code or assets change, not when config changes yarn start - Starts the Node server, which serves the built site from ./dist and the live config from user-data/. Run yarn build first Development​ yarn dev - Starts the development server with hot reloading yarn lint - Lints code to ensure it follows a consistent, neat style yarn test - Runs tests, and outputs results Utils and Checks​ yarn validate-config - If you have quite a long configuration file, you may wish to check that it's all good to go, before deploying the app. This can be done with yarn validate-config or docker exec -it [container-id] yarn validate-config. Your config file needs to be in /user-data/conf.yml (or within your Docker container at /app/user-data/conf.yml). This will first check that your YAML is valid, and then validates it against Dashy's schema. yarn health-check - Checks that the application is up and running on it's specified port, and outputs current status and response times. Useful for integrating into your monitoring service, if you need to maintain high system availability Alternate Start Commands​ yarn build && yarn start - Builds the app, then starts the production Node server. Use this for a manual production-style run on bare metal. With Vite, conf.yml is served from user-data/ at runtime, so config changes only require a page refresh yarn pm2-start - Starts the Node server using PM2, a process manager for Node.js applications, that helps them stay alive. PM2 has some built-in basic monitoring features, and an optional management solution. If you are running the app on bare metal, it is recommended to use this start command Notes​ If you are using NPM, replace yarn with npm run If you are using Docker, precede each command with docker exec -it [container-id]. Container ID can be found by running docker ps","s":"Project Commands","u":"/docs/developing","h":"#project-commands","p":264},{"i":274,"t":"All environmental variables are optional. Currently there are not many environmental variables used, as most of the user preferences are stored under appConfig in the conf.yml file. You can set variables either in your environment, or using the .env file. NODE_ENV - Current environment, can be either development, production or test PORT - The port to expose the running application on HOST - The host that Dashy is running on, domain or IP BASE_URL - The default base path for serving up static assets VITE_APP_DOMAIN - Usually the same as BASE_URL, but accessible in frontend INTEGRITY - Should enable SRI for build script and link resources IS_DOCKER - Computed automatically on build. Indicates if running in container VITE_APP_VERSION - Again, set automatically using package.json during build time BACKUP_DIR - Directory for conf.yml backups DISABLE_CONFIG_BACKUPS - Set to 'true' to skip the pre-save backup step (useful on read-only filesystems or where permissions don't allow it)","s":"Environmental Variables","u":"/docs/developing","h":"#environmental-variables","p":264},{"i":276,"t":"You can set the environment using the NODE_ENV variable. By default, the correct environment should be selected based on the script you run to start the app. The following environments are supported: production, development and test. For more info, see Vue CLI Environment Modes.","s":"Environment Modes","u":"/docs/developing","h":"#environment-modes","p":264},{"i":279,"t":"Like most Git repos, we are following the Github Flow standard. Create a branch (or fork if you don'd have write access) Code some awesome stuff 🧑‍💻 Add, commit and push your changes to your branch/ fork Head over to GitHub and create a Pull Request Fill in the required sections in the template, and hit submit Follow up with any reviews on your code Merge 🎉","s":"Git Flow","u":"/docs/developing","h":"#git-flow","p":264},{"i":281,"t":"The format of your branch name should be something similar to: [TYPE]/[TICKET]_[TITLE] For example, FEATURE/420_Awesome-feature or FIX/690_login-server-error","s":"Git Branch Naming","u":"/docs/developing","h":"#git-branch-naming","p":264},{"i":283,"t":"Using a single emoji at the start of each commit message, to indicate the type task, makes the commit ledger easier to understand, plus it looks cool. 🎨 :art: - Improve structure / format of the code. ⚡️ :zap: - Improve performance. 🔥 :fire: - Remove code or files. 🐛 :bug: - Fix a bug. 🚑️ :ambulance: - Critical hotfix ✨ :sparkles: - Introduce new features. 📝 :memo: - Add or update documentation. 🚀 :rocket: - Deploy stuff. 💄 :lipstick: - Add or update the UI and style files. 🎉 :tada: - Begin a project. ✅ :white_check_mark: - Add, update, or pass tests. 🔒️ :lock: - Fix security issues. 🔖 :bookmark: - Make a Release or Version tag. 🚨 :rotating_light: - Fix compiler / linter warnings. 🚧 :construction: - Work in progress. ⬆️ :arrow_up: - Upgrade dependencies. 👷 :construction_worker: - Add or update CI build system. ♻️ :recycle: - Refactor code. 🩹 :adhesive_bandage: - Simple fix for a non-critical issue. 🔧 :wrench: - Add or update configuration files. 🍱 :bento: - Add or update assets. 🗃️ :card_file_box: - Perform database schema related changes. ✏️ :pencil2: - Fix typos. 🌐 :globe_with_meridians: - Internationalization and translations. For a full list of options, see gitmoji.dev","s":"Commit Emojis","u":"/docs/developing","h":"#commit-emojis","p":264},{"i":285,"t":"Once you've made your changes, and pushed them to your fork or branch, you're ready to open a pull request! For a pull request to be merged, it must: Must be backwards compatible The build, lint and tests (run by GH actions) must pass There must not be any merge conflicts When you submit your PR, include the required info, by filling out the PR template. Including: A brief description of your changes The issue, ticket or discussion number (if applicable) For UI relate updates include a screenshot If any dependencies were added, explain why it was needed, state the cost associated, and confirm it does not introduce any security issues Finally, check the checkboxes, to confirm that the standards are met, and hit submit!","s":"PR Guidelines","u":"/docs/developing","h":"#pr-guidelines","p":264},{"i":287,"t":"New to Web Development? Glad you're here! Dashy is a pretty simple app, so it should make a good candidate for your first PR. Presuming that you already have a basic knowledge of JavaScript, the following articles should point you in the right direction for getting up to speed with the technologies used in this project: Open Source for Beginners Introduction to Vue.js Vue.js Walkthrough ES6 Features Definitive guide to SCSS Complete beginners guide to Docker Docker Classroom - Interactive Tutorials Quick start TypeScript guide Complete TypeScript tutorial series Using TypeScript with Vue.js Git cheat sheet Basics of using NPM As well as Node, Git and Docker- you'll also need an IDE (e.g. VS Code or Vim) and a terminal (Windows users may find WSL more convenient).","s":"Resources for Beginners","u":"/docs/developing","h":"#resources-for-beginners","p":264},{"i":290,"t":"Linting is done using ESLint, and using the Vue.js Styleguide, which is very similar to the AirBnB Styleguide. You can run yarn lint to report and fix issues. While the dev server is running, issues will be reported to the console automatically, and any lint errors will trigger the build to fail. Note that all lint checks must pass before any PR can be merged. Linting is also run as a git pre-commit hook The most significant things to note are: Indentation should be done with two spaces Strings should use single quotes All statements must end in a semi-colon The final element in all objects must be preceded with a comma Maximum line length is 100 There must be exactly one blank line between sections, before function names, and at the end of the file With conditionals, put else on the same line as your if block's closing brace All multiline blocks must use braces Avoid console statements in the frontend Styleguides: Vue: Vue styleguide JavaScript: github.com/airbnb/javascript","s":"Style Guide","u":"/docs/developing","h":"#style-guide","p":264},{"i":292,"t":"Files in the Root: ./​ ╮ ├── package.json # Project meta-data, dependencies and paths to scripts ├── src/ # Project front-end source code ├── server.js # A Node.js server to serve up the /dist directory ├── services/ # All server-side endpoints and utilities ├── vue.config.js # Vue.js configuration ├── Dockerfile # The blueprint for building the Docker container ├── docker-compose.yml # A Docker run command ├── .env # Location for any environmental variables ├── yarn.lock # Auto-generated list of current packages and version numbers ├── docs/ # Markdown documentation ├── README.md # Readme, basic info for getting started ├── LICENSE.md # License for use ╯ Frontend Source: ./src/​ ./src ├── App.vue # Vue.js starting file ├── assets # Static non-compiled assets │ ├── fonts # .ttf font files │ ├── locales # All app text, each language in a separate JSON file │ ╰── interface-icons # SVG icons used in the app ├── components # All front-end Vue web components │ ├── Charts # Charting components for dynamically displaying widget data │ │ ├── Gauge.vue # A speed-dial style chart for showing 0 - 100 values │ │ ╰── PercentageChart.vue # A horizontal bar for showing percentage breakdowns │ ├── Configuration # Components relating to the user config pop-up │ │ ├── AppInfoModal.vue # A modal showing core app info, like version, language, etc │ │ ├── AppVersion.vue # Shows current version from package.json, compares with GitHub │ │ ├── CloudBackupRestore.vue # Form where the user manages cloud sync options │ │ ├── ConfigContainer.vue # Main container, wrapping all other config components │ │ ├── CustomCss.vue # Form where the user can input custom CSS │ │ ╰── JsonEditor.vue # JSON editor, where the user can modify the main config file │ ├── FormElements # Basic form elements used throughout the app │ │ ├── Button.vue # Standard button component │ │ ├── Radio.vue # Standard radio button input │ │ ├── Select.vue # Standard dropdown input selector │ │ ├── Input.vue # Standard text field input component │ │ ╰── Toggle.vue # Standard on / off toggle switch │ ├── InteractiveEditor # Components for the interactive UI config editor │ │ ├── AddNewSectionLauncher # Button that launches the EditSection form, used for adding new section │ │ ├── EditAppConfig.vue # Form for editing appConfig │ │ ├── EditPageInfo.vue # Form for editing pageInfo │ │ ├── EditSection.vue # Form for adding / editing sections │ │ ├── EditItem.vue # Form for adding or editing items │ │ ├── EditModeSaveMenu.vue # The bar at the bottom of screen in edit mode, containing save buttons │ │ ├── EditModeTopBanner.vue # The bar at the top of screen in edit mode │ │ ├── ExportConfigMenu.vue # Modal for viewing / exporting edited config │ │ ├── MoveItemTo.vue # Form for moving / copying items to other sections │ │ ╰── SaveCancelButtons.vue # Buttons visible in all the edit menus, to save or cancel changes │ ├── LinkItems # Components for Sections and Link Items │ │ ├── Collapsable.vue # The collapsible functionality of sections │ │ ├── IframeModal.vue # Pop-up iframe modal, for viewing websites within the app │ │ ├── Item.vue # Main link item, which is displayed within an item group │ │ ├── ItemGroup.vue # Item group is a section containing icons │ │ ├── ItemIcon.vue # The icon used by both items and sections │ │ ├── ItemOpenMethodIcon.vue # A small icon, visible on hover, indicating opening method │ │ ├── ItemContextMenu.vue # The right-click menu, for showing Item opening methods and info │ │ ├── SectionContextMenu.vue # The right-click menu, for showing Section edit/ open options │ │ ╰── StatusIndicator.vue # Traffic light dot, showing if app is online or down │ ├── Minimal View # Components used for the startpage / minimal alternative view │ │ ├── MinimalHeading.vue # Title part of minimal view │ │ ├── MinimalSearch.vue # Search bar for minimal view │ │ ╰── MinimalSection.vue # Tabbed-Item section for minimal view │ ├── PageStrcture # Components relating the main structure of the page │ │ ├── Footer.vue # Footer, visible at the bottom of all pages │ │ ├── Header.vue # Header, visible at the top of pages, and includes title and nav │ │ ├── LoadingScreen.vue # Splash screen shown on first load │ │ ├── Nav.vue # Navigation bar, includes a list of links │ │ ╰── PageTitle.vue # Page title and sub-title, visible within the Header │ ├── Workspace # Components used for the multi-tasking/ Workspace view │ │ ├── MultiTaskingWeb.vue # When multi-tasking enabled, generates new iframe │ │ ├── SideBar.vue # The left sidebar for the workspace view │ │ ├── SideBarItem.vue # App item for the sidebar view │ │ ├── SideBarSection.vue # Collapsible collection of items within workspace sidebar │ │ ├── WebContent.vue # Workspace iframe view, displays content of current app │ │ ╰── WidgetView.vue # Workspace container for displaying widgets in main content │ ├── Widgets # Directory contains all custom widget components │ │ ╰── .... # Too many to list, see widget docs instead │ ╰── Settings # Components relating to the quick-settings, in the top-right │ ├── AuthButtons.vue # Logout button and other app info │ ├── ConfigLauncher.vue # Icon that when clicked will launch the Configuration component │ ├── CustomThemeMaker.vue # Color pickers for letting user build their own theme │ ├── ItemSizeSelector.vue # Set of buttons used to set and save item size │ ├── KeyboardShortcutInfo.vue# Small pop-up displaying the available keyboard shortcuts │ ├── LanguageSwitcher.vue # Dropdown in a modal for changing app language │ ├── LayoutSelector.vue # Set of buttons, letting the user select their desired layout │ ├── SearchBar.vue # The input field in the header, used for searching the app │ ├── SettingsContainer.vue # Container that wraps all the quick-settings components │ ╰── ThemeSelector.vue # Drop-down menu enabling the user to select and change themes ├── main.js # Main front-end entry point ├── registerServiceWorker.js # Registers and manages service workers, for PWA apps ├── router.js # Defines all available application routes ├── styles # Directory of all globally used common SCSS styles │ ├── color-palette.scss # All color variable names and default values │ ├── color-themes.scss # All variable values for built-in themes │ ├── dimensions.scss # Dimensions and sizes as variables │ ├── global-styles.scss # Basics and style resets used globally │ ├── media-queries.scss # Screen sizes and media queries │ ├── style-helpers.scss # SCSS functions used for modifying values │ ├── typography.scss # Font and text styles used globally │ ╰── user-defined-themes.scss # Empty, put any custom styles or themes here ├── mixins # Reusable component bases, extended by other views / components │ ├── ChartingMixin.js # Functions for rendering charts in widget components │ ├── GlancesMixin.js # Functions for fetching system info from Glances for widgets │ ├── HomeMixin.js # Functions for homepage, used by default, minimal and workspace views │ ╰── WidgetMixin.js # Functions for all widgets, like data fetching, updating and error handling ├── utils # Directory of re-used helper functions │ ├── ArrowKeyNavigation.js # Functionality for arrow-key navigation │ ├── Auth.js # Handles all authentication related actions │ ├── CheckSectionVisibility.js # Checks which parts of the page should be visible/ hidden based on config │ ├── ClickOutside.js # A directive for detecting click, used to hide dropdown, modal or context menu │ ├── ConfigHelpers.js # Helper functions for managing configuration │ ├── CloudBackup.js # Functionality for encrypting, processing and network calls │ ├── ConfigSchema.json # The schema, used to validate the users conf.yml file │ ├── ConfigAccumulator.js # Central place for managing and combining config │ ├── ConfigHelpers.js # Collection of helper functions to process config using accumulator │ ├── ConfigValidator.js # A helper script that validates the config file against schema │ ├── CoolConsole.js # Prints info, warning and error messages to browser console, with a cool style │ ├── defaults.js # Global constants and their default values │ ├── emojis.json # List of emojis with unicode and shortcode, used for emoji icon feature │ ├── EmojiUnicodeRegex.js # Regular expression to validate emoji unicode format, for emoji icons │ ├── ErrorHandler.js # Helper function called when an error is returned │ ├── InitServiceWorker.js # Initializes and manages service worker, if enabled │ ├── Search.js # Helper functions for searching/ filtering items in all views │ ├── JsonToYaml.js # Function that parses and converts raw JSON into valid YAML │ ├── KeycloakAuth.js # Singleton class to manage Keycloak authentication │ ├── OidcAuth.js # Manages OIDC authentication with external providers │ ├── HeaderAuth.js # Handles header-based authentication via reverse proxy │ ├── languages.js # Handles fetching, switching and validating languages │ ╰── ThemeHelper.js # Function that handles the fetching and setting of user themes ╰── views # Directory of available pages, corresponding to available routes ├── Home.vue # The home page container ├── About.vue # About page ├── Login.vue # Authentication page ├── Minimal.vue # The minimal view ├── Workspace.vue # The workspace view with apps in sidebar ├── DownloadConfig.vue # Config export page ╰── 404.vue # Not found page Visualisation of Source Directory​","s":"Application Structure","u":"/docs/developing","h":"#application-structure","p":264},{"i":295,"t":"The easiest method of checking performance is to use Chromium's build in auditing tool, Lighthouse. To run the test, open Developer Tools (usually F12) --> Lighthouse and click on the 'Generate Report' button at the bottom.","s":"Performance - Lighthouse","u":"/docs/developing","h":"#performance---lighthouse","p":264},{"i":297,"t":"BundlePhobia is a really useful app that lets you analyze the cost of adding any particular dependency to an application","s":"Dependencies - BundlePhobia","u":"/docs/developing","h":"#dependencies---bundlephobia","p":264},{"i":300,"t":"When running the build command, several warnings appear. These are not errors, and do not affect the security or performance of the application. They will be addressed in a future update WARN A new version of sass-loader is available. Please upgrade for best experience. - Currently we're using an older version of SASS loader, since the more recent releases do not seem to be compatible with the Vue CLI's webpack configuration. WARN asset size limit: The following asset(s) exceed the recommended size limit (244 KiB). - For the PWA to support Windows 10, a splash screen asset is required, and is quite large. This throws a warning, however PWA assets are not loaded until needed, so shouldn't have any impact on application performance. A similar warning is thrown for the Raleway font, and that is looking to be addressed.","s":"Known Warnings","u":"/docs/developing","h":"#known-warnings","p":264},{"i":302,"t":"A series of short tutorials, to guide you through the most common development tasks. Sections: Creating a new theme Writing Translations Adding a new option in the config file Updating Dependencies Writing Netlify Cloud Functions Hiding Page Furniture Adding / Using Environmental Variables Building a Widget Respecting Config Permissions","s":"Development Guides","u":"/docs/development-guides","h":"","p":301},{"i":304,"t":"Adding a new theme is really easy. There're two things you need to do: Pass the theme name to Dashy, so that it can be added to the theme selector dropdown menu, and then write some styles!","s":"Creating a new theme","u":"/docs/development-guides","h":"#creating-a-new-theme","p":301},{"i":306,"t":"Choose a snappy name for your theme, and add it to the builtInThemes array inside defaults.js.","s":"1. Add Theme Name","u":"/docs/development-guides","h":"#1-add-theme-name","p":301},{"i":308,"t":"Put your theme styles inside color-themes.scss. Create a new block, and make sure that data-theme matches the theme name you chose above. For example: html[data-theme='tiger'] { --primary: #f58233; --background: #0b1021; } Then you can go ahead and write your own custom CSS. Although all CSS is supported here, the best way to define your theme is by setting the CSS variables. You can find a list of all CSS variables, here. For a full guide on styling, see Theming Docs. Note that if your theme is just for yourself, and you're not submitting a PR, then you can instead just pass it under appConfig.cssThemes inside your config file. And then put your theme in your own stylesheet, and pass it into the Docker container - see how.","s":"2. Write some Styles","u":"/docs/development-guides","h":"#2-write-some-styles","p":301},{"i":310,"t":"For full docs about Dashy's multi-language support, see Multi-Language Support Dashy is using vue-i18n to manage multi-language support. Adding a new language is pretty straightforward, with just three steps:","s":"Writing Translations","u":"/docs/development-guides","h":"#writing-translations","p":301},{"i":312,"t":"Create a new JSON file in ./src/assets/locales name is a 2-digit ISO-639 code for your language, E.g. for German de.json, French fr.json or Spanish es.json - You can find a list of all ISO codes at iso.org.","s":"1. Create a new Language File","u":"/docs/development-guides","h":"#1-create-a-new-language-file","p":301},{"i":314,"t":"Using en.json as an example, translate the JSON values to your language, while leaving the keys as they are. It's fine to leave out certain items, as if they're missing they will fall-back to English. If you see any attribute which include curly braces ({xxx}), then leave the inner value of these braces as is, as this is for variables. { \"theme-maker\": { \"export-button\": \"Benutzerdefinierte Variablen exportieren\", \"reset-button\": \"Stile zurücksetzen für\", \"show-all-button\": \"Alle Variablen anzeigen\", \"save-button\": \"Speichern\", \"cancel-button\": \"Abbrechen\", \"saved-toast\": \"{theme} Erfolgreich aktualisiert\", \"reset-toast\": \"Benutzerdefinierte Farben für {theme} entfernt\" }, }","s":"2. Translate","u":"/docs/development-guides","h":"#2-translate","p":301},{"i":316,"t":"In ./src/utils/languages.js, you need to do 2 small things: First import your new translation file, do this at the top of the page. E.g. import de from '@/assets/locales/de.json'; Second, add it to the array of languages, e.g: export const languages = [ { name: 'English', code: 'en', locale: en, flag: '🇬🇧', }, { name: 'German', // The name of your language code: 'de', // The ISO code of your language locale: de, // The name of the file you imported (no quotes) flag: '🇩🇪', // An optional flag emoji }, ]; You can also add your new language to the readme file, under the Language Switching section, and optionally include your name/ username if you'd like to be credited for your work. Done! If you are not comfortable with making pull requests, or do not want to modify the code, then feel free to instead send the translated file to me, and I can add it into the application. I will be sure to credit you appropriately.","s":"3. Add your file to the app","u":"/docs/development-guides","h":"#3-add-your-file-to-the-app","p":301},{"i":318,"t":"This section is for, adding a new setting to the config file. All of the users config is specified in ./user-data/conf.yml - see Configuring Docs for info. It's important to first ensure that there isn't a similar option already available, the new option is definitely necessary, and most importantly that it is fully backwards compatible. Next choose the appropriate section to place it under Application settings should be located under appConfig Page info (such as text and metadata) should be under pageInfo Data relating to specific sections should be under section[n].displayData Settings applied to specific items or widgets, should be under item[n] or widget[n] For example, if your option is added under appConfig, you can access it within your component using the $store, this is typically placed in a computed property, e.g: computed: { appConfig() { return this.$store.getters.appConfig; }, ... }, Then, where you want to get the users value within your component, use something like: this.appConfig.myProperty. If the user hasn't specified the value, Don't forget to have a fallback or default for it. If you have a default fallback value, then this would typically be specified in the defaults.js file. You will now need to add the definition of your new attribute into the ConfigSchema. This will make it available in the UI config editor, and also ensure that the config validation check doesn't fail. For example: \"fontAwesomeKey\": { \"type\": \"string\", \"pattern\": \"^[a-z0-9]{10}$\", \"description\": \"API key for font-awesome\", \"example\": \"0821c65656\" } or \"iconSize\": { \"enum\": [ \"small\", \"medium\", \"large\" ], \"default\": \"medium\", \"description\": \"The size of each link item / icon\" } Finally, add your new property to the configuring.md API docs. Put it under the relevant section, and be sure to include field name, data type, a description and mention that it is optional. If your new feature needs more explanation, then you can also document it under the relevant section elsewhere in the documentation. Checklist: Ensure the new attribute is actually necessary, and nothing similar already exists Update the Schema with the parameters for your new option If required, set a default or fallback value (usually in defaults.js) Document the new value in configuring.md, and if required under the relevant section in the docs Ensure your changes are backwards compatible, and that nothing breaks if the attribute isn't specified","s":"Adding a new option in the config file","u":"/docs/development-guides","h":"#adding-a-new-option-in-the-config-file","p":301},{"i":320,"t":"Running yarn upgrade will updated all dependencies based on the ranges specified in the package.json. The yarn.lock file will be updated, as will the contents of ./node_modules, for more info, see the yarn upgrade documentation. npm-check-updates is a useful tool to help with this. It is important to thoroughly test after any big dependency updates.","s":"Updating Dependencies","u":"/docs/development-guides","h":"#updating-dependencies","p":301},{"i":322,"t":"When Dashy is deployed to Netlify, it is effectively running as a static app, and therefore the server-side code for the Node.js endpoints is not available. However Netlify now supports serverless cloud lambda functions, which can be used to replace most functionality.","s":"Developing Netlify Cloud Functions","u":"/docs/development-guides","h":"#developing-netlify-cloud-functions","p":301},{"i":324,"t":"First off all, install the Netlify CLI: npm install netlify-cli -g Then, from within the root of Dashy's directory, start the server, by running: netlify dev","s":"1. Run Netlify Dev Server","u":"/docs/development-guides","h":"#1-run-netlify-dev-server","p":301},{"i":326,"t":"This should be saved in the ./services/serverless-functions directory exports.handler = async () => ({ statusCode: 200, body: 'Return some data here...', });","s":"2. Create a lambda function","u":"/docs/development-guides","h":"#2-create-a-lambda-function","p":301},{"i":328,"t":"In the netlify.toml file, add a 301 redirect, with the path to the original Node.js endpoint, and the name of your cloud function [[redirects]] from = \"/status-check\" to = \"/.netlify/functions/cloud-status-check\" status = 301 force = true","s":"3. Redirect the Node endpoint to the function","u":"/docs/development-guides","h":"#3-redirect-the-node-endpoint-to-the-function","p":301},{"i":330,"t":"For some pages (such as the login page, the minimal start page, etc) the basic page furniture, (like header, footer, nav, etc) is not needed. This section explains how you can hide furniture on a new view (step 1), or add a component that should be hidden on certain views (step 2).","s":"Hiding Page Furniture on Certain Routes","u":"/docs/development-guides","h":"#hiding-page-furniture-on-certain-routes","p":301},{"i":332,"t":"In ./src/utils/config/defaults.js, there's an array called hideFurnitureOn. Append the name of the route (the same as it appears in router.js) here.","s":"1. Add the route name to the should hide array","u":"/docs/development-guides","h":"#1-add-the-route-name-to-the-should-hide-array","p":301},{"i":334,"t":"First, import the helper function: import { shouldBeVisible } from '@/utils/config/SectionHelpers'; Then you can create a computed value, that calls this function, passing in the route name: export default { ... computed: { ... isVisible() { return shouldBeVisible(this.$route.name); }, }, }; Finally, in the markup of your component, just add a v-if statement, referencing your computed value
    ...
    ","s":"2. Add the conditional to the structural component to hide","u":"/docs/development-guides","h":"#2-add-the-conditional-to-the-structural-component-to-hide","p":301},{"i":336,"t":"All environmental variables are optional. Currently there are not many environmental variables used, as most of the user preferences are stored under appConfig in the conf.yml file. You can set variables either in your environment, or using the .env file. Any environmental variables used by the frontend are preceded with VITE_APP_. Vite will merge the contents of your .env file into the app in a similar way to the 'dotenv' package, where any variables that you set on your system will always take preference over the contents of any .env file. If add any new variables, ensure that there is always a fallback (define it in defaults.js), so as to not cause breaking changes. Don't commit the contents of your .env file to git, but instead take a few moments to document what you've added under the appropriate section. Try and follow the concepts outlined in the 12 factor app.","s":"Adding / Using Environmental Variables","u":"/docs/development-guides","h":"#adding--using-environmental-variables","p":301},{"i":339,"t":"If this is your first time working on Dashy, then the Developing Docs instructions for project setup and running. In short, you just need to clone the project, cd into it, install dependencies (yarn) and then start the development server (yarn dev). To build a widget, you'll also need some basic knowledge of Vue.js. The official Vue docs provides a good starting point, as does this guide by Tania Rascia If you just want to jump straight in, then here is a complete implementation of a new example widget, or take a look at the XkcdComic.vue widget, which is pretty simple.","s":"Step 0 - Prerequisites","u":"/docs/development-guides","h":"#step-0---prerequisites","p":301},{"i":341,"t":"Firstly, create a new .vue file under ./src/components/Widgets. All widgets extend from the Widget mixin. This provides some basic functionality that is shared by all widgets. The mixin includes the following options, startLoading(), finishLoading(), error() and update(). Getting user options: options Any user-specific config can be accessed with this.options.something (where something is the data key you're accessing) Loading state: startLoading() and finishLoading() You can show the loader with this.startLoading(), then when your data request completes, hide it again with this.finishLoading() Error handling: error() If something goes wrong (such as API error, or missing user parameters), then call this.error() to show message to user Updating data: update() When the user clicks the update button, or if continuous updates are enabled, then the update() method within your widget will be called","s":"Step 1 - Create Widget","u":"/docs/development-guides","h":"#step-1---create-widget","p":301},{"i":343,"t":"Accessing User Options​ If your widget is going to accept any parameters from the user, then we can access these with this.options.[parmName]. It's best to put these as computed properties, which will enable us to check it exists, is valid, and if needed format it. For example, if we have an optional property called count (to determine number of results), we can do the following, and then reference it within our component with this.count computed: { count() { if (!this.options.count) { return 5; } return this.options.count; }, ... }, Adding an API Endpoint​ If your widget makes a data request, then add the URL for the API endpoint to the widgetApiEndpoints array in defaults.js widgetApiEndpoints: { ... exampleEndpoint: 'https://hub.dummyapis.com/ImagesList', }, Then in your widget file: import { widgetApiEndpoints } from '@/utils/config/defaults'; For GET requests, you may need to add some parameters onto the end of the URL. We can use another computed property for this, for example: endpoint() { return `${widgetApiEndpoints.exampleEndpoint}?count=${this.count}`; }, Making an API Request​ Axios is used for making data requests, so import it into your component: import axios from 'axios'; Under the methods block, we'll create a function called fetchData, here we can use Axios to make a call to our endpoint. fetchData() { this.makeRequest(this.endpoint, this.headers).then(this.processData); }, There are three things happening here: If the response completes successfully, we'll pass the results to another function that will handle them If there's an error, then we call this.error(), which will show a message to the user Whatever the result, once the request has completed, we call this.finishLoading(), which will hide the loader Processing Response​ In the above example, we call the processData() method with the result from the API, so we need to create that under the methods section. How you handle this data will vary depending on what's returned by the API, and what you want to render to the user. But however you do it, you will likely need to create a data variable to store the response, so that it can be easily displayed in the HTML. data() { return { myResults: null, }; }, And then, inside your processData() method, you can set the value of this, with: `this.myResults = 'whatever'` Rendering Response​ Now that the results are in the correct format, and stored as data variables, we can use them within the