From f4abe8afa19262e22924b8cfdd0caf2ed3c22006 Mon Sep 17 00:00:00 2001 From: Lissy93 Date: Sun, 28 Apr 2024 23:29:38 +0000 Subject: [PATCH] Deploy website - based on 0022940c653f8bc3be167ce63de21e61c19b9ab6 --- 404.html | 4 ++-- assets/js/18ba09e8.a254f0fc.js | 1 + assets/js/18ba09e8.b7906927.js | 1 - assets/js/625b1ed7.39397709.js | 1 - assets/js/625b1ed7.b95d042d.js | 1 + assets/js/72e14192.4ebcc2d8.js | 1 + assets/js/72e14192.ba227462.js | 1 - ...ntime~main.83b3f325.js => runtime~main.9dc14f8e.js} | 2 +- docs/alternate-views/index.html | 4 ++-- docs/authentication/index.html | 10 +++++----- docs/backup-restore/index.html | 4 ++-- docs/changelog/index.html | 4 ++-- docs/code-of-conduct/index.html | 4 ++-- docs/configuring/index.html | 4 ++-- docs/contributing/index.html | 4 ++-- docs/credits/index.html | 6 +++--- docs/deployment/index.html | 4 ++-- docs/developing/index.html | 4 ++-- docs/development-guides/index.html | 4 ++-- docs/icons/index.html | 4 ++-- docs/index.html | 4 ++-- docs/license/index.html | 4 ++-- docs/management/index.html | 4 ++-- docs/multi-language-support/index.html | 4 ++-- docs/pages-and-sections/index.html | 4 ++-- docs/privacy/index.html | 4 ++-- docs/quick-start/index.html | 9 +++++---- docs/readme/index.html | 4 ++-- docs/release-workflow/index.html | 4 ++-- docs/searching/index.html | 4 ++-- docs/showcase/index.html | 4 ++-- docs/showcase/readme/index.html | 4 ++-- docs/status-indicators/index.html | 4 ++-- docs/theming/index.html | 4 ++-- docs/troubleshooting/index.html | 4 ++-- docs/widgets/index.html | 4 ++-- index.html | 4 ++-- markdown-page/index.html | 4 ++-- search/index.html | 4 ++-- survey/index.html | 4 ++-- 40 files changed, 77 insertions(+), 76 deletions(-) create mode 100644 assets/js/18ba09e8.a254f0fc.js delete mode 100644 assets/js/18ba09e8.b7906927.js delete mode 100644 assets/js/625b1ed7.39397709.js create mode 100644 assets/js/625b1ed7.b95d042d.js create mode 100644 assets/js/72e14192.4ebcc2d8.js delete mode 100644 assets/js/72e14192.ba227462.js rename assets/js/{runtime~main.83b3f325.js => runtime~main.9dc14f8e.js} (96%) diff --git a/404.html b/404.html index 3e6042b5..da0a4665 100644 --- a/404.html +++ b/404.html @@ -6,13 +6,13 @@ Page Not Found | Dashy - +
Skip to main content

Page Not Found

We could not find what you were looking for.

Please contact the owner of the site that linked you to the original URL and let them know their link is broken.

- + \ No newline at end of file diff --git a/assets/js/18ba09e8.a254f0fc.js b/assets/js/18ba09e8.a254f0fc.js new file mode 100644 index 00000000..800b9c1c --- /dev/null +++ b/assets/js/18ba09e8.a254f0fc.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[849],{5680:(t,e,a)=>{a.d(e,{xA:()=>y,yg:()=>b});var n=a(6540);function l(t,e,a){return e in t?Object.defineProperty(t,e,{value:a,enumerable:!0,configurable:!0,writable:!0}):t[e]=a,t}function g(t,e){var a=Object.keys(t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(t);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),a.push.apply(a,n)}return a}function r(t){for(var e=1;e=0||(l[a]=t[a]);return l}(t,e);if(Object.getOwnPropertySymbols){var g=Object.getOwnPropertySymbols(t);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(t,a)&&(l[a]=t[a])}return l}var i=n.createContext({}),s=function(t){var e=n.useContext(i),a=e;return t&&(a="function"==typeof t?t(e):r(r({},e),t)),a},y=function(t){var e=s(t.components);return n.createElement(i.Provider,{value:e},t.children)},h="mdxType",o={inlineCode:"code",wrapper:function(t){var e=t.children;return n.createElement(n.Fragment,{},e)}},c=n.forwardRef((function(t,e){var a=t.components,l=t.mdxType,g=t.originalType,i=t.parentName,y=u(t,["components","mdxType","originalType","parentName"]),h=s(a),c=l,b=h["".concat(i,".").concat(c)]||h[c]||o[c]||g;return a?n.createElement(b,r(r({ref:e},y),{},{components:a})):n.createElement(b,r({ref:e},y))}));function b(t,e){var a=arguments,l=e&&e.mdxType;if("string"==typeof t||l){var g=a.length,r=new Array(g);r[0]=c;var u={};for(var i in e)hasOwnProperty.call(e,i)&&(u[i]=e[i]);u.originalType=t,u[h]="string"==typeof t?t:l,r[1]=u;for(var s=2;s{a.r(e),a.d(e,{contentTitle:()=>r,default:()=>h,frontMatter:()=>g,metadata:()=>u,toc:()=>i});var n=a(8168),l=(a(6540),a(5680));const g={},r="Credits",u={unversionedId:"credits",id:"credits",isDocsHomePage:!1,title:"Credits",description:"Sponsors",source:"@site/docs/credits.md",sourceDirName:".",slug:"/credits",permalink:"/docs/credits",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/credits.md",version:"current",frontMatter:{}},i=[{value:"Sponsors",id:"sponsors",children:[]},{value:"Contributors",id:"contributors",children:[]},{value:"Helpful Users",id:"helpful-users",children:[]},{value:"Bots",id:"bots",children:[]},{value:"Newest Stargazers",id:"newest-stargazers",children:[]},{value:"Dependencies",id:"dependencies",children:[{value:"Core",id:"core",children:[]},{value:"Utilities",id:"utilities",children:[]},{value:"Frontend Components",id:"frontend-components",children:[]},{value:"Backup & Sync Server",id:"backup--sync-server",children:[]},{value:"External Services",id:"external-services",children:[]},{value:"Actions",id:"actions",children:[]}]},{value:"You",id:"you",children:[]}],s={toc:i},y="wrapper";function h(t){let{components:e,...a}=t;return(0,l.yg)(y,(0,n.A)({},s,a,{components:e,mdxType:"MDXLayout"}),(0,l.yg)("h1",{id:"credits"},"Credits"),(0,l.yg)("h2",{id:"sponsors"},"Sponsors"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/github"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9919?v=4",width:"80;",alt:"github"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"GitHub")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/koconder"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25068?u=582657b23622aaa3dfe68bd028a780f272f456fa&v=4",width:"80;",alt:"koconder"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Vincent Koc")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Admonstrator"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/69824?u=1e226d7a36cdd661c3e4cd486fea140d045b7d57&v=4",width:"80;",alt:"Admonstrator"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aaron Viehl")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/tbjers"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1117052?v=4",width:"80;",alt:"tbjers"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Torgny Bjers")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/emlazzarin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1141361?u=714e3487a3f2e0df721b01a0133945f075d3ff68&v=4",width:"80;",alt:"emlazzarin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Eddy Lazzarin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AnandChowdhary"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2841780?u=747e554b3a7f12eb20b7910e1c87d817844f714f&v=4",width:"80;",alt:"AnandChowdhary"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Anand Chowdhary"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shrippen"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2873570?v=4",width:"80;",alt:"shrippen"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bile0026"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5022496?u=aec96ad173c0ea9baaba93807efa8a848af6595c&v=4",width:"80;",alt:"bile0026"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Zach Biles")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/UlisesGascon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5110813?u=3c41facd8aa26154b9451de237c34b0f78d672a5&v=4",width:"80;",alt:"UlisesGascon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ulises Gasc\xf3n")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/digitalarche"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6546135?u=564756d7f44ab2206819eb3148f6d822673f5066&v=4",width:"80;",alt:"digitalarche"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Digital Archeology")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/InDieTasten"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7047377?u=8d8f8017628b38bc46dcbf3620e194b01d3fb2d1&v=4",width:"80;",alt:"InDieTasten"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/araguaci"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7318668?v=4",width:"80;",alt:"araguaci"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bmcgonag"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7346620?u=2a0f9284f3e12ac1cc15288c254d1ec68a5081e8&v=4",width:"80;",alt:"bmcgonag"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brian McGonagill")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/vlad-timofeev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11474041?u=eee43705b54d2ec9f51fc4fcce5ad18dd17c87e4&v=4",width:"80;",alt:"vlad-timofeev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Vlad Timofeev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/helixzz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12218889?u=d06d0c103dfbdb99450623064f7da3c5a3675fb6&v=4",width:"80;",alt:"helixzz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"HeliXZz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/patvdv"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12430107?u=e8911c2fb91af4d30432f76da8c40927b2830bd7&v=4",width:"80;",alt:"patvdv"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Patrick Van Der Veken")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/plgonzalezrx8"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19900049?v=4",width:"80;",alt:"plgonzalezrx8"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Pedro Gonzalez")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/mryesiller"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24632172?u=0d20f2d615158f87cd60a3398d3efb026c32f291&v=4",width:"80;",alt:"mryesiller"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"G\xf6ksel Ye\u015filler"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/allesauseinerhand"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32039836?v=4",width:"80;",alt:"allesauseinerhand"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/forwardemail"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32481436?v=4",width:"80;",alt:"forwardemail"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Forward Email - Open-source & Privacy-focused Email Service (2023)")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lamtrinhdev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/49742151?u=c5eaca5aa6841a80605cf4f7d0e861a9e6339ef3&v=4",width:"80;",alt:"lamtrinhdev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"LamTrinh.Dev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Bastii717"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/53431819?u=604977bed6ad6875ada890d0d3765a4cacc2fa14&v=4",width:"80;",alt:"Bastii717"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/getumbrel"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/59408891?v=4",width:"80;",alt:"getumbrel"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Umbrel")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/M2TD"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/85460457?v=4",width:"80;",alt:"M2TD"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/frankdez93"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87549420?v=4",width:"80;",alt:"frankdez93"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/terminaltrove"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/121595180?v=4",width:"80;",alt:"terminaltrove"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Terminal Trove")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/NixyJuppie"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/138570196?u=b102c222487905728b858704962d32759df29ebe&v=4",width:"80;",alt:"NixyJuppie"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nixy")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/nrvo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/151435968?u=e1dcb307fd0efdc45cddbe9490a7b956e4da6835&v=4",width:"80;",alt:"nrvo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))))),(0,l.yg)("h2",{id:"contributors"},"Contributors"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Lissy93"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1862727?v=4",width:"80;",alt:"Lissy93"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Sykes")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/liss-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87835202?v=4",width:"80;",alt:"liss-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/marekful"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10281476?v=4",width:"80;",alt:"marekful"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Marcell F\xfcl\xf6p")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/EVOTk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45015615?v=4",width:"80;",alt:"EVOTk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"EVOTk")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/snyk-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19733683?v=4",width:"80;",alt:"snyk-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Snyk Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/azerioxal"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5369885?v=4",width:"80;",alt:"azerioxal"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kenneth Church"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/m42e"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2410802?v=4",width:"80;",alt:"m42e"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Matthias Bilger")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/CrazyWolf13"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/96661824?v=4",width:"80;",alt:"CrazyWolf13"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Tobias")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/pinarruiz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37040888?v=4",width:"80;",alt:"pinarruiz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alejandro Pinar Ruiz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imjimmeh"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2104997?v=4",width:"80;",alt:"imjimmeh"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kashif-se"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5568138?v=4",width:"80;",alt:"kashif-se"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kashif Sohail")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/walkxcode"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/71191962?v=4",width:"80;",alt:"walkxcode"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Walkx"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/josuablejeru"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37913833?v=4",width:"80;",alt:"josuablejeru"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Josua Blejeru ")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/evroon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11857441?v=4",width:"80;",alt:"evroon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Erik Vroon")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jammo2k5"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/8106602?v=4",width:"80;",alt:"jammo2k5"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jammo2k5")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lordpansar"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/14231148?v=4",width:"80;",alt:"lordpansar"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Magnus Sundstr\xf6m")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Cereal916"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7526937?v=4",width:"80;",alt:"Cereal916"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kristian Brasel")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/albcp"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3170731?v=4",width:"80;",alt:"albcp"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alberto"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/altearius"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/270430?v=4",width:"80;",alt:"altearius"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Chris Nielsen")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/UrekD"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/38784343?v=4",width:"80;",alt:"UrekD"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"UrekD")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/z3r0l1nk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/31653632?v=4",width:"80;",alt:"z3r0l1nk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mihai")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/wozboz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/51856582?v=4",width:"80;",alt:"wozboz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Totto16"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32566573?v=4",width:"80;",alt:"Totto16"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Totto16")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/toddejohnson"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/507545?v=4",width:"80;",alt:"toddejohnson"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Todd Johnson"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/remygrandin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1934515?v=4",width:"80;",alt:"remygrandin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Remygrandin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/DimitriDR"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56969769?v=4",width:"80;",alt:"DimitriDR"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dimitri")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Tracreed"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6306365?v=4",width:"80;",alt:"Tracreed"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"David Alasow")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/aviolaris"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/48277853?v=4",width:"80;",alt:"aviolaris"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Andreas Violaris")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rubenandre"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9402773?v=4",width:"80;",alt:"rubenandre"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"R\xfaben Silva")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rtm516"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5401186?v=4",width:"80;",alt:"rtm516"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Rtm516"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/zcq100"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/425234?v=4",width:"80;",alt:"zcq100"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/onedr0p"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/213795?v=4",width:"80;",alt:"onedr0p"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\u15ea\u0454\u03bd\u03b9\u03b7 \u15f7\u03c5\u043d\u029f")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/stanly0726"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37040069?v=4",width:"80;",alt:"stanly0726"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stanly0726")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Bogyie"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/82003678?v=4",width:"80;",alt:"Bogyie"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Bogyeong Kim")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Tuzi555"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62188066?v=4",width:"80;",alt:"Tuzi555"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jakub Tuzar")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/berksmbl"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10000339?v=4",width:"80;",alt:"berksmbl"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Berk S\xfcmb\xfcl"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alucarddelta"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/20882097?v=4",width:"80;",alt:"alucarddelta"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brent")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/k073l"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/21180271?v=4",width:"80;",alt:"k073l"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kt-alt"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/51970249?v=4",width:"80;",alt:"kt-alt"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/mmihaly"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/50031464?v=4",width:"80;",alt:"mmihaly"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/patrickheeney"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1407228?v=4",width:"80;",alt:"patrickheeney"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Patrick Heeney")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rokiden"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11071229?v=4",width:"80;",alt:"rokiden"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Denis Kazimirov"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/zigotica"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/178855?v=4",width:"80;",alt:"zigotica"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sergi Meseguer")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BySempron"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/15928132?v=4",width:"80;",alt:"BySempron"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sergio")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ssrangisetti"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/46807508?v=4",width:"80;",alt:"ssrangisetti"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/itsmejoeeey"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9375730?v=4",width:"80;",alt:"itsmejoeeey"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Joey Miller")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/hockeymikey"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1435185?v=4",width:"80;",alt:"hockeymikey"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/thomaswienecke"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11446531?v=4",width:"80;",alt:"thomaswienecke"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Thomas Wienecke"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/deneor"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1063265?v=4",width:"80;",alt:"deneor"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/daentech"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/358678?v=4",width:"80;",alt:"daentech"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dan Gilbert")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/moemoeq"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1808434?v=4",width:"80;",alt:"moemoeq"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"CHAIYEON CHO")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BOZG"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6022344?v=4",width:"80;",alt:"BOZG"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stephen Rigney")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/a-mnich"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56564725?v=4",width:"80;",alt:"a-mnich"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alexander Mnich")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alexdelprete"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7027842?v=4",width:"80;",alt:"alexdelprete"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alessandro Del Prete"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alayham"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/518776?v=4",width:"80;",alt:"alayham"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Al-Ayham Saleh")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/sachahjkl"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32895534?v=4",width:"80;",alt:"sachahjkl"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sacha")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shazzx"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/131521332?v=4",width:"80;",alt:"shazzx"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Shazz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ThinkSalat"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/31082405?v=4",width:"80;",alt:"ThinkSalat"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Shawn Salat")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/royshreyaa"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/139828242?v=4",width:"80;",alt:"royshreyaa"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Smexhy"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/4880625?v=4",width:"80;",alt:"Smexhy"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/stavros-k"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/47820033?v=4",width:"80;",alt:"stavros-k"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stavros Kois")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/XenonR"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/18627623?v=4",width:"80;",alt:"XenonR"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Steffen Schmidt")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/StevKast"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/17804308?v=4",width:"80;",alt:"StevKast"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Steven Kast")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AmadeusGraves"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/18572939?v=4",width:"80;",alt:"AmadeusGraves"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\xc1ngel Fern\xe1ndez S\xe1nchez")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/turnrye"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/701035?v=4",width:"80;",alt:"turnrye"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ryan Turner")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rubjo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/42270947?v=4",width:"80;",alt:"rubjo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/PrynsTag"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56314705?v=4",width:"80;",alt:"PrynsTag"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Prince Carl Velasco")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/PlusaN"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/61884717?v=4",width:"80;",alt:"PlusaN"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Andrey")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/oka4shi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/67847553?v=4",width:"80;",alt:"oka4shi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"OKAMOTO Shigehiro")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bubylou"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3878640?v=4",width:"80;",alt:"bubylou"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nicholas Malcolm")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/miclav"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11891522?v=4",width:"80;",alt:"miclav"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Michael Lavaire")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imsakg"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62212589?v=4",width:"80;",alt:"imsakg"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mert Sefa AKGUN"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ethan-hann"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/36464732?v=4",width:"80;",alt:"ethan-hann"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ethan Hann")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/tazboyz16"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12215340?v=4",width:"80;",alt:"tazboyz16"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/sur1v"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19678230?v=4",width:"80;",alt:"sur1v"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jos\xe9 Ignacio")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/soaibsafi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11424812?v=4",width:"80;",alt:"soaibsafi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Soaibuzzaman")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/pablomalo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25877142?v=4",width:"80;",alt:"pablomalo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/markxoe"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/59475466?v=4",width:"80;",alt:"markxoe"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mark Oude Elberink"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lxjv"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/63261955?v=4",width:"80;",alt:"lxjv"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Laker Turner")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kxenoxx"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9744900?v=4",width:"80;",alt:"kxenoxx"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kxenox")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jrobles98"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/30221842?v=4",width:"80;",alt:"jrobles98"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jnach"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/33467747?v=4",width:"80;",alt:"jnach"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jnach")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imlonghao"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/4951333?v=4",width:"80;",alt:"imlonghao"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Imlonghao")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/icy-comet"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/50461557?v=4",width:"80;",alt:"icy-comet"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aniket Teredesai"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/huangshaohuai"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24775925?v=4",width:"80;",alt:"huangshaohuai"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/flechaig"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10887132?v=4",width:"80;",alt:"flechaig"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dr460nf1r3"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12834713?v=4",width:"80;",alt:"dr460nf1r3"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nico")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/baifengheixi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/98794233?v=4",width:"80;",alt:"baifengheixi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/allozavrr"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/63748214?v=4",width:"80;",alt:"allozavrr"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/XertDev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/16572811?v=4",width:"80;",alt:"XertDev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Xert"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/emiran-orange"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/71817149?v=4",width:"80;",alt:"emiran-orange"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/edugof"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19559978?v=4",width:"80;",alt:"edugof"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Eduardo Gomez")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Dylan-Bs"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/35694107?v=4",width:"80;",alt:"Dylan-Bs"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dylan Bersans")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dougaldhub"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25713235?v=4",width:"80;",alt:"dougaldhub"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Doug Lock")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dkyeremeh"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/52928432?v=4",width:"80;",alt:"dkyeremeh"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Desmond Kyeremeh")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/deepsourcebot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/60907429?v=4",width:"80;",alt:"deepsourcebot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"DeepSource Bot"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/DawidPietrykowski"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/53954695?v=4",width:"80;",alt:"DawidPietrykowski"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/skaarj1989"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/34756939?v=4",width:"80;",alt:"skaarj1989"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"David")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/clsty"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/129247596?v=4",width:"80;",alt:"clsty"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Celestial.y")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bskim45"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5674934?v=4",width:"80;",alt:"bskim45"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Bumsoo Kim")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BhasherBEL"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45831883?v=4",width:"80;",alt:"BhasherBEL"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brieuc Dubois")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BeginCI"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/57495754?v=4",width:"80;",alt:"BeginCI"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Begin"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BRAVO68WEB"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/41448663?v=4",width:"80;",alt:"BRAVO68WEB"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jyotirmoy Bandyopadhyaya [Bravo68]")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AaronPorts"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32810520?v=4",width:"80;",alt:"AaronPorts"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Artyom")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alydemah"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/652035?v=4",width:"80;",alt:"alydemah"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aly Mohamed")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/5idereal"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/30827929?v=4",width:"80;",alt:"5idereal"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"5idereal")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/0n1cOn3"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/27576311?v=4",width:"80;",alt:"0n1cOn3"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"0n1cOn3")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/maximemoreillon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29086128?v=4",width:"80;",alt:"maximemoreillon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Maxime Moreillon"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Glitch3dPenguin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3271160?v=4",width:"80;",alt:"Glitch3dPenguin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Max Kulik")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/markusdd"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25175069?v=4",width:"80;",alt:"markusdd"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Markus Krause")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/asenov"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/280619?v=4",width:"80;",alt:"asenov"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\u041c\u0438\u0440\u043e\u0441\u043b\u0430\u0432 \u0410\u0441\u0435\u043d\u043e\u0432")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/luispabon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6388823?v=4",width:"80;",alt:"luispabon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Luis Pabon")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/LeoColman"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1577251?v=4",width:"80;",alt:"LeoColman"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Leonardo Colman Lopes")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/KierenConnell"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/46445781?v=4",width:"80;",alt:"KierenConnell"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kieren Connell"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kieraneglin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/569917?v=4",width:"80;",alt:"kieraneglin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kieran")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Singebob"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24290044?v=4",width:"80;",alt:"Singebob"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jeremy Chauvin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Hellhium"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11504877?v=4",width:"80;",alt:"Hellhium"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jemy SCHNEPP")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jjmung"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6049600?v=4",width:"80;",alt:"jjmung"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"JJ Munguia")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/b1thunt3r"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/791091?v=4",width:"80;",alt:"b1thunt3r"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ishan Jain")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Compunctus"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5058853?v=4",width:"80;",alt:"Compunctus"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Iaroslav Dronskii"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/nealian"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/865115?v=4",width:"80;",alt:"nealian"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ian Neal")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/FraglyG"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56320839?v=4",width:"80;",alt:"FraglyG"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Hendrik Strydom")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/hubortje"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62364169?v=4",width:"80;",alt:"hubortje"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Harald T\xf6pfer")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/gbrown09"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3360055?v=4",width:"80;",alt:"gbrown09"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Garrett Brown")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/FormatToday"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/20515769?v=4",width:"80;",alt:"FormatToday"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"FormatToday")))))),(0,l.yg)("h2",{id:"helpful-users"},"Helpful Users"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/evotk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45015615?v=4",width:"80;",alt:"evotk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Evotk")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shadowking001"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/43928955?v=4",width:"80;",alt:"shadowking001"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"LawrenceP.")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/urekd"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/38784343?v=4",width:"80;",alt:"urekd"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Urekd")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dylanbeme"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/41838333?v=4",width:"80;",alt:"dylanbeme"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"DylanH")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/milesteg1"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29298312?v=4",width:"80;",alt:"milesteg1"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Milesteg1")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lu4t"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29571529?v=4",width:"80;",alt:"lu4t"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Lu4t")))))),(0,l.yg)("h2",{id:"bots"},"Bots"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/liss-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87835202?v=4",width:"80;",alt:"liss-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/snyk-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19733683?v=4",width:"80;",alt:"snyk-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Snyk Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/netlify"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7892489?v=4",width:"80;",alt:"netlify"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Netlify")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/viezly"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/78649443?v=4",width:"80;",alt:"viezly"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Viezly")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/muse-dev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/42072939?v=4",width:"80;",alt:"muse-dev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"MuseDev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/github-actions[bot]"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/in/15368?v=4",width:"80;",alt:"github-actions[bot]"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"github-actions[bot]"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dependabot[bot]"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/in/29110?v=4",width:"80;",alt:"dependabot[bot]"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"dependabot[bot]")))))),(0,l.yg)("blockquote",null,(0,l.yg)("p",{parentName:"blockquote"},"The above section is auto-generated, using ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/marketplace/actions/contribute-list"},"contribute-list")," by @akhilmhdh.")),(0,l.yg)("h2",{id:"newest-stargazers"},"Newest Stargazers"),(0,l.yg)("p",null,(0,l.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/stargazers"},(0,l.yg)("img",{parentName:"a",src:"https://reporoster.com/stars/dark/Lissy93/dashy",alt:"Recent Star Gazers"}))),(0,l.yg)("p",null,(0,l.yg)("a",{parentName:"p",href:"https://github.com/lissy93/dashy/network/members"},(0,l.yg)("img",{parentName:"a",src:"https://reporoster.com/forks/dark/lissy93/dashy",alt:"Recent Forkers"}))),(0,l.yg)("hr",null),(0,l.yg)("h2",{id:"dependencies"},"Dependencies"),(0,l.yg)("p",null,"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. For a full breakdown of dependency licenses, please see ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/.github/LEGAL.md"},"Legal")),(0,l.yg)("h3",{id:"core"},"Core"),(0,l.yg)("p",null,"At it's core, the application uses ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/vuejs/vue"},(0,l.yg)("strong",{parentName:"a"},"Vue.js")),", as well as it's services with ",(0,l.yg)("a",{parentName:"p",href:"https://vuex.vuejs.org/"},(0,l.yg)("strong",{parentName:"a"},"VueX"))," for state management. Styling is done with ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/sass/sass"},(0,l.yg)("strong",{parentName:"a"},"SCSS")),", JavaScript is currently ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/babel/babel"},(0,l.yg)("strong",{parentName:"a"},"Babel")),", (but I am in the process of converting to ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/Microsoft/TypeScript"},(0,l.yg)("strong",{parentName:"a"},"TypeScript")),"). Linting is done with ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/eslint/eslint"},(0,l.yg)("strong",{parentName:"a"},"ESLint"))," and ",(0,l.yg)("a",{parentName:"p",href:"https://prettier.io/"},(0,l.yg)("strong",{parentName:"a"},"Prettier")),", both following the ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/airbnb/javascript"},(0,l.yg)("strong",{parentName:"a"},"AirBnB Styleguide")),". The config is defined in ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/yaml/yaml"},(0,l.yg)("strong",{parentName:"a"},"YAML")),", with a simple ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/nodejs/node"},(0,l.yg)("strong",{parentName:"a"},"Node.js"))," server to serve up the static app and the optional API endpoints, and container deployment is done with ",(0,l.yg)("a",{parentName:"p",href:"https://www.docker.com/"},(0,l.yg)("strong",{parentName:"a"},"Docker")),"."),(0,l.yg)("h3",{id:"utilities"},"Utilities"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/brix/crypto-js"},(0,l.yg)("inlineCode",{parentName:"a"},"crypto-js"))," - Encryption implementations by @evanvosberg and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/axios/axios"},(0,l.yg)("inlineCode",{parentName:"a"},"axios"))," - Promise based HTTP client by @mzabriskie and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/ajv-validator/ajv"},(0,l.yg)("inlineCode",{parentName:"a"},"ajv"))," - JSON schema Validator by @epoberezkin and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/kazupon/vue-i18n"},(0,l.yg)("inlineCode",{parentName:"a"},"i18n"))," - Internationalization plugin by @kazupon and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/frappe/charts"},(0,l.yg)("inlineCode",{parentName:"a"},"frappe-charts"))," - Lightweight charting library by @frappe ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"))),(0,l.yg)("h3",{id:"frontend-components"},"Frontend Components"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/sagalbot/vue-select"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-select"))," - Dropdown component by @sagalbot ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/euvl/vue-js-modal"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-js-modal"))," - Modal component by @euvl ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/Akryum/v-tooltip"},(0,l.yg)("inlineCode",{parentName:"a"},"v-tooltip"))," - Tooltip component by @Akryum ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/jairoblatt/vue-material-tabs"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-material-tabs"))," - Tab view component by @jairoblatt ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/yansenlei/VJsoneditor"},(0,l.yg)("inlineCode",{parentName:"a"},"VJsoneditor"))," - Interactive JSON editor component by @yansenlei ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"),(0,l.yg)("ul",{parentName:"li"},(0,l.yg)("li",{parentName:"ul"},"Forked from ",(0,l.yg)("a",{parentName:"li",href:"https://github.com/josdejong/jsoneditor"},(0,l.yg)("inlineCode",{parentName:"a"},"JsonEditor"))," by @josdejong ",(0,l.yg)("inlineCode",{parentName:"li"},"Apache-2.0 License")))),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/shakee93/vue-toasted"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-toasted"))," - Toast notification component by @shakee93 ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/saintplay/vue-swatches"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-swatches"))," - Color palete picker by @saintplay ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"))),(0,l.yg)("h3",{id:"backup--sync-server"},"Backup & Sync Server"),(0,l.yg)("p",null,"Although the app is purely frontend, there is an optional cloud sync feature, for off-site backup and restore. This is built as a serverless function on ",(0,l.yg)("a",{parentName:"p",href:"https://workers.cloudflare.com/"},"Cloudflare workers")," using ",(0,l.yg)("a",{parentName:"p",href:"https://developers.cloudflare.com/workers/runtime-apis/kv"},"KV")," and ",(0,l.yg)("a",{parentName:"p",href:"https://developers.cloudflare.com/workers/runtime-apis/web-crypto"},"web crypto")),(0,l.yg)("h3",{id:"external-services"},"External Services"),(0,l.yg)("p",null,"The 1-Click deploy demo uses ",(0,l.yg)("a",{parentName:"p",href:"https://play-with-docker.com/"},"Play-with-Docker Labs"),". Code is hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://github.com"},"GitHub"),", Docker images are hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://hub.docker.com/"},"DockerHub"),", and the demos are hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://www.netlify.com/"},"Netlify"),"."),(0,l.yg)("h3",{id:"actions"},"Actions"),(0,l.yg)("p",null,"This repo makes heavy use of ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/features/actions"},"GitHub actions")," to run automated workflows. The following base actions are totally awesome, and have been extremely useful. Full credit to their respective authors"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/butlerlogic/action-autotag"},(0,l.yg)("inlineCode",{parentName:"a"},"action-autotag"))," by @butlerlogic - Tags new versions when app version changes"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/joeizzard/action-wiki-sync"},(0,l.yg)("inlineCode",{parentName:"a"},"action-wiki-sync"))," by @joeizzard - Copies the markdown docs to the GH wiki"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/bubkoo/contributors-list"},(0,l.yg)("inlineCode",{parentName:"a"},"contributors-list"))," by @bubkoo - Generates contributors SVG"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/akhilmhdh/contributors-readme-action"},(0,l.yg)("inlineCode",{parentName:"a"},"contributors-readme-action"))," by @akhilmhdh - Inserts contributors into credits"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/eter-evans/create-pull-request"},(0,l.yg)("inlineCode",{parentName:"a"},"create-pull-request"))," by @eter-evans - Creates pull request"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JamesIves/github-pages-deploy-action"},(0,l.yg)("inlineCode",{parentName:"a"},"github-pages-deploy-action"))," by @JamesIves - Deploy app to GH pages"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/apexskier/github-release-commenter"},(0,l.yg)("inlineCode",{parentName:"a"},"github-release-commenter"))," by @apexskier - Updates issues when fix is released"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JamesIves/github-sponsors-readme-action"},(0,l.yg)("inlineCode",{parentName:"a"},"github-sponsors-readme-action"))," by @JamesIves - Inserts sponsors into credits"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JasonEtco/is-sponsor-label-action"},(0,l.yg)("inlineCode",{parentName:"a"},"is-sponsor-label-action"))," by @JasonEtco - Label sponsors, for priority support"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/tomsun28/issues-translate-action"},(0,l.yg)("inlineCode",{parentName:"a"},"issues-translate-action"))," by @tomsun28 - Translates non-English issues and comments"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/sobolevn/misspell-fixer-action"},(0,l.yg)("inlineCode",{parentName:"a"},"misspell-fixer-action"))," by @sobolevn - Finds and fixes typos"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/uhyo/please-star-first"},(0,l.yg)("inlineCode",{parentName:"a"},"please-star-first"))," by @uhyo - Closes certain issues by non-stargazers"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/exercism/pr-commenter-action"},(0,l.yg)("inlineCode",{parentName:"a"},"pr-commenter-action"))," by @exercism - Adds info comments to PR based on the files changed"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/snyk/actions/node"},(0,l.yg)("inlineCode",{parentName:"a"},"snyk note"))," by @snyk - Reports vulnerabilities in the code or dependencies"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/alstr/todo-to-issue-action"},(0,l.yg)("inlineCode",{parentName:"a"},"todo-to-issue-action"))," by @alstr - Opens issues from todos in the code"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/Simek/yarn-lock-changes"},(0,l.yg)("inlineCode",{parentName:"a"},"yarn-lock-changes"))," by @Simek - Outputs dependency changes in PR")),(0,l.yg)("hr",null),(0,l.yg)("h2",{id:"you"},"You"),(0,l.yg)("p",null,"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,l.yg)("a",{parentName:"p",href:"/docs/contributing"},"Contributing Page")," for ways that you can get involved. Huge thank you to everyone who has already contributed! \ud83d\udc96"))}h.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/18ba09e8.b7906927.js b/assets/js/18ba09e8.b7906927.js deleted file mode 100644 index ae6bdae6..00000000 --- a/assets/js/18ba09e8.b7906927.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[849],{5680:(t,e,a)=>{a.d(e,{xA:()=>y,yg:()=>b});var n=a(6540);function l(t,e,a){return e in t?Object.defineProperty(t,e,{value:a,enumerable:!0,configurable:!0,writable:!0}):t[e]=a,t}function g(t,e){var a=Object.keys(t);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(t);e&&(n=n.filter((function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable}))),a.push.apply(a,n)}return a}function r(t){for(var e=1;e=0||(l[a]=t[a]);return l}(t,e);if(Object.getOwnPropertySymbols){var g=Object.getOwnPropertySymbols(t);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(t,a)&&(l[a]=t[a])}return l}var i=n.createContext({}),s=function(t){var e=n.useContext(i),a=e;return t&&(a="function"==typeof t?t(e):r(r({},e),t)),a},y=function(t){var e=s(t.components);return n.createElement(i.Provider,{value:e},t.children)},h="mdxType",o={inlineCode:"code",wrapper:function(t){var e=t.children;return n.createElement(n.Fragment,{},e)}},c=n.forwardRef((function(t,e){var a=t.components,l=t.mdxType,g=t.originalType,i=t.parentName,y=u(t,["components","mdxType","originalType","parentName"]),h=s(a),c=l,b=h["".concat(i,".").concat(c)]||h[c]||o[c]||g;return a?n.createElement(b,r(r({ref:e},y),{},{components:a})):n.createElement(b,r({ref:e},y))}));function b(t,e){var a=arguments,l=e&&e.mdxType;if("string"==typeof t||l){var g=a.length,r=new Array(g);r[0]=c;var u={};for(var i in e)hasOwnProperty.call(e,i)&&(u[i]=e[i]);u.originalType=t,u[h]="string"==typeof t?t:l,r[1]=u;for(var s=2;s{a.r(e),a.d(e,{contentTitle:()=>r,default:()=>h,frontMatter:()=>g,metadata:()=>u,toc:()=>i});var n=a(8168),l=(a(6540),a(5680));const g={},r="Credits",u={unversionedId:"credits",id:"credits",isDocsHomePage:!1,title:"Credits",description:"Sponsors",source:"@site/docs/credits.md",sourceDirName:".",slug:"/credits",permalink:"/docs/credits",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/credits.md",version:"current",frontMatter:{}},i=[{value:"Sponsors",id:"sponsors",children:[]},{value:"Contributors",id:"contributors",children:[]},{value:"Helpful Users",id:"helpful-users",children:[]},{value:"Bots",id:"bots",children:[]},{value:"Newest Stargazers",id:"newest-stargazers",children:[]},{value:"Dependencies",id:"dependencies",children:[{value:"Core",id:"core",children:[]},{value:"Utilities",id:"utilities",children:[]},{value:"Frontend Components",id:"frontend-components",children:[]},{value:"Backup & Sync Server",id:"backup--sync-server",children:[]},{value:"External Services",id:"external-services",children:[]},{value:"Actions",id:"actions",children:[]}]},{value:"You",id:"you",children:[]}],s={toc:i},y="wrapper";function h(t){let{components:e,...a}=t;return(0,l.yg)(y,(0,n.A)({},s,a,{components:e,mdxType:"MDXLayout"}),(0,l.yg)("h1",{id:"credits"},"Credits"),(0,l.yg)("h2",{id:"sponsors"},"Sponsors"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/koconder"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25068?u=582657b23622aaa3dfe68bd028a780f272f456fa&v=4",width:"80;",alt:"koconder"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Vincent Koc")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Admonstrator"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/69824?u=1e226d7a36cdd661c3e4cd486fea140d045b7d57&v=4",width:"80;",alt:"Admonstrator"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aaron Viehl")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/tbjers"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1117052?v=4",width:"80;",alt:"tbjers"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Torgny Bjers")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/emlazzarin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1141361?u=714e3487a3f2e0df721b01a0133945f075d3ff68&v=4",width:"80;",alt:"emlazzarin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Eddy Lazzarin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AnandChowdhary"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2841780?u=747e554b3a7f12eb20b7910e1c87d817844f714f&v=4",width:"80;",alt:"AnandChowdhary"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Anand Chowdhary")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shrippen"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2873570?v=4",width:"80;",alt:"shrippen"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bile0026"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5022496?u=aec96ad173c0ea9baaba93807efa8a848af6595c&v=4",width:"80;",alt:"bile0026"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Zach Biles")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/UlisesGascon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5110813?u=3c41facd8aa26154b9451de237c34b0f78d672a5&v=4",width:"80;",alt:"UlisesGascon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ulises Gasc\xf3n")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/digitalarche"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6546135?u=d033c9c16e8367987aec3f9ff5922bc67dd1eedf&v=4",width:"80;",alt:"digitalarche"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Digital Archeology")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/InDieTasten"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7047377?u=8d8f8017628b38bc46dcbf3620e194b01d3fb2d1&v=4",width:"80;",alt:"InDieTasten"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/araguaci"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7318668?v=4",width:"80;",alt:"araguaci"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bmcgonag"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7346620?u=2a0f9284f3e12ac1cc15288c254d1ec68a5081e8&v=4",width:"80;",alt:"bmcgonag"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brian McGonagill"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/vlad-timofeev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11474041?u=eee43705b54d2ec9f51fc4fcce5ad18dd17c87e4&v=4",width:"80;",alt:"vlad-timofeev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Vlad Timofeev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/helixzz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12218889?u=d06d0c103dfbdb99450623064f7da3c5a3675fb6&v=4",width:"80;",alt:"helixzz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"HeliXZz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/patvdv"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12430107?u=e8911c2fb91af4d30432f76da8c40927b2830bd7&v=4",width:"80;",alt:"patvdv"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Patrick Van Der Veken")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/mryesiller"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24632172?u=0d20f2d615158f87cd60a3398d3efb026c32f291&v=4",width:"80;",alt:"mryesiller"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"G\xf6ksel Ye\u015filler")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/allesauseinerhand"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32039836?v=4",width:"80;",alt:"allesauseinerhand"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/forwardemail"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32481436?v=4",width:"80;",alt:"forwardemail"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Forward Email - Open-source & Privacy-focused Email Service (2023)"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lamtrinhdev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/49742151?v=4",width:"80;",alt:"lamtrinhdev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"LamTrinh.Dev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Bastii717"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/53431819?u=604977bed6ad6875ada890d0d3765a4cacc2fa14&v=4",width:"80;",alt:"Bastii717"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/M2TD"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/85460457?v=4",width:"80;",alt:"M2TD"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/frankdez93"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87549420?v=4",width:"80;",alt:"frankdez93"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/terminaltrove"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/121595180?v=4",width:"80;",alt:"terminaltrove"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Terminal Trove")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/NixyJuppie"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/138570196?u=b102c222487905728b858704962d32759df29ebe&v=4",width:"80;",alt:"NixyJuppie"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nixy"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/nrvo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/151435968?u=e1dcb307fd0efdc45cddbe9490a7b956e4da6835&v=4",width:"80;",alt:"nrvo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))))),(0,l.yg)("h2",{id:"contributors"},"Contributors"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Lissy93"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1862727?v=4",width:"80;",alt:"Lissy93"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Sykes")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/liss-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87835202?v=4",width:"80;",alt:"liss-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/marekful"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10281476?v=4",width:"80;",alt:"marekful"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Marcell F\xfcl\xf6p")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/EVOTk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45015615?v=4",width:"80;",alt:"EVOTk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"EVOTk")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/snyk-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19733683?v=4",width:"80;",alt:"snyk-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Snyk Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/azerioxal"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5369885?v=4",width:"80;",alt:"azerioxal"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kenneth Church"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/m42e"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2410802?v=4",width:"80;",alt:"m42e"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Matthias Bilger")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/CrazyWolf13"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/96661824?v=4",width:"80;",alt:"CrazyWolf13"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Tobias")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/pinarruiz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37040888?v=4",width:"80;",alt:"pinarruiz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alejandro Pinar Ruiz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imjimmeh"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/2104997?v=4",width:"80;",alt:"imjimmeh"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kashif-se"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5568138?v=4",width:"80;",alt:"kashif-se"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kashif Sohail")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/walkxcode"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/71191962?v=4",width:"80;",alt:"walkxcode"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Walkx"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/josuablejeru"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37913833?v=4",width:"80;",alt:"josuablejeru"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Josua Blejeru ")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/evroon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11857441?v=4",width:"80;",alt:"evroon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Erik Vroon")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jammo2k5"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/8106602?v=4",width:"80;",alt:"jammo2k5"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jammo2k5")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lordpansar"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/14231148?v=4",width:"80;",alt:"lordpansar"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Magnus Sundstr\xf6m")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Cereal916"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7526937?v=4",width:"80;",alt:"Cereal916"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kristian Brasel")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/albcp"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3170731?v=4",width:"80;",alt:"albcp"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alberto"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/altearius"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/270430?v=4",width:"80;",alt:"altearius"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Chris Nielsen")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/UrekD"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/38784343?v=4",width:"80;",alt:"UrekD"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"UrekD")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/z3r0l1nk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/31653632?v=4",width:"80;",alt:"z3r0l1nk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mihai")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/wozboz"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/51856582?v=4",width:"80;",alt:"wozboz"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Totto16"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32566573?v=4",width:"80;",alt:"Totto16"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Totto16")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/toddejohnson"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/507545?v=4",width:"80;",alt:"toddejohnson"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Todd Johnson"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/remygrandin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1934515?v=4",width:"80;",alt:"remygrandin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Remygrandin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/DimitriDR"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56969769?v=4",width:"80;",alt:"DimitriDR"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dimitri")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Tracreed"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6306365?v=4",width:"80;",alt:"Tracreed"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"David Alasow")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/aviolaris"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/48277853?v=4",width:"80;",alt:"aviolaris"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Andreas Violaris")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rubenandre"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9402773?v=4",width:"80;",alt:"rubenandre"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"R\xfaben Silva")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rtm516"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5401186?v=4",width:"80;",alt:"rtm516"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Rtm516"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/zcq100"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/425234?v=4",width:"80;",alt:"zcq100"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/onedr0p"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/213795?v=4",width:"80;",alt:"onedr0p"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\u15ea\u0454\u03bd\u03b9\u03b7 \u15f7\u03c5\u043d\u029f")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/stanly0726"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/37040069?v=4",width:"80;",alt:"stanly0726"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stanly0726")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Bogyie"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/82003678?v=4",width:"80;",alt:"Bogyie"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Bogyeong Kim")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Tuzi555"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62188066?v=4",width:"80;",alt:"Tuzi555"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jakub Tuzar")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/berksmbl"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10000339?v=4",width:"80;",alt:"berksmbl"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Berk S\xfcmb\xfcl"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alucarddelta"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/20882097?v=4",width:"80;",alt:"alucarddelta"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brent")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/k073l"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/21180271?v=4",width:"80;",alt:"k073l"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kt-alt"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/51970249?v=4",width:"80;",alt:"kt-alt"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/mmihaly"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/50031464?v=4",width:"80;",alt:"mmihaly"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/patrickheeney"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1407228?v=4",width:"80;",alt:"patrickheeney"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Patrick Heeney")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rokiden"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11071229?v=4",width:"80;",alt:"rokiden"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Denis Kazimirov"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/zigotica"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/178855?v=4",width:"80;",alt:"zigotica"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sergi Meseguer")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BySempron"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/15928132?v=4",width:"80;",alt:"BySempron"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sergio")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ssrangisetti"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/46807508?v=4",width:"80;",alt:"ssrangisetti"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/itsmejoeeey"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9375730?v=4",width:"80;",alt:"itsmejoeeey"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Joey Miller")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/hockeymikey"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1435185?v=4",width:"80;",alt:"hockeymikey"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/thomaswienecke"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11446531?v=4",width:"80;",alt:"thomaswienecke"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Thomas Wienecke"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/deneor"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1063265?v=4",width:"80;",alt:"deneor"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/daentech"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/358678?v=4",width:"80;",alt:"daentech"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dan Gilbert")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/moemoeq"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1808434?v=4",width:"80;",alt:"moemoeq"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"CHAIYEON CHO")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BOZG"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6022344?v=4",width:"80;",alt:"BOZG"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stephen Rigney")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/a-mnich"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56564725?v=4",width:"80;",alt:"a-mnich"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alexander Mnich")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alexdelprete"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7027842?v=4",width:"80;",alt:"alexdelprete"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alessandro Del Prete"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alayham"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/518776?v=4",width:"80;",alt:"alayham"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Al-Ayham Saleh")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/sachahjkl"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32895534?v=4",width:"80;",alt:"sachahjkl"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Sacha")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shazzx"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/131521332?v=4",width:"80;",alt:"shazzx"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Shazz")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ThinkSalat"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/31082405?v=4",width:"80;",alt:"ThinkSalat"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Shawn Salat")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/royshreyaa"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/139828242?v=4",width:"80;",alt:"royshreyaa"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Smexhy"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/4880625?v=4",width:"80;",alt:"Smexhy"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/stavros-k"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/47820033?v=4",width:"80;",alt:"stavros-k"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Stavros Kois")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/XenonR"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/18627623?v=4",width:"80;",alt:"XenonR"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Steffen Schmidt")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/StevKast"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/17804308?v=4",width:"80;",alt:"StevKast"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Steven Kast")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AmadeusGraves"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/18572939?v=4",width:"80;",alt:"AmadeusGraves"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\xc1ngel Fern\xe1ndez S\xe1nchez")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/turnrye"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/701035?v=4",width:"80;",alt:"turnrye"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ryan Turner")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/rubjo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/42270947?v=4",width:"80;",alt:"rubjo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/PrynsTag"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56314705?v=4",width:"80;",alt:"PrynsTag"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Prince Carl Velasco")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/PlusaN"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/61884717?v=4",width:"80;",alt:"PlusaN"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Andrey")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/oka4shi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/67847553?v=4",width:"80;",alt:"oka4shi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"OKAMOTO Shigehiro")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bubylou"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3878640?v=4",width:"80;",alt:"bubylou"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nicholas Malcolm")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/miclav"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11891522?v=4",width:"80;",alt:"miclav"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Michael Lavaire")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imsakg"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62212589?v=4",width:"80;",alt:"imsakg"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mert Sefa AKGUN"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/ethan-hann"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/36464732?v=4",width:"80;",alt:"ethan-hann"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ethan Hann")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/tazboyz16"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12215340?v=4",width:"80;",alt:"tazboyz16"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/sur1v"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19678230?v=4",width:"80;",alt:"sur1v"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jos\xe9 Ignacio")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/soaibsafi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11424812?v=4",width:"80;",alt:"soaibsafi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Soaibuzzaman")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/pablomalo"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25877142?v=4",width:"80;",alt:"pablomalo"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/markxoe"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/59475466?v=4",width:"80;",alt:"markxoe"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Mark Oude Elberink"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lxjv"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/63261955?v=4",width:"80;",alt:"lxjv"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Laker Turner")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kxenoxx"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/9744900?v=4",width:"80;",alt:"kxenoxx"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kxenox")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jrobles98"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/30221842?v=4",width:"80;",alt:"jrobles98"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jnach"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/33467747?v=4",width:"80;",alt:"jnach"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jnach")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/imlonghao"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/4951333?v=4",width:"80;",alt:"imlonghao"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Imlonghao")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/icy-comet"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/50461557?v=4",width:"80;",alt:"icy-comet"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aniket Teredesai"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/huangshaohuai"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24775925?v=4",width:"80;",alt:"huangshaohuai"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/flechaig"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/10887132?v=4",width:"80;",alt:"flechaig"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dr460nf1r3"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/12834713?v=4",width:"80;",alt:"dr460nf1r3"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Nico")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/baifengheixi"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/98794233?v=4",width:"80;",alt:"baifengheixi"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/allozavrr"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/63748214?v=4",width:"80;",alt:"allozavrr"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/XertDev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/16572811?v=4",width:"80;",alt:"XertDev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Xert"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/emiran-orange"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/71817149?v=4",width:"80;",alt:"emiran-orange"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/edugof"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19559978?v=4",width:"80;",alt:"edugof"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Eduardo Gomez")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Dylan-Bs"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/35694107?v=4",width:"80;",alt:"Dylan-Bs"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Dylan Bersans")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dougaldhub"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25713235?v=4",width:"80;",alt:"dougaldhub"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Doug Lock")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dkyeremeh"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/52928432?v=4",width:"80;",alt:"dkyeremeh"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Desmond Kyeremeh")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/deepsourcebot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/60907429?v=4",width:"80;",alt:"deepsourcebot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"DeepSource Bot"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/DawidPietrykowski"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/53954695?v=4",width:"80;",alt:"DawidPietrykowski"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Null")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/skaarj1989"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/34756939?v=4",width:"80;",alt:"skaarj1989"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"David")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/clsty"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/129247596?v=4",width:"80;",alt:"clsty"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Celestial.y")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/bskim45"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5674934?v=4",width:"80;",alt:"bskim45"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Bumsoo Kim")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BhasherBEL"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45831883?v=4",width:"80;",alt:"BhasherBEL"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Brieuc Dubois")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BeginCI"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/57495754?v=4",width:"80;",alt:"BeginCI"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Begin"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/BRAVO68WEB"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/41448663?v=4",width:"80;",alt:"BRAVO68WEB"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jyotirmoy Bandyopadhyaya [Bravo68]")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/AaronPorts"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/32810520?v=4",width:"80;",alt:"AaronPorts"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Artyom")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/alydemah"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/652035?v=4",width:"80;",alt:"alydemah"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Aly Mohamed")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/5idereal"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/30827929?v=4",width:"80;",alt:"5idereal"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"5idereal")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/0n1cOn3"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/27576311?v=4",width:"80;",alt:"0n1cOn3"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"0n1cOn3")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/maximemoreillon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29086128?v=4",width:"80;",alt:"maximemoreillon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Maxime Moreillon"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Glitch3dPenguin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3271160?v=4",width:"80;",alt:"Glitch3dPenguin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Max Kulik")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/markusdd"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/25175069?v=4",width:"80;",alt:"markusdd"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Markus Krause")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/asenov"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/280619?v=4",width:"80;",alt:"asenov"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"\u041c\u0438\u0440\u043e\u0441\u043b\u0430\u0432 \u0410\u0441\u0435\u043d\u043e\u0432")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/luispabon"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6388823?v=4",width:"80;",alt:"luispabon"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Luis Pabon")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/LeoColman"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/1577251?v=4",width:"80;",alt:"LeoColman"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Leonardo Colman Lopes")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/KierenConnell"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/46445781?v=4",width:"80;",alt:"KierenConnell"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kieren Connell"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/kieraneglin"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/569917?v=4",width:"80;",alt:"kieraneglin"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Kieran")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Singebob"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/24290044?v=4",width:"80;",alt:"Singebob"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jeremy Chauvin")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Hellhium"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/11504877?v=4",width:"80;",alt:"Hellhium"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Jemy SCHNEPP")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/jjmung"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/6049600?v=4",width:"80;",alt:"jjmung"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"JJ Munguia")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/b1thunt3r"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/791091?v=4",width:"80;",alt:"b1thunt3r"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ishan Jain")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/Compunctus"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/5058853?v=4",width:"80;",alt:"Compunctus"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Iaroslav Dronskii"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/nealian"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/865115?v=4",width:"80;",alt:"nealian"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Ian Neal")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/FraglyG"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/56320839?v=4",width:"80;",alt:"FraglyG"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Hendrik Strydom")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/hubortje"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/62364169?v=4",width:"80;",alt:"hubortje"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Harald T\xf6pfer")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/gbrown09"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/3360055?v=4",width:"80;",alt:"gbrown09"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Garrett Brown")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/FormatToday"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/20515769?v=4",width:"80;",alt:"FormatToday"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"FormatToday")))))),(0,l.yg)("h2",{id:"helpful-users"},"Helpful Users"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/evotk"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/45015615?v=4",width:"80;",alt:"evotk"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Evotk")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/shadowking001"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/43928955?v=4",width:"80;",alt:"shadowking001"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"LawrenceP.")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/urekd"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/38784343?v=4",width:"80;",alt:"urekd"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Urekd")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dylanbeme"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/41838333?v=4",width:"80;",alt:"dylanbeme"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"DylanH")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/milesteg1"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29298312?v=4",width:"80;",alt:"milesteg1"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Milesteg1")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/lu4t"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/29571529?v=4",width:"80;",alt:"lu4t"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Lu4t")))))),(0,l.yg)("h2",{id:"bots"},"Bots"),(0,l.yg)("table",null,(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/liss-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/87835202?v=4",width:"80;",alt:"liss-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Alicia Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/snyk-bot"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/19733683?v=4",width:"80;",alt:"snyk-bot"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Snyk Bot")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/netlify"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/7892489?v=4",width:"80;",alt:"netlify"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Netlify")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/viezly"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/78649443?v=4",width:"80;",alt:"viezly"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"Viezly")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/muse-dev"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/u/42072939?v=4",width:"80;",alt:"muse-dev"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"MuseDev")))),(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/github-actions[bot]"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/in/15368?v=4",width:"80;",alt:"github-actions[bot]"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"github-actions[bot]"))))),(0,l.yg)("tr",null,(0,l.yg)("td",{align:"center"},(0,l.yg)("a",{href:"https://github.com/dependabot[bot]"},(0,l.yg)("img",{src:"https://avatars.githubusercontent.com/in/29110?v=4",width:"80;",alt:"dependabot[bot]"}),(0,l.yg)("br",null),(0,l.yg)("sub",null,(0,l.yg)("b",null,"dependabot[bot]")))))),(0,l.yg)("blockquote",null,(0,l.yg)("p",{parentName:"blockquote"},"The above section is auto-generated, using ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/marketplace/actions/contribute-list"},"contribute-list")," by @akhilmhdh.")),(0,l.yg)("h2",{id:"newest-stargazers"},"Newest Stargazers"),(0,l.yg)("p",null,(0,l.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/stargazers"},(0,l.yg)("img",{parentName:"a",src:"https://reporoster.com/stars/dark/Lissy93/dashy",alt:"Recent Star Gazers"}))),(0,l.yg)("p",null,(0,l.yg)("a",{parentName:"p",href:"https://github.com/lissy93/dashy/network/members"},(0,l.yg)("img",{parentName:"a",src:"https://reporoster.com/forks/dark/lissy93/dashy",alt:"Recent Forkers"}))),(0,l.yg)("hr",null),(0,l.yg)("h2",{id:"dependencies"},"Dependencies"),(0,l.yg)("p",null,"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. For a full breakdown of dependency licenses, please see ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/.github/LEGAL.md"},"Legal")),(0,l.yg)("h3",{id:"core"},"Core"),(0,l.yg)("p",null,"At it's core, the application uses ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/vuejs/vue"},(0,l.yg)("strong",{parentName:"a"},"Vue.js")),", as well as it's services with ",(0,l.yg)("a",{parentName:"p",href:"https://vuex.vuejs.org/"},(0,l.yg)("strong",{parentName:"a"},"VueX"))," for state management. Styling is done with ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/sass/sass"},(0,l.yg)("strong",{parentName:"a"},"SCSS")),", JavaScript is currently ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/babel/babel"},(0,l.yg)("strong",{parentName:"a"},"Babel")),", (but I am in the process of converting to ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/Microsoft/TypeScript"},(0,l.yg)("strong",{parentName:"a"},"TypeScript")),"). Linting is done with ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/eslint/eslint"},(0,l.yg)("strong",{parentName:"a"},"ESLint"))," and ",(0,l.yg)("a",{parentName:"p",href:"https://prettier.io/"},(0,l.yg)("strong",{parentName:"a"},"Prettier")),", both following the ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/airbnb/javascript"},(0,l.yg)("strong",{parentName:"a"},"AirBnB Styleguide")),". The config is defined in ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/yaml/yaml"},(0,l.yg)("strong",{parentName:"a"},"YAML")),", with a simple ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/nodejs/node"},(0,l.yg)("strong",{parentName:"a"},"Node.js"))," server to serve up the static app and the optional API endpoints, and container deployment is done with ",(0,l.yg)("a",{parentName:"p",href:"https://www.docker.com/"},(0,l.yg)("strong",{parentName:"a"},"Docker")),"."),(0,l.yg)("h3",{id:"utilities"},"Utilities"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/brix/crypto-js"},(0,l.yg)("inlineCode",{parentName:"a"},"crypto-js"))," - Encryption implementations by @evanvosberg and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/axios/axios"},(0,l.yg)("inlineCode",{parentName:"a"},"axios"))," - Promise based HTTP client by @mzabriskie and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/ajv-validator/ajv"},(0,l.yg)("inlineCode",{parentName:"a"},"ajv"))," - JSON schema Validator by @epoberezkin and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/kazupon/vue-i18n"},(0,l.yg)("inlineCode",{parentName:"a"},"i18n"))," - Internationalization plugin by @kazupon and community ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/frappe/charts"},(0,l.yg)("inlineCode",{parentName:"a"},"frappe-charts"))," - Lightweight charting library by @frappe ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"))),(0,l.yg)("h3",{id:"frontend-components"},"Frontend Components"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/sagalbot/vue-select"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-select"))," - Dropdown component by @sagalbot ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/euvl/vue-js-modal"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-js-modal"))," - Modal component by @euvl ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/Akryum/v-tooltip"},(0,l.yg)("inlineCode",{parentName:"a"},"v-tooltip"))," - Tooltip component by @Akryum ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/jairoblatt/vue-material-tabs"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-material-tabs"))," - Tab view component by @jairoblatt ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/yansenlei/VJsoneditor"},(0,l.yg)("inlineCode",{parentName:"a"},"VJsoneditor"))," - Interactive JSON editor component by @yansenlei ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"),(0,l.yg)("ul",{parentName:"li"},(0,l.yg)("li",{parentName:"ul"},"Forked from ",(0,l.yg)("a",{parentName:"li",href:"https://github.com/josdejong/jsoneditor"},(0,l.yg)("inlineCode",{parentName:"a"},"JsonEditor"))," by @josdejong ",(0,l.yg)("inlineCode",{parentName:"li"},"Apache-2.0 License")))),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/shakee93/vue-toasted"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-toasted"))," - Toast notification component by @shakee93 ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT")),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/saintplay/vue-swatches"},(0,l.yg)("inlineCode",{parentName:"a"},"vue-swatches"))," - Color palete picker by @saintplay ",(0,l.yg)("inlineCode",{parentName:"li"},"MIT"))),(0,l.yg)("h3",{id:"backup--sync-server"},"Backup & Sync Server"),(0,l.yg)("p",null,"Although the app is purely frontend, there is an optional cloud sync feature, for off-site backup and restore. This is built as a serverless function on ",(0,l.yg)("a",{parentName:"p",href:"https://workers.cloudflare.com/"},"Cloudflare workers")," using ",(0,l.yg)("a",{parentName:"p",href:"https://developers.cloudflare.com/workers/runtime-apis/kv"},"KV")," and ",(0,l.yg)("a",{parentName:"p",href:"https://developers.cloudflare.com/workers/runtime-apis/web-crypto"},"web crypto")),(0,l.yg)("h3",{id:"external-services"},"External Services"),(0,l.yg)("p",null,"The 1-Click deploy demo uses ",(0,l.yg)("a",{parentName:"p",href:"https://play-with-docker.com/"},"Play-with-Docker Labs"),". Code is hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://github.com"},"GitHub"),", Docker images are hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://hub.docker.com/"},"DockerHub"),", and the demos are hosted on ",(0,l.yg)("a",{parentName:"p",href:"https://www.netlify.com/"},"Netlify"),"."),(0,l.yg)("h3",{id:"actions"},"Actions"),(0,l.yg)("p",null,"This repo makes heavy use of ",(0,l.yg)("a",{parentName:"p",href:"https://github.com/features/actions"},"GitHub actions")," to run automated workflows. The following base actions are totally awesome, and have been extremely useful. Full credit to their respective authors"),(0,l.yg)("ul",null,(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/butlerlogic/action-autotag"},(0,l.yg)("inlineCode",{parentName:"a"},"action-autotag"))," by @butlerlogic - Tags new versions when app version changes"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/joeizzard/action-wiki-sync"},(0,l.yg)("inlineCode",{parentName:"a"},"action-wiki-sync"))," by @joeizzard - Copies the markdown docs to the GH wiki"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/bubkoo/contributors-list"},(0,l.yg)("inlineCode",{parentName:"a"},"contributors-list"))," by @bubkoo - Generates contributors SVG"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/akhilmhdh/contributors-readme-action"},(0,l.yg)("inlineCode",{parentName:"a"},"contributors-readme-action"))," by @akhilmhdh - Inserts contributors into credits"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/eter-evans/create-pull-request"},(0,l.yg)("inlineCode",{parentName:"a"},"create-pull-request"))," by @eter-evans - Creates pull request"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JamesIves/github-pages-deploy-action"},(0,l.yg)("inlineCode",{parentName:"a"},"github-pages-deploy-action"))," by @JamesIves - Deploy app to GH pages"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/apexskier/github-release-commenter"},(0,l.yg)("inlineCode",{parentName:"a"},"github-release-commenter"))," by @apexskier - Updates issues when fix is released"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JamesIves/github-sponsors-readme-action"},(0,l.yg)("inlineCode",{parentName:"a"},"github-sponsors-readme-action"))," by @JamesIves - Inserts sponsors into credits"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/JasonEtco/is-sponsor-label-action"},(0,l.yg)("inlineCode",{parentName:"a"},"is-sponsor-label-action"))," by @JasonEtco - Label sponsors, for priority support"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/tomsun28/issues-translate-action"},(0,l.yg)("inlineCode",{parentName:"a"},"issues-translate-action"))," by @tomsun28 - Translates non-English issues and comments"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/sobolevn/misspell-fixer-action"},(0,l.yg)("inlineCode",{parentName:"a"},"misspell-fixer-action"))," by @sobolevn - Finds and fixes typos"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/uhyo/please-star-first"},(0,l.yg)("inlineCode",{parentName:"a"},"please-star-first"))," by @uhyo - Closes certain issues by non-stargazers"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/exercism/pr-commenter-action"},(0,l.yg)("inlineCode",{parentName:"a"},"pr-commenter-action"))," by @exercism - Adds info comments to PR based on the files changed"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/snyk/actions/node"},(0,l.yg)("inlineCode",{parentName:"a"},"snyk note"))," by @snyk - Reports vulnerabilities in the code or dependencies"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/alstr/todo-to-issue-action"},(0,l.yg)("inlineCode",{parentName:"a"},"todo-to-issue-action"))," by @alstr - Opens issues from todos in the code"),(0,l.yg)("li",{parentName:"ul"},(0,l.yg)("a",{parentName:"li",href:"https://github.com/Simek/yarn-lock-changes"},(0,l.yg)("inlineCode",{parentName:"a"},"yarn-lock-changes"))," by @Simek - Outputs dependency changes in PR")),(0,l.yg)("hr",null),(0,l.yg)("h2",{id:"you"},"You"),(0,l.yg)("p",null,"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,l.yg)("a",{parentName:"p",href:"/docs/contributing"},"Contributing Page")," for ways that you can get involved. Huge thank you to everyone who has already contributed! \ud83d\udc96"))}h.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/625b1ed7.39397709.js b/assets/js/625b1ed7.39397709.js deleted file mode 100644 index 54575a90..00000000 --- a/assets/js/625b1ed7.39397709.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[831],{5680:(e,a,t)=>{t.d(a,{xA:()=>p,yg:()=>y});var n=t(6540);function o(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function i(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function r(e){for(var a=1;a=0||(o[t]=e[t]);return o}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(o[t]=e[t])}return o}var l=n.createContext({}),c=function(e){var a=n.useContext(l),t=a;return e&&(t="function"==typeof e?e(a):r(r({},a),e)),t},p=function(e){var a=c(e.components);return n.createElement(l.Provider,{value:a},e.children)},u="mdxType",h={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},d=n.forwardRef((function(e,a){var t=e.components,o=e.mdxType,i=e.originalType,l=e.parentName,p=s(e,["components","mdxType","originalType","parentName"]),u=c(t),d=o,y=u["".concat(l,".").concat(d)]||u[d]||h[d]||i;return t?n.createElement(y,r(r({ref:a},p),{},{components:t})):n.createElement(y,r({ref:a},p))}));function y(e,a){var t=arguments,o=a&&a.mdxType;if("string"==typeof e||o){var i=t.length,r=new Array(i);r[0]=d;var s={};for(var l in a)hasOwnProperty.call(a,l)&&(s[l]=a[l]);s.originalType=e,s[u]="string"==typeof e?e:o,r[1]=s;for(var c=2;c{t.r(a),t.d(a,{contentTitle:()=>r,default:()=>u,frontMatter:()=>i,metadata:()=>s,toc:()=>l});var n=t(8168),o=(t(6540),t(5680));const i={},r="Authentication",s={unversionedId:"authentication",id:"authentication",isDocsHomePage:!1,title:"Authentication",description:"- Basic Auth",source:"@site/docs/authentication.md",sourceDirName:".",slug:"/authentication",permalink:"/docs/authentication",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/authentication.md",version:"current",frontMatter:{},sidebar:"dashySidebar",previous:{title:"Status Indicators",permalink:"/docs/status-indicators"},next:{title:"Keyboard Shortcuts",permalink:"/docs/searching"}},l=[{value:"Built-In Auth",id:"built-in-auth",children:[{value:"Setting Up Authentication",id:"setting-up-authentication",children:[]},{value:"Hash Password",id:"hash-password",children:[]},{value:"Logging In and Out",id:"logging-in-and-out",children:[]},{value:"Enabling Guest Access",id:"enabling-guest-access",children:[]},{value:"Granular Access",id:"granular-access",children:[]},{value:"Permissions",id:"permissions",children:[]},{value:"Security",id:"security",children:[]}]},{value:"Keycloak",id:"keycloak",children:[{value:"1. Deploy Keycloak",id:"1-deploy-keycloak",children:[]},{value:"2. Setup Keycloak Users",id:"2-setup-keycloak-users",children:[]},{value:"3. Enable Keycloak in Dashy Config File",id:"3-enable-keycloak-in-dashy-config-file",children:[]},{value:"4. Add groups and roles (Optional)",id:"4-add-groups-and-roles-optional",children:[]}]},{value:"Alternative Authentication Methods",id:"alternative-authentication-methods",children:[{value:"Authentication Server",id:"authentication-server",children:[]},{value:"VPN",id:"vpn",children:[]},{value:"IP-Based Access",id:"ip-based-access",children:[]},{value:"Web Server Authentication",id:"web-server-authentication",children:[]},{value:"OAuth Services",id:"oauth-services",children:[]},{value:"Static Site Hosting Providers",id:"static-site-hosting-providers",children:[]}]}],c={toc:l},p="wrapper";function u(e){let{components:a,...t}=e;return(0,o.yg)(p,(0,n.A)({},c,t,{components:a,mdxType:"MDXLayout"}),(0,o.yg)("h1",{id:"authentication"},"Authentication"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#built-in-auth"},"Basic Auth"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#setting-up-authentication"},"Setting Up Authentication")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#hash-password"},"Hash Password")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#logging-in-and-out"},"Logging In and Out")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#enabling-guest-access"},"Guest Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#granular-access"},"Per-User Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#security"},"Security Considerations")))),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#keycloak"},"Keycloak Auth"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#1-deploy-keycloak"},"Deploying Keycloak")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#2-setup-keycloak-users"},"Setting up Keycloak")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#3-enable-keycloak-in-dashy-config-file"},"Configuring Dashy for Keycloak")))),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#alternative-authentication-methods"},"Alternative Authentication Methods"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#vpn"},"VPN")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#ip-based-access"},"IP-Based Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#web-server-authentication"},"Web Server Authentication")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#oauth-services"},"OAuth Services")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#static-site-hosting-providers"},"Auth on Cloud Hosting Services"))))),(0,o.yg)("blockquote",null,(0,o.yg)("p",{parentName:"blockquote"},"[!IMPORTANT]","\nDashy's built-in auth is not indented 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,o.yg)("p",{parentName:"blockquote"},"In cases where Dashy is only accessibly within your home network, and you just want to add a login page, then the built-in auth may be sufficient, but keep in mind that configuration can still be accessed.")),(0,o.yg)("h2",{id:"built-in-auth"},"Built-In Auth"),(0,o.yg)("p",null,"Dashy has a basic login page included, and frontend authentication. You can enable this by adding users to the ",(0,o.yg)("inlineCode",{parentName:"p"},"auth")," section under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig")," in your ",(0,o.yg)("inlineCode",{parentName:"p"},"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."),(0,o.yg)("blockquote",null,(0,o.yg)("p",{parentName:"blockquote"},"[!NOTE]","\nSince the auth is initiated in the main app entry point (for security), a rebuild is required to apply changes to the auth configuration.\nYou can trigger a rebuild through the UI, under Config --\x3e Rebuild, or by running ",(0,o.yg)("inlineCode",{parentName:"p"},"yarn build")," in the root directory.")),(0,o.yg)("h3",{id:"setting-up-authentication"},"Setting Up Authentication"),(0,o.yg)("p",null,"The ",(0,o.yg)("inlineCode",{parentName:"p"},"auth")," property takes an array of users. Each user needs to include a username, hash and optional user type (",(0,o.yg)("inlineCode",{parentName:"p"},"admin")," or ",(0,o.yg)("inlineCode",{parentName:"p"},"normal"),"). The hash property is a ",(0,o.yg)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/SHA-2"},"SHA-256 Hash")," of your desired password."),(0,o.yg)("p",null,"For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"appConfig:\n auth:\n users:\n - user: alicia\n hash: 4D1E58C90B3B94BCAD9848ECCACD6D2A8C9FBC5CA913304BBA5CDEAB36FEEFA3\n type: admin\n - user: bob\n hash: 5E884898DA28047151D0E56F8DC6292773603D0D6AABBDD62A11EF721D1542D8\n")),(0,o.yg)("h3",{id:"hash-password"},"Hash Password"),(0,o.yg)("p",null,"Dashy uses ",(0,o.yg)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Sha-256"},"SHA-256 Hash"),", a 64-character string, which you can generate using an online tool, such as ",(0,o.yg)("a",{parentName:"p",href:"https://passwordsgenerator.net/sha256-hash-generator/"},"this one")," or ",(0,o.yg)("a",{parentName:"p",href:"https://gchq.github.io/CyberChef/"},"CyberChef")," (which can be self-hosted/ ran locally)."),(0,o.yg)("p",null,"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."),(0,o.yg)("h3",{id:"logging-in-and-out"},"Logging In and Out"),(0,o.yg)("p",null,"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."),(0,o.yg)("h3",{id:"enabling-guest-access"},"Enabling Guest Access"),(0,o.yg)("p",null,"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,o.yg)("inlineCode",{parentName:"p"},"appConfig.auth.enableGuestAccess: true"),"."),(0,o.yg)("h3",{id:"granular-access"},"Granular Access"),(0,o.yg)("p",null,"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."),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"hideForUsers")," - Page, Section or Item will be visible to all users, except for those specified in this list"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"showForUsers")," - Page, Section or Item will be hidden from all users, except for those specified in this list"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"hideForGuests")," - Page, Section or Item will be visible for logged in users, but not for guests")),(0,o.yg)("p",null,"For Example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"- name: Code Analysis & Monitoring\n icon: fas fa-code\n displayData:\n cols: 2\n hideForUsers: [alicia, bob]\n items:\n ...\n")),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"- 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")),(0,o.yg)("h3",{id:"permissions"},"Permissions"),(0,o.yg)("p",null,"Any user who is not an admin (with ",(0,o.yg)("inlineCode",{parentName:"p"},"type: admin"),") will not be able to write changes to disk."),(0,o.yg)("p",null,"You can also prevent any user from writing changes to disk, using ",(0,o.yg)("inlineCode",{parentName:"p"},"preventWriteToDisk"),". Or prevent any changes from being saved locally in browser storage, using ",(0,o.yg)("inlineCode",{parentName:"p"},"preventLocalSave"),". Both properties can be found under ",(0,o.yg)("a",{parentName:"p",href:"/docs/configuring#appconfig-optional"},(0,o.yg)("inlineCode",{parentName:"a"},"appConfig")),"."),(0,o.yg)("p",null,"To disable all UI config features, including View Config, set ",(0,o.yg)("inlineCode",{parentName:"p"},"disableConfiguration"),". Alternatively you can disable UI config features for all non admin users by setting ",(0,o.yg)("inlineCode",{parentName:"p"},"disableConfigurationForNonAdmin")," to true."),(0,o.yg)("h3",{id:"security"},"Security"),(0,o.yg)("p",null,"With basic auth, all logic is happening on the client-side, which could mean a skilled user could manipulate the code to view parts of your configuration, including the hash. If the SHA-256 hash is of a common password, it may be possible to determine it, using a lookup table, in order to find the original password. Which can be used to manually generate the auth token, that can then be inserted into session storage, to become a valid logged in user. Therefore, you should always use a long, strong and unique password, and if you instance contains security-critical info and/ or is exposed directly to the internet, and alternative authentication method may be better. The purpose of the login page is merely to prevent immediate unauthorized access to your homepage."),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},(0,o.yg)("a",{parentName:"strong",href:"#top"},"\u2b06\ufe0f Back to Top"))),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"keycloak"},"Keycloak"),(0,o.yg)("p",null,"Dashy also supports using a ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/"},"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."),(0,o.yg)("p",null,(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/about.html"},"Keycloak")," is a Java-based ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/keycloak/keycloak"},"open source"),", high-performance, secure authentication system, supported by ",(0,o.yg)("a",{parentName:"p",href:"https://www.redhat.com/en"},"RedHat"),". It is easy to setup (",(0,o.yg)("a",{parentName:"p",href:"https://quay.io/repository/keycloak/keycloak"},"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,o.yg)("a",{parentName:"p",href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_development/topics/themes.html"},"themes"),", ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/extensions.html"},"plugins"),", ",(0,o.yg)("a",{parentName:"p",href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_admin/topics/authentication/password-policies.html"},"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."),(0,o.yg)("h3",{id:"1-deploy-keycloak"},"1. Deploy Keycloak"),(0,o.yg)("p",null,"First thing to do is to spin up a new instance of Keycloak. You will need ",(0,o.yg)("a",{parentName:"p",href:"https://docs.docker.com/engine/install/"},"Docker installed"),", and can then choose a tag, and pull the container from ",(0,o.yg)("a",{parentName:"p",href:"https://quay.io/repository/keycloak/keycloak"},"quay.io/keycloak/keycloak")),(0,o.yg)("p",null,"Use the following run command, replacing the attributes (default credentials, port and name), or incorporate this into your docker-compose file."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-bash"},"docker run -d \\\n -p 8081:8080 \\\n --name auth-server \\\n -e KEYCLOAK_USER=admin \\\n -e KEYCLOAK_PASSWORD=admin \\\n quay.io/keycloak/keycloak:15.0.2\n")),(0,o.yg)("p",null,"If you need to pull from DockerHub, a non-official image is available ",(0,o.yg)("a",{parentName:"p",href:"https://registry.hub.docker.com/r/jboss/keycloak"},"here"),". Or if you would prefer not to use Docker, you can also directly install Keycloak from source, following ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/docs/latest/getting_started/index.html"},"this guide"),"."),(0,o.yg)("p",null,"You should now be able to access the Keycloak web interface, using the port specified above (e.g. ",(0,o.yg)("inlineCode",{parentName:"p"},"http://127.0.0.1:8081"),"), login with the default credentials, and when prompted create a new password."),(0,o.yg)("h3",{id:"2-setup-keycloak-users"},"2. Setup Keycloak Users"),(0,o.yg)("p",null,"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."),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"Head over to the admin console"),(0,o.yg)("li",{parentName:"ol"},"In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm'"),(0,o.yg)("li",{parentName:"ol"},"Give your realm a name, and hit 'Create'")),(0,o.yg)("p",null,"You can now create your first user."),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"In the left-hand menu, click 'Users', then 'Add User'"),(0,o.yg)("li",{parentName:"ol"},"Fill in the form, including username and hit 'Save'"),(0,o.yg)("li",{parentName:"ol"},"Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login")),(0,o.yg)("p",null,"The last thing we need to do in the Keycloak admin console is to create a new client"),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right"),(0,o.yg)("li",{parentName:"ol"},"Choose a 'Client ID', set 'Client Protocol' to 'openid-connect', and for 'Valid Redirect URIs' put a URL pattern to where you're hosting Dashy (if you're just testing locally, then * is fine), and do the same for the 'Web Origins' field"),(0,o.yg)("li",{parentName:"ol"},"Make note of your client-id, and click 'Save'")),(0,o.yg)("h3",{id:"3-enable-keycloak-in-dashy-config-file"},"3. Enable Keycloak in Dashy Config File"),(0,o.yg)("p",null,"Now that your Keycloak instance is up and running, all that's left to do is to configure Dashy to use it. Under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig"),", set ",(0,o.yg)("inlineCode",{parentName:"p"},"auth.enableKeycloak: true"),", then fill in the details in ",(0,o.yg)("inlineCode",{parentName:"p"},"auth.keycloak"),", including: ",(0,o.yg)("inlineCode",{parentName:"p"},"serverUrl")," - the URL where your Keycloak instance is hosted, ",(0,o.yg)("inlineCode",{parentName:"p"},"realm")," - the name you gave your Realm, and ",(0,o.yg)("inlineCode",{parentName:"p"},"clientId")," - the Client ID you chose.\nFor example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"appConfig:\n ...\n auth:\n enableKeycloak: true\n keycloak:\n serverUrl: 'http://localhost:8081'\n realm: 'alicia-homelab'\n clientId: 'dashy'\n")),(0,o.yg)("p",null,"Note that if you are using Keycloak V 17 or older, you will also need to set ",(0,o.yg)("inlineCode",{parentName:"p"},"legacySupport: true")," (also under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig.auth.keycloak"),"). This is because the API endpoint was updated in later versions."),(0,o.yg)("p",null,"If you use Keycloak with an external Identity Provier, you can set the ",(0,o.yg)("inlineCode",{parentName:"p"},"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'."),(0,o.yg)("h3",{id:"4-add-groups-and-roles-optional"},"4. Add groups and roles (Optional)"),(0,o.yg)("p",null,"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,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/docs/latest/server_admin/index.html#assigning-permissions-and-access-using-roles-and-groups"},"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,o.yg)("inlineCode",{parentName:"p"},"displayData.showForKeycloakUser")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"displayData.hideForKeycloakUser"),".\nBoth show and hide configurations accept a list of ",(0,o.yg)("inlineCode",{parentName:"p"},"groups")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"roles")," that limit access. If a users data matches one or more items in these lists they will be allowed or excluded as defined."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,o.yg)("p",null,"Depending on how you're hosting Dashy and Keycloak, you may also need to set some HTTP headers, to prevent a CORS error. This would typically be the ",(0,o.yg)("inlineCode",{parentName:"p"},"Access-Control-Allow-Origin [URL-of Dashy]")," on your Keycloak instance. See the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/management.md#setting-headers"},"Setting Headers")," guide in the management docs for more info."),(0,o.yg)("p",null,"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."),(0,o.yg)("p",null,"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,o.yg)("inlineCode",{parentName:"p"},"docker-compose.yml")," file, and this is recommended."),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"alternative-authentication-methods"},"Alternative Authentication Methods"),(0,o.yg)("p",null,"If you are self-hosting Dashy, and require secure authentication to prevent unauthorized access, then you can either use Keycloak, or one of the following options:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#authentication-server"},"Authentication Server")," - Put Dashy behind a self-hosted auth server"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#vpn"},"VPN")," - Use a VPN to tunnel into the network where Dashy is running"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#ip-based-access"},"IP-Based Access")," - Disallow access from all IP addresses, except your own"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#web-server-authentication"},"Web Server Authentication")," - Enable user control within your web server or proxy"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#oauth-services"},"OAuth Services")," - Implement a user management system using a cloud provider"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#static-site-hosting-providers"},"Password Protection (for cloud providers)")," - Enable password-protection on your site")),(0,o.yg)("h3",{id:"authentication-server"},"Authentication Server"),(0,o.yg)("h4",{id:"authelia"},"Authelia"),(0,o.yg)("p",null,(0,o.yg)("a",{parentName:"p",href:"https://www.authelia.com/"},"Authelia")," is an open-source full-featured authentication server, which can be self-hosted and either on bare metal, in a Docker container or in a Kubernetes cluster. It allows for fine-grained access control rules based on IP, path, users etc, and supports 2FA, simple password access or bypass policies for your domains."),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"git clone https://github.com/authelia/authelia.git")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"cd authelia/examples/compose/lite")),(0,o.yg)("li",{parentName:"ul"},"Modify the ",(0,o.yg)("inlineCode",{parentName:"li"},"users_database.yml")," the default username and password is authelia"),(0,o.yg)("li",{parentName:"ul"},"Modify the ",(0,o.yg)("inlineCode",{parentName:"li"},"configuration.yml")," and ",(0,o.yg)("inlineCode",{parentName:"li"},"docker-compose.yml")," with your respective domains and secrets"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"docker-compose up -d"))),(0,o.yg)("p",null,"For more information, see the ",(0,o.yg)("a",{parentName:"p",href:"https://www.authelia.com/docs/"},"Authelia docs")),(0,o.yg)("h3",{id:"vpn"},"VPN"),(0,o.yg)("p",null,"A catch-all solution to accessing services running from your home network remotely is to use a VPN. It means you do not need to worry about implementing complex authentication rules, or trusting the login implementation of individual applications. However it can be inconvenient to use on a day-to-day basis, and some public and corporate WiFi block VPN connections. Two popular VPN protocols are ",(0,o.yg)("a",{parentName:"p",href:"https://openvpn.net/"},"OpenVPN")," and ",(0,o.yg)("a",{parentName:"p",href:"https://www.wireguard.com/"},"WireGuard")),(0,o.yg)("h3",{id:"ip-based-access"},"IP-Based Access"),(0,o.yg)("p",null,"If you have a static IP or use a VPN to access your running services, then you can use conditional access to block access to Dashy from everyone except users of your pre-defined IP address. This feature is offered by most cloud providers, and supported by most web servers."),(0,o.yg)("h4",{id:"apache"},"Apache"),(0,o.yg)("p",null,"In Apache, this is configured in your ",(0,o.yg)("inlineCode",{parentName:"p"},".htaccess")," file in Dashy's root folder, and should look something like:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"Order Deny,Allow\nDeny from all\nAllow from [your-ip]\n")),(0,o.yg)("h4",{id:"nginx"},"NGINX"),(0,o.yg)("p",null,"In NGINX you can specify ",(0,o.yg)("a",{parentName:"p",href:"https://docs.nginx.com/nginx/admin-guide/security-controls/controlling-access-proxied-http/"},"control access")," rules for a given site in your ",(0,o.yg)("inlineCode",{parentName:"p"},"nginx.conf")," or hosts file. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"server {\n listen 8080;\n server_name www.dashy.example.com;\n location / {\n root /path/to/dashy/;\n passenger_enabled on;\n allow [your-ip];\n deny all;\n }\n }\n")),(0,o.yg)("h4",{id:"caddy"},"Caddy"),(0,o.yg)("p",null,"In Caddy, ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/caddyfile/matchers"},"Request Matchers")," can be used to filter requests"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'dashy.site {\n @public_networks not remote_ip [your-ip]\n respond @public_networks "Access denied" 403\n}\n')),(0,o.yg)("h3",{id:"web-server-authentication"},"Web Server Authentication"),(0,o.yg)("p",null,"Most web servers make password protecting certain apps very easy. Note that you should also set up HTTPS and have a valid certificate in order for this to be secure."),(0,o.yg)("h4",{id:"apache-1"},"Apache"),(0,o.yg)("p",null,"First crate a ",(0,o.yg)("inlineCode",{parentName:"p"},".htaccess")," file in Dashy's route directory. Specify the auth type and path to where you want to store the password file (usually the same folder). For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'AuthType Basic\nAuthName "Please Sign into Dashy"\nAuthUserFile /path/dashy/.htpasswd\nrequire valid-user\n')),(0,o.yg)("p",null,"Then create a ",(0,o.yg)("inlineCode",{parentName:"p"},".htpasswd")," file in the same directory. List users and their hashed passwords here, with one user on each line, and a colon between username and password (e.g. ",(0,o.yg)("inlineCode",{parentName:"p"},"[username]:[hashed-password]"),"). You will need to generate an MD5 hash of your desired password, this can be done with an ",(0,o.yg)("a",{parentName:"p",href:"https://www.web2generators.com/apache-tools/htpasswd-generator"},"online tool"),". Your file will look something like:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"alicia:$apr1$jv0spemw$RzOX5/GgY69JMkgV6u16l0\n")),(0,o.yg)("h4",{id:"nginx-1"},"NGINX"),(0,o.yg)("p",null,"NGINX has an ",(0,o.yg)("a",{parentName:"p",href:"https://nginx.org/en/docs/http/ngx_http_auth_basic_module.html"},"authentication module")," which can be used to add passwords to given sites, and is fairly simple to set up. Similar to above, you will need to create a ",(0,o.yg)("inlineCode",{parentName:"p"},".htpasswd")," file. Then just enable auth and specify the path to that file, for example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'location / {\n auth_basic "closed site";\n auth_basic_user_file conf/htpasswd;\n}\n')),(0,o.yg)("h4",{id:"caddy-1"},"Caddy"),(0,o.yg)("p",null,"Caddy has a ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/caddyfile/directives/basicauth"},"basic-auth")," directive, where you specify a username and hash. The password hash needs to be base-64 encoded, the ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/command-line#caddy-hash-password"},(0,o.yg)("inlineCode",{parentName:"a"},"caddy hash-password"))," command can help with this. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"basicauth /secret/* {\n alicia JDJhJDEwJEVCNmdaNEg2Ti5iejRMYkF3MFZhZ3VtV3E1SzBWZEZ5Q3VWc0tzOEJwZE9TaFlZdEVkZDhX\n}\n")),(0,o.yg)("p",null,"For more info about implementing a single sign on for all your apps with Caddy, see ",(0,o.yg)("a",{parentName:"p",href:"https://joshstrange.com/securing-your-self-hosted-apps-with-single-signon/"},"this tutorial")),(0,o.yg)("h4",{id:"lighttpd"},"Lighttpd"),(0,o.yg)("p",null,"You can use the ",(0,o.yg)("a",{parentName:"p",href:"https://doc.lighttpd.net/lighttpd2/mod_auth.html"},"mod_auth")," module to secure your site with Lighttpd. Like with Apache, you need to first create a password file listing your usernames and hashed passwords, but in Lighttpd, it's usually called ",(0,o.yg)("inlineCode",{parentName:"p"},".lighttpdpassword"),"."),(0,o.yg)("p",null,"Then in your ",(0,o.yg)("inlineCode",{parentName:"p"},"lighttpd.conf")," file (usually in the ",(0,o.yg)("inlineCode",{parentName:"p"},"/etc/lighttpd/")," directory), load in the mod_auth module, and configure it's directives. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'server.modules += ( "mod_auth" )\nauth.debug = 2\nauth.backend = "plain"\nauth.backend.plain.userfile = "/home/lighttpd/.lighttpdpassword"\n\n$HTTP["host"] == "dashy.my-domain.net" {\n server.document-root = "/home/lighttpd/dashy.my-domain.net/http"\n server.errorlog = "/var/log/lighttpd/dashy.my-domain.net/error.log"\n accesslog.filename = "/var/log/lighttpd/dashy.my-domain.net/access.log"\n auth.require = (\n "/docs/" => (\n "method" => "basic",\n "realm" => "Password protected area",\n "require" => "user=alicia"\n )\n )\n}\n')),(0,o.yg)("p",null,"Restart your web server for changes to take effect."),(0,o.yg)("h3",{id:"oauth-services"},"OAuth Services"),(0,o.yg)("p",null,"There are also authentication services, such as ",(0,o.yg)("a",{parentName:"p",href:"https://www.ory.sh/"},"Ory.sh"),", ",(0,o.yg)("a",{parentName:"p",href:"https://developer.okta.com/"},"Okta"),", ",(0,o.yg)("a",{parentName:"p",href:"https://auth0.com/"},"Auth0"),", ",(0,o.yg)("a",{parentName:"p",href:"https://firebase.google.com/docs/auth/"},"Firebase"),". Implementing one of these solutions would involve some changes to the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/src/utils/Auth.js"},(0,o.yg)("inlineCode",{parentName:"a"},"Auth.js"))," file, but should be fairly straightforward."),(0,o.yg)("h3",{id:"static-site-hosting-providers"},"Static Site Hosting Providers"),(0,o.yg)("p",null,"If you are hosting Dashy on a cloud platform, you will probably find that it has built-in support for password protected access to web apps. For more info, see the relevant docs for your provider, for example: ",(0,o.yg)("a",{parentName:"p",href:"https://docs.netlify.com/visitor-access/password-protection/"},"Netlify Password Protection"),", ",(0,o.yg)("a",{parentName:"p",href:"https://www.cloudflare.com/teams/access/"},"Cloudflare Access"),", ",(0,o.yg)("a",{parentName:"p",href:"https://aws.amazon.com/cognito/"},"AWS Cognito"),", ",(0,o.yg)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/azure/app-service/scenario-secure-app-authentication-app-service"},"Azure Authentication")," and ",(0,o.yg)("a",{parentName:"p",href:"https://vercel.com/docs/platform/projects#password-protection"},"Vercel Password Protection"),"."),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},(0,o.yg)("a",{parentName:"strong",href:"#top"},"\u2b06\ufe0f Back to Top"))))}u.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/625b1ed7.b95d042d.js b/assets/js/625b1ed7.b95d042d.js new file mode 100644 index 00000000..77268e03 --- /dev/null +++ b/assets/js/625b1ed7.b95d042d.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[831],{5680:(e,a,t)=>{t.d(a,{xA:()=>u,yg:()=>y});var n=t(6540);function o(e,a,t){return a in e?Object.defineProperty(e,a,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[a]=t,e}function i(e,a){var t=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);a&&(n=n.filter((function(a){return Object.getOwnPropertyDescriptor(e,a).enumerable}))),t.push.apply(t,n)}return t}function r(e){for(var a=1;a=0||(o[t]=e[t]);return o}(e,a);if(Object.getOwnPropertySymbols){var i=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,t)&&(o[t]=e[t])}return o}var l=n.createContext({}),p=function(e){var a=n.useContext(l),t=a;return e&&(t="function"==typeof e?e(a):r(r({},a),e)),t},u=function(e){var a=p(e.components);return n.createElement(l.Provider,{value:a},e.children)},c="mdxType",h={inlineCode:"code",wrapper:function(e){var a=e.children;return n.createElement(n.Fragment,{},a)}},d=n.forwardRef((function(e,a){var t=e.components,o=e.mdxType,i=e.originalType,l=e.parentName,u=s(e,["components","mdxType","originalType","parentName"]),c=p(t),d=o,y=c["".concat(l,".").concat(d)]||c[d]||h[d]||i;return t?n.createElement(y,r(r({ref:a},u),{},{components:t})):n.createElement(y,r({ref:a},u))}));function y(e,a){var t=arguments,o=a&&a.mdxType;if("string"==typeof e||o){var i=t.length,r=new Array(i);r[0]=d;var s={};for(var l in a)hasOwnProperty.call(a,l)&&(s[l]=a[l]);s.originalType=e,s[c]="string"==typeof e?e:o,r[1]=s;for(var p=2;p{t.r(a),t.d(a,{contentTitle:()=>r,default:()=>c,frontMatter:()=>i,metadata:()=>s,toc:()=>l});var n=t(8168),o=(t(6540),t(5680));const i={},r="Authentication",s={unversionedId:"authentication",id:"authentication",isDocsHomePage:!1,title:"Authentication",description:"- Basic Auth",source:"@site/docs/authentication.md",sourceDirName:".",slug:"/authentication",permalink:"/docs/authentication",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/authentication.md",version:"current",frontMatter:{},sidebar:"dashySidebar",previous:{title:"Status Indicators",permalink:"/docs/status-indicators"},next:{title:"Keyboard Shortcuts",permalink:"/docs/searching"}},l=[{value:"Built-In Auth",id:"built-in-auth",children:[{value:"Setting Up Authentication",id:"setting-up-authentication",children:[]},{value:"Hash Password",id:"hash-password",children:[]},{value:"Logging In and Out",id:"logging-in-and-out",children:[]},{value:"Enabling Guest Access",id:"enabling-guest-access",children:[]},{value:"Granular Access",id:"granular-access",children:[]},{value:"Permissions",id:"permissions",children:[]},{value:"Using Environment Variables for Passwords",id:"using-environment-variables-for-passwords",children:[]},{value:"Adding HTTP Auth to Configuration",id:"adding-http-auth-to-configuration",children:[]},{value:"Security",id:"security",children:[]}]},{value:"HTTP Auth",id:"http-auth",children:[]},{value:"Keycloak",id:"keycloak",children:[{value:"1. Deploy Keycloak",id:"1-deploy-keycloak",children:[]},{value:"2. Setup Keycloak Users",id:"2-setup-keycloak-users",children:[]},{value:"3. Enable Keycloak in Dashy Config File",id:"3-enable-keycloak-in-dashy-config-file",children:[]},{value:"4. Add groups and roles (Optional)",id:"4-add-groups-and-roles-optional",children:[]}]},{value:"Alternative Authentication Methods",id:"alternative-authentication-methods",children:[{value:"Authentication Server",id:"authentication-server",children:[]},{value:"VPN",id:"vpn",children:[]},{value:"IP-Based Access",id:"ip-based-access",children:[]},{value:"Web Server Authentication",id:"web-server-authentication",children:[]},{value:"OAuth Services",id:"oauth-services",children:[]},{value:"Static Site Hosting Providers",id:"static-site-hosting-providers",children:[]}]}],p={toc:l},u="wrapper";function c(e){let{components:a,...t}=e;return(0,o.yg)(u,(0,n.A)({},p,t,{components:a,mdxType:"MDXLayout"}),(0,o.yg)("h1",{id:"authentication"},"Authentication"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#built-in-auth"},"Basic Auth"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#setting-up-authentication"},"Setting Up Authentication")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#hash-password"},"Hash Password")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#logging-in-and-out"},"Logging In and Out")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#enabling-guest-access"},"Guest Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#granular-access"},"Per-User Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#using-environment-variables-for-passwords"},"Using Environment Variables for Passwords")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#adding-http-auth-to-configuration"},"Adding HTTP Auth to Configuration")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#security"},"Security Considerations")))),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#http-auth"},"HTTP Auth")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#keycloak"},"Keycloak Auth"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#1-deploy-keycloak"},"Deploying Keycloak")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#2-setup-keycloak-users"},"Setting up Keycloak")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#3-enable-keycloak-in-dashy-config-file"},"Configuring Dashy for Keycloak")))),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#alternative-authentication-methods"},"Alternative Authentication Methods"),(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#vpn"},"VPN")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#ip-based-access"},"IP-Based Access")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#web-server-authentication"},"Web Server Authentication")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#oauth-services"},"OAuth Services")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#static-site-hosting-providers"},"Auth on Cloud Hosting Services"))))),(0,o.yg)("blockquote",null,(0,o.yg)("p",{parentName:"blockquote"},"[!IMPORTANT]","\nDashy's built-in auth is not indented 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,o.yg)("p",{parentName:"blockquote"},"In cases where Dashy is only accessibly within your home network, and you just want to add a login page, then the built-in auth may be sufficient, but keep in mind that configuration can still be accessed.")),(0,o.yg)("h2",{id:"built-in-auth"},"Built-In Auth"),(0,o.yg)("p",null,"Dashy has a basic login page included, and frontend authentication. You can enable this by adding users to the ",(0,o.yg)("inlineCode",{parentName:"p"},"auth")," section under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig")," in your ",(0,o.yg)("inlineCode",{parentName:"p"},"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."),(0,o.yg)("blockquote",null,(0,o.yg)("p",{parentName:"blockquote"},"[!NOTE]","\nSince the auth is initiated in the main app entry point (for security), a rebuild is required to apply changes to the auth configuration.\nYou can trigger a rebuild through the UI, under Config --\x3e Rebuild, or by running ",(0,o.yg)("inlineCode",{parentName:"p"},"yarn build")," in the root directory.")),(0,o.yg)("h3",{id:"setting-up-authentication"},"Setting Up Authentication"),(0,o.yg)("p",null,"The ",(0,o.yg)("inlineCode",{parentName:"p"},"auth")," property takes an array of users. Each user needs to include a username, hash and optional user type (",(0,o.yg)("inlineCode",{parentName:"p"},"admin")," or ",(0,o.yg)("inlineCode",{parentName:"p"},"normal"),"). The hash property is a ",(0,o.yg)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/SHA-2"},"SHA-256 Hash")," of your desired password."),(0,o.yg)("p",null,"For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"appConfig:\n auth:\n users:\n - user: alicia\n hash: 4D1E58C90B3B94BCAD9848ECCACD6D2A8C9FBC5CA913304BBA5CDEAB36FEEFA3\n type: admin\n - user: bob\n hash: 5E884898DA28047151D0E56F8DC6292773603D0D6AABBDD62A11EF721D1542D8\n")),(0,o.yg)("h3",{id:"hash-password"},"Hash Password"),(0,o.yg)("p",null,"Dashy uses ",(0,o.yg)("a",{parentName:"p",href:"https://en.wikipedia.org/wiki/Sha-256"},"SHA-256 Hash"),", a 64-character string, which you can generate using an online tool, such as ",(0,o.yg)("a",{parentName:"p",href:"https://passwordsgenerator.net/sha256-hash-generator/"},"this one")," or ",(0,o.yg)("a",{parentName:"p",href:"https://gchq.github.io/CyberChef/"},"CyberChef")," (which can be self-hosted/ ran locally)."),(0,o.yg)("p",null,"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."),(0,o.yg)("h3",{id:"logging-in-and-out"},"Logging In and Out"),(0,o.yg)("p",null,"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."),(0,o.yg)("h3",{id:"enabling-guest-access"},"Enabling Guest Access"),(0,o.yg)("p",null,"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,o.yg)("inlineCode",{parentName:"p"},"appConfig.auth.enableGuestAccess: true"),"."),(0,o.yg)("h3",{id:"granular-access"},"Granular Access"),(0,o.yg)("p",null,"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."),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"hideForUsers")," - Page, Section or Item will be visible to all users, except for those specified in this list"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"showForUsers")," - Page, Section or Item will be hidden from all users, except for those specified in this list"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"hideForGuests")," - Page, Section or Item will be visible for logged in users, but not for guests")),(0,o.yg)("p",null,"For Example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"- name: Code Analysis & Monitoring\n icon: fas fa-code\n displayData:\n cols: 2\n hideForUsers: [alicia, bob]\n items:\n ...\n")),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"- 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")),(0,o.yg)("h3",{id:"permissions"},"Permissions"),(0,o.yg)("p",null,"Any user who is not an admin (with ",(0,o.yg)("inlineCode",{parentName:"p"},"type: admin"),") will not be able to write changes to disk."),(0,o.yg)("p",null,"You can also prevent any user from writing changes to disk, using ",(0,o.yg)("inlineCode",{parentName:"p"},"preventWriteToDisk"),". Or prevent any changes from being saved locally in browser storage, using ",(0,o.yg)("inlineCode",{parentName:"p"},"preventLocalSave"),". Both properties can be found under ",(0,o.yg)("a",{parentName:"p",href:"/docs/configuring#appconfig-optional"},(0,o.yg)("inlineCode",{parentName:"a"},"appConfig")),"."),(0,o.yg)("p",null,"To disable all UI config features, including View Config, set ",(0,o.yg)("inlineCode",{parentName:"p"},"disableConfiguration"),". Alternatively you can disable UI config features for all non admin users by setting ",(0,o.yg)("inlineCode",{parentName:"p"},"disableConfigurationForNonAdmin")," to true."),(0,o.yg)("h3",{id:"using-environment-variables-for-passwords"},"Using Environment Variables for Passwords"),(0,o.yg)("p",null,"If you don't want to hash your password, you can instead leave out the ",(0,o.yg)("inlineCode",{parentName:"p"},"hash")," attribute, and replace it with ",(0,o.yg)("inlineCode",{parentName:"p"},"password")," which should have the value of an environmental variable name you wish to use."),(0,o.yg)("p",null,"Note that env var must begin with ",(0,o.yg)("inlineCode",{parentName:"p"},"VUE_APP_"),", and you must set this variable before building the app."),(0,o.yg)("p",null,"For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"}," auth:\n users:\n - user: bob\n password: VUE_APP_BOB\n")),(0,o.yg)("p",null,"Just be sure to set ",(0,o.yg)("inlineCode",{parentName:"p"},"VUE_APP_BOB='my super secret password'")," before build-time."),(0,o.yg)("h3",{id:"adding-http-auth-to-configuration"},"Adding HTTP Auth to Configuration"),(0,o.yg)("p",null,"If you'd also like to prevent direct visit access to your configuration file, you can set the ",(0,o.yg)("inlineCode",{parentName:"p"},"ENABLE_HTTP_AUTH")," environmental variable."),(0,o.yg)("h3",{id:"security"},"Security"),(0,o.yg)("p",null,"With basic auth, all logic is happening on the client-side, which could mean a skilled user could manipulate the code to view parts of your configuration, including the hash. If the SHA-256 hash is of a common password, it may be possible to determine it, using a lookup table, in order to find the original password. Which can be used to manually generate the auth token, that can then be inserted into session storage, to become a valid logged in user. Therefore, you should always use a long, strong and unique password, and if you instance contains security-critical info and/ or is exposed directly to the internet, and alternative authentication method may be better. The purpose of the login page is merely to prevent immediate unauthorized access to your homepage."),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},(0,o.yg)("a",{parentName:"strong",href:"#top"},"\u2b06\ufe0f Back to Top"))),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"http-auth"},"HTTP Auth"),(0,o.yg)("p",null,"If you'd like to protect all your config files from direct access, you can set the ",(0,o.yg)("inlineCode",{parentName:"p"},"BASIC_AUTH_USERNAME")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"BASIC_AUTH_PASSWORD")," environmental variables. You'll then be prompted to enter these credentials when visiting Dashy."),(0,o.yg)("p",null,"Then, if you'd like your frontend to automatically log you in, without prompting you for credentials, then also specify ",(0,o.yg)("inlineCode",{parentName:"p"},"VUE_APP_BASIC_AUTH_USERNAME")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"VUE_APP_BASIC_AUTH_PASSWORD"),". This is useful for when you're hosting Dashy on a private server, and you want to prevent unauthorized access to your config files, while still allowing the frontend to access them. Note that a rebuild is required for these changes to take effect."),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},(0,o.yg)("a",{parentName:"strong",href:"#top"},"\u2b06\ufe0f Back to Top"))),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"keycloak"},"Keycloak"),(0,o.yg)("p",null,"Dashy also supports using a ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/"},"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."),(0,o.yg)("p",null,(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/about.html"},"Keycloak")," is a Java-based ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/keycloak/keycloak"},"open source"),", high-performance, secure authentication system, supported by ",(0,o.yg)("a",{parentName:"p",href:"https://www.redhat.com/en"},"RedHat"),". It is easy to setup (",(0,o.yg)("a",{parentName:"p",href:"https://quay.io/repository/keycloak/keycloak"},"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,o.yg)("a",{parentName:"p",href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_development/topics/themes.html"},"themes"),", ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/extensions.html"},"plugins"),", ",(0,o.yg)("a",{parentName:"p",href:"https://wjw465150.gitbooks.io/keycloak-documentation/content/server_admin/topics/authentication/password-policies.html"},"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."),(0,o.yg)("h3",{id:"1-deploy-keycloak"},"1. Deploy Keycloak"),(0,o.yg)("p",null,"First thing to do is to spin up a new instance of Keycloak. You will need ",(0,o.yg)("a",{parentName:"p",href:"https://docs.docker.com/engine/install/"},"Docker installed"),", and can then choose a tag, and pull the container from ",(0,o.yg)("a",{parentName:"p",href:"https://quay.io/repository/keycloak/keycloak"},"quay.io/keycloak/keycloak")),(0,o.yg)("p",null,"Use the following run command, replacing the attributes (default credentials, port and name), or incorporate this into your docker-compose file."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-bash"},"docker run -d \\\n -p 8081:8080 \\\n --name auth-server \\\n -e KEYCLOAK_USER=admin \\\n -e KEYCLOAK_PASSWORD=admin \\\n quay.io/keycloak/keycloak:15.0.2\n")),(0,o.yg)("p",null,"If you need to pull from DockerHub, a non-official image is available ",(0,o.yg)("a",{parentName:"p",href:"https://registry.hub.docker.com/r/jboss/keycloak"},"here"),". Or if you would prefer not to use Docker, you can also directly install Keycloak from source, following ",(0,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/docs/latest/getting_started/index.html"},"this guide"),"."),(0,o.yg)("p",null,"You should now be able to access the Keycloak web interface, using the port specified above (e.g. ",(0,o.yg)("inlineCode",{parentName:"p"},"http://127.0.0.1:8081"),"), login with the default credentials, and when prompted create a new password."),(0,o.yg)("h3",{id:"2-setup-keycloak-users"},"2. Setup Keycloak Users"),(0,o.yg)("p",null,"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."),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"Head over to the admin console"),(0,o.yg)("li",{parentName:"ol"},"In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm'"),(0,o.yg)("li",{parentName:"ol"},"Give your realm a name, and hit 'Create'")),(0,o.yg)("p",null,"You can now create your first user."),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"In the left-hand menu, click 'Users', then 'Add User'"),(0,o.yg)("li",{parentName:"ol"},"Fill in the form, including username and hit 'Save'"),(0,o.yg)("li",{parentName:"ol"},"Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login")),(0,o.yg)("p",null,"The last thing we need to do in the Keycloak admin console is to create a new client"),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right"),(0,o.yg)("li",{parentName:"ol"},"Choose a 'Client ID', set 'Client Protocol' to 'openid-connect', and for 'Valid Redirect URIs' put a URL pattern to where you're hosting Dashy (if you're just testing locally, then * is fine), and do the same for the 'Web Origins' field"),(0,o.yg)("li",{parentName:"ol"},"Make note of your client-id, and click 'Save'")),(0,o.yg)("h3",{id:"3-enable-keycloak-in-dashy-config-file"},"3. Enable Keycloak in Dashy Config File"),(0,o.yg)("p",null,"Now that your Keycloak instance is up and running, all that's left to do is to configure Dashy to use it. Under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig"),", set ",(0,o.yg)("inlineCode",{parentName:"p"},"auth.enableKeycloak: true"),", then fill in the details in ",(0,o.yg)("inlineCode",{parentName:"p"},"auth.keycloak"),", including: ",(0,o.yg)("inlineCode",{parentName:"p"},"serverUrl")," - the URL where your Keycloak instance is hosted, ",(0,o.yg)("inlineCode",{parentName:"p"},"realm")," - the name you gave your Realm, and ",(0,o.yg)("inlineCode",{parentName:"p"},"clientId")," - the Client ID you chose.\nFor example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"appConfig:\n ...\n auth:\n enableKeycloak: true\n keycloak:\n serverUrl: 'http://localhost:8081'\n realm: 'alicia-homelab'\n clientId: 'dashy'\n")),(0,o.yg)("p",null,"Note that if you are using Keycloak V 17 or older, you will also need to set ",(0,o.yg)("inlineCode",{parentName:"p"},"legacySupport: true")," (also under ",(0,o.yg)("inlineCode",{parentName:"p"},"appConfig.auth.keycloak"),"). This is because the API endpoint was updated in later versions."),(0,o.yg)("p",null,"If you use Keycloak with an external Identity Provier, you can set the ",(0,o.yg)("inlineCode",{parentName:"p"},"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'."),(0,o.yg)("h3",{id:"4-add-groups-and-roles-optional"},"4. Add groups and roles (Optional)"),(0,o.yg)("p",null,"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,o.yg)("a",{parentName:"p",href:"https://www.keycloak.org/docs/latest/server_admin/index.html#assigning-permissions-and-access-using-roles-and-groups"},"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,o.yg)("inlineCode",{parentName:"p"},"displayData.showForKeycloakUser")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"displayData.hideForKeycloakUser"),".\nBoth show and hide configurations accept a list of ",(0,o.yg)("inlineCode",{parentName:"p"},"groups")," and ",(0,o.yg)("inlineCode",{parentName:"p"},"roles")," that limit access. If a users data matches one or more items in these lists they will be allowed or excluded as defined."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,o.yg)("p",null,"Depending on how you're hosting Dashy and Keycloak, you may also need to set some HTTP headers, to prevent a CORS error. This would typically be the ",(0,o.yg)("inlineCode",{parentName:"p"},"Access-Control-Allow-Origin [URL-of Dashy]")," on your Keycloak instance. See the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/management.md#setting-headers"},"Setting Headers")," guide in the management docs for more info."),(0,o.yg)("p",null,"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."),(0,o.yg)("p",null,"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,o.yg)("inlineCode",{parentName:"p"},"docker-compose.yml")," file, and this is recommended."),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"alternative-authentication-methods"},"Alternative Authentication Methods"),(0,o.yg)("p",null,"If you are self-hosting Dashy, and require secure authentication to prevent unauthorized access, then you can either use Keycloak, or one of the following options:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#authentication-server"},"Authentication Server")," - Put Dashy behind a self-hosted auth server"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#vpn"},"VPN")," - Use a VPN to tunnel into the network where Dashy is running"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#ip-based-access"},"IP-Based Access")," - Disallow access from all IP addresses, except your own"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#web-server-authentication"},"Web Server Authentication")," - Enable user control within your web server or proxy"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#oauth-services"},"OAuth Services")," - Implement a user management system using a cloud provider"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"#static-site-hosting-providers"},"Password Protection (for cloud providers)")," - Enable password-protection on your site")),(0,o.yg)("h3",{id:"authentication-server"},"Authentication Server"),(0,o.yg)("h4",{id:"authelia"},"Authelia"),(0,o.yg)("p",null,(0,o.yg)("a",{parentName:"p",href:"https://www.authelia.com/"},"Authelia")," is an open-source full-featured authentication server, which can be self-hosted and either on bare metal, in a Docker container or in a Kubernetes cluster. It allows for fine-grained access control rules based on IP, path, users etc, and supports 2FA, simple password access or bypass policies for your domains."),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"git clone https://github.com/authelia/authelia.git")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"cd authelia/examples/compose/lite")),(0,o.yg)("li",{parentName:"ul"},"Modify the ",(0,o.yg)("inlineCode",{parentName:"li"},"users_database.yml")," the default username and password is authelia"),(0,o.yg)("li",{parentName:"ul"},"Modify the ",(0,o.yg)("inlineCode",{parentName:"li"},"configuration.yml")," and ",(0,o.yg)("inlineCode",{parentName:"li"},"docker-compose.yml")," with your respective domains and secrets"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("inlineCode",{parentName:"li"},"docker-compose up -d"))),(0,o.yg)("p",null,"For more information, see the ",(0,o.yg)("a",{parentName:"p",href:"https://www.authelia.com/docs/"},"Authelia docs")),(0,o.yg)("h3",{id:"vpn"},"VPN"),(0,o.yg)("p",null,"A catch-all solution to accessing services running from your home network remotely is to use a VPN. It means you do not need to worry about implementing complex authentication rules, or trusting the login implementation of individual applications. However it can be inconvenient to use on a day-to-day basis, and some public and corporate WiFi block VPN connections. Two popular VPN protocols are ",(0,o.yg)("a",{parentName:"p",href:"https://openvpn.net/"},"OpenVPN")," and ",(0,o.yg)("a",{parentName:"p",href:"https://www.wireguard.com/"},"WireGuard")),(0,o.yg)("h3",{id:"ip-based-access"},"IP-Based Access"),(0,o.yg)("p",null,"If you have a static IP or use a VPN to access your running services, then you can use conditional access to block access to Dashy from everyone except users of your pre-defined IP address. This feature is offered by most cloud providers, and supported by most web servers."),(0,o.yg)("h4",{id:"apache"},"Apache"),(0,o.yg)("p",null,"In Apache, this is configured in your ",(0,o.yg)("inlineCode",{parentName:"p"},".htaccess")," file in Dashy's root folder, and should look something like:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"Order Deny,Allow\nDeny from all\nAllow from [your-ip]\n")),(0,o.yg)("h4",{id:"nginx"},"NGINX"),(0,o.yg)("p",null,"In NGINX you can specify ",(0,o.yg)("a",{parentName:"p",href:"https://docs.nginx.com/nginx/admin-guide/security-controls/controlling-access-proxied-http/"},"control access")," rules for a given site in your ",(0,o.yg)("inlineCode",{parentName:"p"},"nginx.conf")," or hosts file. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"server {\n listen 8080;\n server_name www.dashy.example.com;\n location / {\n root /path/to/dashy/;\n passenger_enabled on;\n allow [your-ip];\n deny all;\n }\n }\n")),(0,o.yg)("h4",{id:"caddy"},"Caddy"),(0,o.yg)("p",null,"In Caddy, ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/caddyfile/matchers"},"Request Matchers")," can be used to filter requests"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'dashy.site {\n @public_networks not remote_ip [your-ip]\n respond @public_networks "Access denied" 403\n}\n')),(0,o.yg)("h3",{id:"web-server-authentication"},"Web Server Authentication"),(0,o.yg)("p",null,"Most web servers make password protecting certain apps very easy. Note that you should also set up HTTPS and have a valid certificate in order for this to be secure."),(0,o.yg)("h4",{id:"apache-1"},"Apache"),(0,o.yg)("p",null,"First crate a ",(0,o.yg)("inlineCode",{parentName:"p"},".htaccess")," file in Dashy's route directory. Specify the auth type and path to where you want to store the password file (usually the same folder). For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'AuthType Basic\nAuthName "Please Sign into Dashy"\nAuthUserFile /path/dashy/.htpasswd\nrequire valid-user\n')),(0,o.yg)("p",null,"Then create a ",(0,o.yg)("inlineCode",{parentName:"p"},".htpasswd")," file in the same directory. List users and their hashed passwords here, with one user on each line, and a colon between username and password (e.g. ",(0,o.yg)("inlineCode",{parentName:"p"},"[username]:[hashed-password]"),"). You will need to generate an MD5 hash of your desired password, this can be done with an ",(0,o.yg)("a",{parentName:"p",href:"https://www.web2generators.com/apache-tools/htpasswd-generator"},"online tool"),". Your file will look something like:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"alicia:$apr1$jv0spemw$RzOX5/GgY69JMkgV6u16l0\n")),(0,o.yg)("h4",{id:"nginx-1"},"NGINX"),(0,o.yg)("p",null,"NGINX has an ",(0,o.yg)("a",{parentName:"p",href:"https://nginx.org/en/docs/http/ngx_http_auth_basic_module.html"},"authentication module")," which can be used to add passwords to given sites, and is fairly simple to set up. Similar to above, you will need to create a ",(0,o.yg)("inlineCode",{parentName:"p"},".htpasswd")," file. Then just enable auth and specify the path to that file, for example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'location / {\n auth_basic "closed site";\n auth_basic_user_file conf/htpasswd;\n}\n')),(0,o.yg)("h4",{id:"caddy-1"},"Caddy"),(0,o.yg)("p",null,"Caddy has a ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/caddyfile/directives/basicauth"},"basic-auth")," directive, where you specify a username and hash. The password hash needs to be base-64 encoded, the ",(0,o.yg)("a",{parentName:"p",href:"https://caddyserver.com/docs/command-line#caddy-hash-password"},(0,o.yg)("inlineCode",{parentName:"a"},"caddy hash-password"))," command can help with this. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},"basicauth /secret/* {\n alicia JDJhJDEwJEVCNmdaNEg2Ti5iejRMYkF3MFZhZ3VtV3E1SzBWZEZ5Q3VWc0tzOEJwZE9TaFlZdEVkZDhX\n}\n")),(0,o.yg)("p",null,"For more info about implementing a single sign on for all your apps with Caddy, see ",(0,o.yg)("a",{parentName:"p",href:"https://joshstrange.com/securing-your-self-hosted-apps-with-single-signon/"},"this tutorial")),(0,o.yg)("h4",{id:"lighttpd"},"Lighttpd"),(0,o.yg)("p",null,"You can use the ",(0,o.yg)("a",{parentName:"p",href:"https://doc.lighttpd.net/lighttpd2/mod_auth.html"},"mod_auth")," module to secure your site with Lighttpd. Like with Apache, you need to first create a password file listing your usernames and hashed passwords, but in Lighttpd, it's usually called ",(0,o.yg)("inlineCode",{parentName:"p"},".lighttpdpassword"),"."),(0,o.yg)("p",null,"Then in your ",(0,o.yg)("inlineCode",{parentName:"p"},"lighttpd.conf")," file (usually in the ",(0,o.yg)("inlineCode",{parentName:"p"},"/etc/lighttpd/")," directory), load in the mod_auth module, and configure it's directives. For example:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-text"},'server.modules += ( "mod_auth" )\nauth.debug = 2\nauth.backend = "plain"\nauth.backend.plain.userfile = "/home/lighttpd/.lighttpdpassword"\n\n$HTTP["host"] == "dashy.my-domain.net" {\n server.document-root = "/home/lighttpd/dashy.my-domain.net/http"\n server.errorlog = "/var/log/lighttpd/dashy.my-domain.net/error.log"\n accesslog.filename = "/var/log/lighttpd/dashy.my-domain.net/access.log"\n auth.require = (\n "/docs/" => (\n "method" => "basic",\n "realm" => "Password protected area",\n "require" => "user=alicia"\n )\n )\n}\n')),(0,o.yg)("p",null,"Restart your web server for changes to take effect."),(0,o.yg)("h3",{id:"oauth-services"},"OAuth Services"),(0,o.yg)("p",null,"There are also authentication services, such as ",(0,o.yg)("a",{parentName:"p",href:"https://www.ory.sh/"},"Ory.sh"),", ",(0,o.yg)("a",{parentName:"p",href:"https://developer.okta.com/"},"Okta"),", ",(0,o.yg)("a",{parentName:"p",href:"https://auth0.com/"},"Auth0"),", ",(0,o.yg)("a",{parentName:"p",href:"https://firebase.google.com/docs/auth/"},"Firebase"),". Implementing one of these solutions would involve some changes to the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/src/utils/Auth.js"},(0,o.yg)("inlineCode",{parentName:"a"},"Auth.js"))," file, but should be fairly straightforward."),(0,o.yg)("h3",{id:"static-site-hosting-providers"},"Static Site Hosting Providers"),(0,o.yg)("p",null,"If you are hosting Dashy on a cloud platform, you will probably find that it has built-in support for password protected access to web apps. For more info, see the relevant docs for your provider, for example: ",(0,o.yg)("a",{parentName:"p",href:"https://docs.netlify.com/visitor-access/password-protection/"},"Netlify Password Protection"),", ",(0,o.yg)("a",{parentName:"p",href:"https://www.cloudflare.com/teams/access/"},"Cloudflare Access"),", ",(0,o.yg)("a",{parentName:"p",href:"https://aws.amazon.com/cognito/"},"AWS Cognito"),", ",(0,o.yg)("a",{parentName:"p",href:"https://docs.microsoft.com/en-us/azure/app-service/scenario-secure-app-authentication-app-service"},"Azure Authentication")," and ",(0,o.yg)("a",{parentName:"p",href:"https://vercel.com/docs/platform/projects#password-protection"},"Vercel Password Protection"),"."),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},(0,o.yg)("a",{parentName:"strong",href:"#top"},"\u2b06\ufe0f Back to Top"))))}c.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/72e14192.4ebcc2d8.js b/assets/js/72e14192.4ebcc2d8.js new file mode 100644 index 00000000..d7eee153 --- /dev/null +++ b/assets/js/72e14192.4ebcc2d8.js @@ -0,0 +1 @@ +"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[814],{5680:(e,t,a)=>{a.d(t,{xA:()=>y,yg:()=>d});var n=a(6540);function i(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function o(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function r(e){for(var t=1;t=0||(i[a]=e[a]);return i}(e,t);if(Object.getOwnPropertySymbols){var o=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(i[a]=e[a])}return i}var s=n.createContext({}),p=function(e){var t=n.useContext(s),a=t;return e&&(a="function"==typeof e?e(t):r(r({},t),e)),a},y=function(e){var t=p(e.components);return n.createElement(s.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},h=n.forwardRef((function(e,t){var a=e.components,i=e.mdxType,o=e.originalType,s=e.parentName,y=l(e,["components","mdxType","originalType","parentName"]),u=p(a),h=i,d=u["".concat(s,".").concat(h)]||u[h]||c[h]||o;return a?n.createElement(d,r(r({ref:t},y),{},{components:a})):n.createElement(d,r({ref:t},y))}));function d(e,t){var a=arguments,i=t&&t.mdxType;if("string"==typeof e||i){var o=a.length,r=new Array(o);r[0]=h;var l={};for(var s in t)hasOwnProperty.call(t,s)&&(l[s]=t[s]);l.originalType=e,l[u]="string"==typeof e?e:i,r[1]=l;for(var p=2;p{a.r(t),a.d(t,{contentTitle:()=>r,default:()=>u,frontMatter:()=>o,metadata:()=>l,toc:()=>s});var n=a(8168),i=(a(6540),a(5680));const o={},r="Quick Start",l={unversionedId:"quick-start",id:"quick-start",isDocsHomePage:!1,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",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/quick-start.md",version:"current",frontMatter:{},sidebar:"dashySidebar",next:{title:"Deployment",permalink:"/docs/deployment"}},s=[{value:"1. Prerequisites",id:"1-prerequisites",children:[]},{value:"2. Installation",id:"2-installation",children:[]},{value:"3. User Data Directory",id:"3-user-data-directory",children:[]},{value:"4. Configure",id:"4-configure",children:[]},{value:"5. Further Customisation",id:"5-further-customisation",children:[]},{value:"6. Final Note",id:"6-final-note",children:[]},{value:"Alternative Deployment Method 1 - From Source",id:"alternative-deployment-method-1---from-source",children:[]},{value:"Alternative Deployment Method 2 - Netlify",id:"alternative-deployment-method-2---netlify",children:[]},{value:"Alternative Deployment Method 3 - Cloud Services",id:"alternative-deployment-method-3---cloud-services",children:[]}],p={toc:s},y="wrapper";function u(e){let{components:t,...a}=e;return(0,i.yg)(y,(0,n.A)({},p,a,{components:t,mdxType:"MDXLayout"}),(0,i.yg)("h1",{id:"quick-start"},"Quick Start"),(0,i.yg)("p",null,"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"),(0,i.yg)("p",null,(0,i.yg)("strong",{parentName:"p"},"TLDR;")," Run ",(0,i.yg)("inlineCode",{parentName:"p"},"docker run -p 8080:8080 lissy93/dashy"),", then open ",(0,i.yg)("inlineCode",{parentName:"p"},"http://localhost:8080")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"1-prerequisites"},"1. Prerequisites"),(0,i.yg)("p",null,"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,i.yg)("a",{parentName:"p",href:"https://docs.docker.com/get-docker/"},"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,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md"},"Deployment Docs"),"."),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"2-installation"},"2. Installation"),(0,i.yg)("p",null,"To pull the latest image, and build and start the app run:"),(0,i.yg)("pre",null,(0,i.yg)("code",{parentName:"pre",className:"language-bash"},"docker run -d \\\n -p 8080:8080 \\\n -v ~/my-conf.yml:/app/user-data/conf.yml \\\n --name my-dashboard \\\n --restart=always \\\n lissy93/dashy:latest\n")),(0,i.yg)("p",null,"Either replace the -v path to point to your config file, or leave it out. For a full list of available options, then see ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md#deploy-with-docker"},"Dashy with Docker")," Docs. If you'd prefer to use Docker Compose, then see ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md#using-docker-compose"},"Dashy with Docker Compose")," Docs. Alternate registries, architectures and pinned versions are also supported."),(0,i.yg)("p",null,"Your dashboard should now be up and running at ",(0,i.yg)("inlineCode",{parentName:"p"},"http://localhost:8080")," (or your servers IP address/ domain, and the port that you chose). The first time you build, it may take a few minutes."),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"3-user-data-directory"},"3. User Data Directory"),(0,i.yg)("p",null,"Your config file should be placed inside ",(0,i.yg)("inlineCode",{parentName:"p"},"user-data/")," (in Docker, that's ",(0,i.yg)("inlineCode",{parentName:"p"},"/app/user-data/"),")."),(0,i.yg)("p",null,"This directory can also contain some optional assets you wish to use within your dashboard, like icons, fonts, styles, scripts, etc."),(0,i.yg)("p",null,"Any files placed here will be served up to the root of the domain, and override the contents of ",(0,i.yg)("inlineCode",{parentName:"p"},"public/"),".\nFor example, if you had ",(0,i.yg)("inlineCode",{parentName:"p"},"user-data/favicon.ico")," this would be accessible at ",(0,i.yg)("inlineCode",{parentName:"p"},"http://my-dashy-instance.local/favicon.ico")),(0,i.yg)("p",null,"Example Files in ",(0,i.yg)("inlineCode",{parentName:"p"},"user-data"),":"),(0,i.yg)("ul",null,(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"conf.yml")," - This is the only file that is compulsary, it's your main Dashy config"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"**.yml")," - Include more config files, if you'd like to have multiple pages, see ",(0,i.yg)("a",{parentName:"li",href:"/docs/pages-and-sections#multi-page-support"},"Multi-page support")," for docs"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"favicon.ico")," - The default favicon, shown in the browser's tab title"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"initialization.html")," - Static HTML page displayed before the app has finished compiling, see ",(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/public/initialization.html"},(0,i.yg)("inlineCode",{parentName:"a"},"public/initialization.html"))),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"robots.txt")," - Search engine crawl rules, override this if you want your dashboard to be indexable"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"manifest.json")," - PWA configuration file, for installing Dashy on mobile devices"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"index.html")," - The main index page which initializes the client-side app, copy it from ",(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/public/index.html"},(0,i.yg)("inlineCode",{parentName:"a"},"/public/index.html"))),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"**.html")," - Write your own HTML pages, and access them at ",(0,i.yg)("inlineCode",{parentName:"li"},"http://my-dashy-instance.local/my-page.html")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"fonts/")," - Custom fonts (be sure to include the ones already in ",(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/tree/master/public/fonts"},(0,i.yg)("inlineCode",{parentName:"a"},"public/fonts"))),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"item-icons/")," - To use your own icons for items on your dashboard, see ",(0,i.yg)("a",{parentName:"li",href:"/docs/icons#local-icons"},"Icons --\x3e Local Icons")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"web-icons/")," - Override Dashy logo"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("inlineCode",{parentName:"li"},"widget-resources/")," - Fonts, icons and assets for custom widgets")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"4-configure"},"4. Configure"),(0,i.yg)("p",null,"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,i.yg)("a",{parentName:"p",href:"https://yaml.org/"},"YAML Format"),", and saved in ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml"},(0,i.yg)("inlineCode",{parentName:"a"},"/user-data/conf.yml")),".\nThe format on the config file is pretty straight forward. There are three root attributes:"),(0,i.yg)("ul",null,(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#pageinfo"},(0,i.yg)("inlineCode",{parentName:"a"},"pageInfo"))," - Dashboard meta data, like title, description, nav bar links and footer text"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#appconfig-optional"},(0,i.yg)("inlineCode",{parentName:"a"},"appConfig"))," - Dashboard settings, like themes, authentication, language and customization"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#section"},(0,i.yg)("inlineCode",{parentName:"a"},"sections"))," - An array of sections, each including an array of items"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#pages-optional"},(0,i.yg)("inlineCode",{parentName:"a"},"pages"))," - Have multiples pages in your dashboard")),(0,i.yg)("p",null,"You can view a full list of all available config options in the ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md"},"Configuring Docs"),"."),(0,i.yg)("pre",null,(0,i.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,i.yg)("p",null,"Notes:"),(0,i.yg)("ul",null,(0,i.yg)("li",{parentName:"ul"},"You can use a Docker volume to pass a config file from your host system to the container",(0,i.yg)("ul",{parentName:"li"},(0,i.yg)("li",{parentName:"ul"},"E.g. ",(0,i.yg)("inlineCode",{parentName:"li"},"-v ./host-system/my-local-conf.yml:/app/user-data/conf.yml")))),(0,i.yg)("li",{parentName:"ul"},"It's also possible to edit your config directly through the UI, and changes will be saved in this file"),(0,i.yg)("li",{parentName:"ul"},"Check your config against Dashy's schema, with ",(0,i.yg)("inlineCode",{parentName:"li"},"docker exec -it [container-id] yarn validate-config")),(0,i.yg)("li",{parentName:"ul"},"You might find it helpful to look at some examples, a collection of which can be ",(0,i.yg)("a",{parentName:"li",href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10"},"found here")),(0,i.yg)("li",{parentName:"ul"},"It's also possible to load a remote config, e.g. from a GitHub Gist")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"5-further-customisation"},"5. Further Customisation"),(0,i.yg)("p",null,"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,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/management.md"},"Management Docs"),"."),(0,i.yg)("p",null,"You might also want to check out the docs for specific features you'd like to use:"),(0,i.yg)("ul",null,(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/authentication"},"Authentication")," - Setting up authentication to protect your dashboard"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/alternate-views"},"Alternate Views")," - Using the startpage and workspace view"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/backup-restore"},"Backup & Restore")," - Guide to Dashy's cloud sync feature"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/icons"},"Icons")," - Outline of all available icon types for sections and items"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/multi-language-support"},"Localisation")," - How to change language, or add your own"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/status-indicators"},"Status Indicators")," - Using Dashy to monitor uptime and status of your apps"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/searching"},"Search & Shortcuts")," - Using instant filter, web search and custom hotkeys"),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"/docs/theming"},"Theming")," - Complete guide to applying, writing and modifying themes and styles")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"6-final-note"},"6. Final Note"),(0,i.yg)("p",null,"If you need any help or support in getting Dashy running, head over to the ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/discussions"},"Discussions")," page. If you think you've found a bug, please do ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/issues/new/choose"},"raise it")," so it can be fixed. For contact options, see the ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/.github/SUPPORT.md"},"Support Page"),"."),(0,i.yg)("p",null,"If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/contributing.md"},"Contributing Page"),". Huge thanks to ",(0,i.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/credits.md"},"everyone")," who has already helped out!"),(0,i.yg)("p",null,"Enjoy your dashboard :)"),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"alternative-deployment-method-1---from-source"},"Alternative Deployment Method 1 - From Source"),(0,i.yg)("p",null,"You can also easily run the app on your system without Docker. For this ",(0,i.yg)("a",{parentName:"p",href:"https://git-scm.com/downloads"},"Git"),", ",(0,i.yg)("a",{parentName:"p",href:"https://nodejs.org/"},"Node.js"),", and ",(0,i.yg)("a",{parentName:"p",href:"https://yarnpkg.com/"},"Yarn")," are required."),(0,i.yg)("pre",null,(0,i.yg)("code",{parentName:"pre",className:"language-bash"},"git clone https://github.com/Lissy93/dashy.git && cd dashy\nyarn # Install dependencies\nyarn build # Build the app\nyarn start # Start the app\n")),(0,i.yg)("p",null,"Then edit ",(0,i.yg)("inlineCode",{parentName:"p"},"./user-data/conf.yml")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"alternative-deployment-method-2---netlify"},"Alternative Deployment Method 2 - Netlify"),(0,i.yg)("p",null,"Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many ",(0,i.yg)("a",{parentName:"p",href:"/docs/deployment#deploy-to-cloud-service"},"other cloud providers"),"). All you need it a GitHub account."),(0,i.yg)("ol",null,(0,i.yg)("li",{parentName:"ol"},"Fork Dashy's repository on GitHub"),(0,i.yg)("li",{parentName:"ol"},(0,i.yg)("a",{parentName:"li",href:"app.netlify.com/login/"},"Log in")," to Netlify with GitHub"),(0,i.yg)("li",{parentName:"ol"},'Click "New site from Git" and select your forked repo, then click ',(0,i.yg)("strong",{parentName:"li"},"Deploy"),"!"),(0,i.yg)("li",{parentName:"ol"},"You can then edit the config in ",(0,i.yg)("inlineCode",{parentName:"li"},"./user-data/conf.yml")," in your repo, and Netlify will rebuild the app")),(0,i.yg)("hr",null),(0,i.yg)("h2",{id:"alternative-deployment-method-3---cloud-services"},"Alternative Deployment Method 3 - Cloud Services"),(0,i.yg)("p",null,"Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:"),(0,i.yg)("ul",null,(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy"},(0,i.yg)("img",{src:"https://i.ibb.co/ZxtzrP3/netlify.png",width:"18"})," Deploy to Netlify")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://heroku.com/deploy?template=https://github.com/Lissy93/dashy"},(0,i.yg)("img",{src:"https://i.ibb.co/d2P1WZ7/heroku.png",width:"18"})," Deploy to Heroku")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy"},(0,i.yg)("img",{src:"https://i.ibb.co/Ld2FZzb/vercel.png",width:"18"})," Deploy to Vercel")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://render.com/deploy?repo=https://github.com/lissy93/dashy/tree/deploy_render"},(0,i.yg)("img",{src:"https://i.ibb.co/xCHtzgh/render.png",width:"18"})," Deploy to Render")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://deploy.cloud.run/?git_repo=https://github.com/lissy93/dashy.git"},(0,i.yg)("img",{src:"https://i.ibb.co/J7MGymY/googlecloud.png",width:"18"})," Deploy to GCP")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml"},(0,i.yg)("img",{src:"https://i.ibb.co/HVWVYF7/docker.png",width:"18"})," Deploy to PWD")),(0,i.yg)("li",{parentName:"ul"},(0,i.yg)("a",{parentName:"li",href:"https://easypanel.io/docs/templates/dashy"},(0,i.yg)("img",{src:"https://i.ibb.co/7NxnM2P/easypanel.png",width:"18"})," Deploy to Easypanel"))))}u.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/72e14192.ba227462.js b/assets/js/72e14192.ba227462.js deleted file mode 100644 index b35bc50a..00000000 --- a/assets/js/72e14192.ba227462.js +++ /dev/null @@ -1 +0,0 @@ -"use strict";(self.webpackChunkdashy=self.webpackChunkdashy||[]).push([[814],{5680:(e,t,a)=>{a.d(t,{xA:()=>y,yg:()=>d});var n=a(6540);function o(e,t,a){return t in e?Object.defineProperty(e,t,{value:a,enumerable:!0,configurable:!0,writable:!0}):e[t]=a,e}function r(e,t){var a=Object.keys(e);if(Object.getOwnPropertySymbols){var n=Object.getOwnPropertySymbols(e);t&&(n=n.filter((function(t){return Object.getOwnPropertyDescriptor(e,t).enumerable}))),a.push.apply(a,n)}return a}function i(e){for(var t=1;t=0||(o[a]=e[a]);return o}(e,t);if(Object.getOwnPropertySymbols){var r=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(o[a]=e[a])}return o}var l=n.createContext({}),p=function(e){var t=n.useContext(l),a=t;return e&&(a="function"==typeof e?e(t):i(i({},t),e)),a},y=function(e){var t=p(e.components);return n.createElement(l.Provider,{value:t},e.children)},u="mdxType",c={inlineCode:"code",wrapper:function(e){var t=e.children;return n.createElement(n.Fragment,{},t)}},h=n.forwardRef((function(e,t){var a=e.components,o=e.mdxType,r=e.originalType,l=e.parentName,y=s(e,["components","mdxType","originalType","parentName"]),u=p(a),h=o,d=u["".concat(l,".").concat(h)]||u[h]||c[h]||r;return a?n.createElement(d,i(i({ref:t},y),{},{components:a})):n.createElement(d,i({ref:t},y))}));function d(e,t){var a=arguments,o=t&&t.mdxType;if("string"==typeof e||o){var r=a.length,i=new Array(r);i[0]=h;var s={};for(var l in t)hasOwnProperty.call(t,l)&&(s[l]=t[l]);s.originalType=e,s[u]="string"==typeof e?e:o,i[1]=s;for(var p=2;p{a.r(t),a.d(t,{contentTitle:()=>i,default:()=>u,frontMatter:()=>r,metadata:()=>s,toc:()=>l});var n=a(8168),o=(a(6540),a(5680));const r={},i="Quick Start",s={unversionedId:"quick-start",id:"quick-start",isDocsHomePage:!1,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",editUrl:"https://github.com/Lissy93/dashy/edit/gh-pages/docs/docs/quick-start.md",version:"current",frontMatter:{},sidebar:"dashySidebar",next:{title:"Deployment",permalink:"/docs/deployment"}},l=[{value:"1. Prerequisites",id:"1-prerequisites",children:[]},{value:"2. Installation",id:"2-installation",children:[]},{value:"3. Configure",id:"3-configure",children:[]},{value:"4. Further Customisation",id:"4-further-customisation",children:[]},{value:"5. Final Note",id:"5-final-note",children:[]},{value:"Alternative Deployment Method 1 - From Source",id:"alternative-deployment-method-1---from-source",children:[]},{value:"Alternative Deployment Method 2 - Netlify",id:"alternative-deployment-method-2---netlify",children:[]},{value:"Alternative Deployment Method 3 - Cloud Services",id:"alternative-deployment-method-3---cloud-services",children:[]}],p={toc:l},y="wrapper";function u(e){let{components:t,...a}=e;return(0,o.yg)(y,(0,n.A)({},p,a,{components:t,mdxType:"MDXLayout"}),(0,o.yg)("h1",{id:"quick-start"},"Quick Start"),(0,o.yg)("p",null,"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"),(0,o.yg)("p",null,(0,o.yg)("strong",{parentName:"p"},"TLDR;")," Run ",(0,o.yg)("inlineCode",{parentName:"p"},"docker run -p 8080:8080 lissy93/dashy"),", then open ",(0,o.yg)("inlineCode",{parentName:"p"},"http://localhost:8080")),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"1-prerequisites"},"1. Prerequisites"),(0,o.yg)("p",null,"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,o.yg)("a",{parentName:"p",href:"https://docs.docker.com/get-docker/"},"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,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md"},"Deployment Docs"),"."),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"2-installation"},"2. Installation"),(0,o.yg)("p",null,"To pull the latest image, and build and start the app run:"),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-bash"},"docker run -d \\\n -p 8080:8080 \\\n -v ~/my-conf.yml:/app/user-data/conf.yml \\\n --name my-dashboard \\\n --restart=always \\\n lissy93/dashy:latest\n")),(0,o.yg)("p",null,"Either replace the -v path to point to your config file, or leave it out. For a full list of available options, then see ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md#deploy-with-docker"},"Dashy with Docker")," Docs. If you'd prefer to use Docker Compose, then see ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/deployment.md#using-docker-compose"},"Dashy with Docker Compose")," Docs. Alternate registries, architectures and pinned versions are also supported."),(0,o.yg)("p",null,"Your dashboard should now be up and running at ",(0,o.yg)("inlineCode",{parentName:"p"},"http://localhost:8080")," (or your servers IP address/ domain, and the port that you chose). The first time you build, it may take a few minutes."),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"3-configure"},"3. Configure"),(0,o.yg)("p",null,"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,o.yg)("a",{parentName:"p",href:"https://yaml.org/"},"YAML Format"),", and saved in ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/user-data/conf.yml"},(0,o.yg)("inlineCode",{parentName:"a"},"/user-data/conf.yml")),".\nThe format on the config file is pretty straight forward. There are three root attributes:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#pageinfo"},(0,o.yg)("inlineCode",{parentName:"a"},"pageInfo"))," - Dashboard meta data, like title, description, nav bar links and footer text"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#appconfig-optional"},(0,o.yg)("inlineCode",{parentName:"a"},"appConfig"))," - Dashboard settings, like themes, authentication, language and customization"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md#section"},(0,o.yg)("inlineCode",{parentName:"a"},"sections"))," - An array of sections, each including an array of items")),(0,o.yg)("p",null,"You can view a full list of all available config options in the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/configuring.md"},"Configuring Docs"),"."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-yaml"},"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")),(0,o.yg)("p",null,"Notes:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},"You can use a Docker volume to pass a config file from your host system to the container",(0,o.yg)("ul",{parentName:"li"},(0,o.yg)("li",{parentName:"ul"},"E.g. ",(0,o.yg)("inlineCode",{parentName:"li"},"-v ./host-system/my-local-conf.yml:/app/user-data/conf.yml")))),(0,o.yg)("li",{parentName:"ul"},"It's also possible to edit your config directly through the UI, and changes will be saved in this file"),(0,o.yg)("li",{parentName:"ul"},"Check your config against Dashy's schema, with ",(0,o.yg)("inlineCode",{parentName:"li"},"docker exec -it [container-id] yarn validate-config")),(0,o.yg)("li",{parentName:"ul"},"You might find it helpful to look at some examples, a collection of which can be ",(0,o.yg)("a",{parentName:"li",href:"https://gist.github.com/Lissy93/000f712a5ce98f212817d20bc16bab10"},"found here")),(0,o.yg)("li",{parentName:"ul"},"After editing your config, the app will rebuild in the background, which may take a minute")),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"4-further-customisation"},"4. Further Customisation"),(0,o.yg)("p",null,"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,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/management.md"},"Management Docs"),"."),(0,o.yg)("p",null,"You might also want to check out the docs for specific features you'd like to use:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/authentication"},"Authentication")," - Setting up authentication to protect your dashboard"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/alternate-views"},"Alternate Views")," - Using the startpage and workspace view"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/backup-restore"},"Backup & Restore")," - Guide to Dashy's cloud sync feature"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/icons"},"Icons")," - Outline of all available icon types for sections and items"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/multi-language-support"},"Localisation")," - How to change language, or add your own"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/status-indicators"},"Status Indicators")," - Using Dashy to monitor uptime and status of your apps"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/searching"},"Search & Shortcuts")," - Using instant filter, web search and custom hotkeys"),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"/docs/theming"},"Theming")," - Complete guide to applying, writing and modifying themes and styles")),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"5-final-note"},"5. Final Note"),(0,o.yg)("p",null,"If you need any help or support in getting Dashy running, head over to the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/discussions"},"Discussions")," page. If you think you've found a bug, please do ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/issues/new/choose"},"raise it")," so it can be fixed. For contact options, see the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/.github/SUPPORT.md"},"Support Page"),"."),(0,o.yg)("p",null,"If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/contributing.md"},"Contributing Page"),". Huge thanks to ",(0,o.yg)("a",{parentName:"p",href:"https://github.com/Lissy93/dashy/blob/master/docs/credits.md"},"everyone")," who has already helped out!"),(0,o.yg)("p",null,"Enjoy your dashboard :)"),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"alternative-deployment-method-1---from-source"},"Alternative Deployment Method 1 - From Source"),(0,o.yg)("p",null,"You can also easily run the app on your system without Docker. For this ",(0,o.yg)("a",{parentName:"p",href:"https://git-scm.com/downloads"},"Git"),", ",(0,o.yg)("a",{parentName:"p",href:"https://nodejs.org/"},"Node.js"),", and ",(0,o.yg)("a",{parentName:"p",href:"https://yarnpkg.com/"},"Yarn")," are required."),(0,o.yg)("pre",null,(0,o.yg)("code",{parentName:"pre",className:"language-bash"},"git clone https://github.com/Lissy93/dashy.git && cd dashy\nyarn # Install dependencies\nyarn build # Build the app\nyarn start # Start the app\n")),(0,o.yg)("p",null,"Then edit ",(0,o.yg)("inlineCode",{parentName:"p"},"./user-data/conf.yml")," and rebuild the app with ",(0,o.yg)("inlineCode",{parentName:"p"},"yarn build")),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"alternative-deployment-method-2---netlify"},"Alternative Deployment Method 2 - Netlify"),(0,o.yg)("p",null,"Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many ",(0,o.yg)("a",{parentName:"p",href:"/docs/deployment#deploy-to-cloud-service"},"other cloud providers"),"). All you need it a GitHub account."),(0,o.yg)("ol",null,(0,o.yg)("li",{parentName:"ol"},"Fork Dashy's repository on GitHub"),(0,o.yg)("li",{parentName:"ol"},(0,o.yg)("a",{parentName:"li",href:"app.netlify.com/login/"},"Log in")," to Netlify with GitHub"),(0,o.yg)("li",{parentName:"ol"},'Click "New site from Git" and select your forked repo, then click ',(0,o.yg)("strong",{parentName:"li"},"Deploy"),"!"),(0,o.yg)("li",{parentName:"ol"},"You can then edit the config in ",(0,o.yg)("inlineCode",{parentName:"li"},"./user-data/conf.yml")," in your repo, and Netlify will rebuild the app")),(0,o.yg)("hr",null),(0,o.yg)("h2",{id:"alternative-deployment-method-3---cloud-services"},"Alternative Deployment Method 3 - Cloud Services"),(0,o.yg)("p",null,"Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:"),(0,o.yg)("ul",null,(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy"},(0,o.yg)("img",{src:"https://i.ibb.co/ZxtzrP3/netlify.png",width:"18"})," Deploy to Netlify")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://heroku.com/deploy?template=https://github.com/Lissy93/dashy"},(0,o.yg)("img",{src:"https://i.ibb.co/d2P1WZ7/heroku.png",width:"18"})," Deploy to Heroku")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://vercel.com/new/project?template=https://github.com/lissy93/dashy"},(0,o.yg)("img",{src:"https://i.ibb.co/Ld2FZzb/vercel.png",width:"18"})," Deploy to Vercel")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://render.com/deploy?repo=https://github.com/lissy93/dashy/tree/deploy_render"},(0,o.yg)("img",{src:"https://i.ibb.co/xCHtzgh/render.png",width:"18"})," Deploy to Render")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://deploy.cloud.run/?git_repo=https://github.com/lissy93/dashy.git"},(0,o.yg)("img",{src:"https://i.ibb.co/J7MGymY/googlecloud.png",width:"18"})," Deploy to GCP")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml"},(0,o.yg)("img",{src:"https://i.ibb.co/HVWVYF7/docker.png",width:"18"})," Deploy to PWD")),(0,o.yg)("li",{parentName:"ul"},(0,o.yg)("a",{parentName:"li",href:"https://easypanel.io/docs/templates/dashy"},(0,o.yg)("img",{src:"https://i.ibb.co/7NxnM2P/easypanel.png",width:"18"})," Deploy to Easypanel"))))}u.isMDXComponent=!0}}]); \ No newline at end of file diff --git a/assets/js/runtime~main.83b3f325.js b/assets/js/runtime~main.9dc14f8e.js similarity index 96% rename from assets/js/runtime~main.83b3f325.js rename to assets/js/runtime~main.9dc14f8e.js index a8cea3fb..d1a59e23 100644 --- a/assets/js/runtime~main.83b3f325.js +++ b/assets/js/runtime~main.9dc14f8e.js @@ -1 +1 @@ -(()=>{"use strict";var e,t,r,a,f,d={},o={};function b(e){var t=o[e];if(void 0!==t)return t.exports;var r=o[e]={id:e,loaded:!1,exports:{}};return d[e].call(r.exports,r,r.exports,b),r.loaded=!0,r.exports}b.m=d,b.c=o,e=[],b.O=(t,r,a,f)=>{if(!r){var d=1/0;for(i=0;i=f)&&Object.keys(b.O).every((e=>b.O[e](r[c])))?r.splice(c--,1):(o=!1,f0&&e[i-1][2]>f;i--)e[i]=e[i-1];e[i]=[r,a,f]},b.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return b.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,b.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var f=Object.create(null);b.r(f);var d={};t=t||[null,r({}),r([]),r(r)];for(var o=2&a&&e;"object"==typeof o&&!~t.indexOf(o);o=r(o))Object.getOwnPropertyNames(o).forEach((t=>d[t]=()=>e[t]));return d.default=()=>e,b.d(f,d),f},b.d=(e,t)=>{for(var r in t)b.o(t,r)&&!b.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},b.f={},b.e=e=>Promise.all(Object.keys(b.f).reduce(((t,r)=>(b.f[r](e,t),t)),[])),b.u=e=>"assets/js/"+({13:"9d9f8394",61:"1f391b9e",87:"e505b3a5",134:"393be207",138:"27d9d47d",146:"2140bf91",151:"0bfe2fe2",158:"0276dc25",228:"dea0f9ea",245:"f9f76190",278:"87124b4e",306:"52e2a80b",326:"f62d4472",330:"9b4185c1",332:"044d5aa6",401:"17896441",459:"4d54d076",496:"4bdb0d83",539:"9beb87c2",546:"ccab7630",552:"9be9f402",581:"935f2afb",586:"ac5280f2",588:"a3713279",634:"c4f5d8e4",676:"f52ebc2d",695:"1f5fb0df",714:"1be78505",778:"ffd4f506",790:"1745c531",814:"72e14192",831:"625b1ed7",847:"25b7c3f2",849:"18ba09e8",903:"a63906ba",992:"68d50eef"}[e]||e)+"."+{13:"87407cb8",61:"6f0bef76",87:"ef765b4b",90:"fffd597a",134:"2c0ad88d",138:"30b89259",146:"f2e2506e",151:"055a0f41",158:"b31c8ebe",228:"154146d6",245:"04df3d8f",278:"78387f6b",306:"2d1ea665",326:"2688d702",330:"fdb3a866",332:"cb2eebc2",343:"18574212",345:"8ee5f24c",401:"2d945a38",459:"be1fdcfc",496:"db41030b",504:"1b68801e",539:"70b0135c",546:"9d741c50",552:"7783963b",581:"f58a9a76",586:"a02d052e",588:"83fddf58",593:"8fb47887",634:"a3178ce4",676:"f5a0cf24",695:"55482f11",714:"93b17e9c",778:"eb67f751",790:"44ad1d35",814:"ba227462",831:"39397709",847:"a717e408",849:"b7906927",896:"acea2065",903:"a4ad446a",913:"dd95d7ef",992:"62b70386"}[e]+".js",b.miniCssF=e=>"assets/css/styles.0350c134.css",b.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),b.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},f="dashy:",b.l=(e,t,r,d)=>{if(a[e])a[e].push(t);else{var o,c;if(void 0!==r)for(var n=document.getElementsByTagName("script"),i=0;i{o.onerror=o.onload=null,clearTimeout(l);var f=a[e];if(delete a[e],o.parentNode&&o.parentNode.removeChild(o),f&&f.forEach((e=>e(r))),t)return t(r)},l=setTimeout(u.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=u.bind(null,o.onerror),o.onload=u.bind(null,o.onload),c&&document.head.appendChild(o)}},b.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},b.p="/",b.gca=function(e){return e={17896441:"401","9d9f8394":"13","1f391b9e":"61",e505b3a5:"87","393be207":"134","27d9d47d":"138","2140bf91":"146","0bfe2fe2":"151","0276dc25":"158",dea0f9ea:"228",f9f76190:"245","87124b4e":"278","52e2a80b":"306",f62d4472:"326","9b4185c1":"330","044d5aa6":"332","4d54d076":"459","4bdb0d83":"496","9beb87c2":"539",ccab7630:"546","9be9f402":"552","935f2afb":"581",ac5280f2:"586",a3713279:"588",c4f5d8e4:"634",f52ebc2d:"676","1f5fb0df":"695","1be78505":"714",ffd4f506:"778","1745c531":"790","72e14192":"814","625b1ed7":"831","25b7c3f2":"847","18ba09e8":"849",a63906ba:"903","68d50eef":"992"}[e]||e,b.p+b.u(e)},(()=>{var e={354:0,869:0};b.f.j=(t,r)=>{var a=b.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(354|869)$/.test(t))e[t]=0;else{var f=new Promise(((r,f)=>a=e[t]=[r,f]));r.push(a[2]=f);var d=b.p+b.u(t),o=new Error;b.l(d,(r=>{if(b.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var f=r&&("load"===r.type?"missing":r.type),d=r&&r.target&&r.target.src;o.message="Loading chunk "+t+" failed.\n("+f+": "+d+")",o.name="ChunkLoadError",o.type=f,o.request=d,a[1](o)}}),"chunk-"+t,t)}},b.O.j=t=>0===e[t];var t=(t,r)=>{var a,f,d=r[0],o=r[1],c=r[2],n=0;if(d.some((t=>0!==e[t]))){for(a in o)b.o(o,a)&&(b.m[a]=o[a]);if(c)var i=c(b)}for(t&&t(r);n{"use strict";var e,t,r,a,f,d={},o={};function b(e){var t=o[e];if(void 0!==t)return t.exports;var r=o[e]={id:e,loaded:!1,exports:{}};return d[e].call(r.exports,r,r.exports,b),r.loaded=!0,r.exports}b.m=d,b.c=o,e=[],b.O=(t,r,a,f)=>{if(!r){var d=1/0;for(i=0;i=f)&&Object.keys(b.O).every((e=>b.O[e](r[c])))?r.splice(c--,1):(o=!1,f0&&e[i-1][2]>f;i--)e[i]=e[i-1];e[i]=[r,a,f]},b.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return b.d(t,{a:t}),t},r=Object.getPrototypeOf?e=>Object.getPrototypeOf(e):e=>e.__proto__,b.t=function(e,a){if(1&a&&(e=this(e)),8&a)return e;if("object"==typeof e&&e){if(4&a&&e.__esModule)return e;if(16&a&&"function"==typeof e.then)return e}var f=Object.create(null);b.r(f);var d={};t=t||[null,r({}),r([]),r(r)];for(var o=2&a&&e;"object"==typeof o&&!~t.indexOf(o);o=r(o))Object.getOwnPropertyNames(o).forEach((t=>d[t]=()=>e[t]));return d.default=()=>e,b.d(f,d),f},b.d=(e,t)=>{for(var r in t)b.o(t,r)&&!b.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},b.f={},b.e=e=>Promise.all(Object.keys(b.f).reduce(((t,r)=>(b.f[r](e,t),t)),[])),b.u=e=>"assets/js/"+({13:"9d9f8394",61:"1f391b9e",87:"e505b3a5",134:"393be207",138:"27d9d47d",146:"2140bf91",151:"0bfe2fe2",158:"0276dc25",228:"dea0f9ea",245:"f9f76190",278:"87124b4e",306:"52e2a80b",326:"f62d4472",330:"9b4185c1",332:"044d5aa6",401:"17896441",459:"4d54d076",496:"4bdb0d83",539:"9beb87c2",546:"ccab7630",552:"9be9f402",581:"935f2afb",586:"ac5280f2",588:"a3713279",634:"c4f5d8e4",676:"f52ebc2d",695:"1f5fb0df",714:"1be78505",778:"ffd4f506",790:"1745c531",814:"72e14192",831:"625b1ed7",847:"25b7c3f2",849:"18ba09e8",903:"a63906ba",992:"68d50eef"}[e]||e)+"."+{13:"87407cb8",61:"6f0bef76",87:"ef765b4b",90:"fffd597a",134:"2c0ad88d",138:"30b89259",146:"f2e2506e",151:"055a0f41",158:"b31c8ebe",228:"154146d6",245:"04df3d8f",278:"78387f6b",306:"2d1ea665",326:"2688d702",330:"fdb3a866",332:"cb2eebc2",343:"18574212",345:"8ee5f24c",401:"2d945a38",459:"be1fdcfc",496:"db41030b",504:"1b68801e",539:"70b0135c",546:"9d741c50",552:"7783963b",581:"f58a9a76",586:"a02d052e",588:"83fddf58",593:"8fb47887",634:"a3178ce4",676:"f5a0cf24",695:"55482f11",714:"93b17e9c",778:"eb67f751",790:"44ad1d35",814:"4ebcc2d8",831:"b95d042d",847:"a717e408",849:"a254f0fc",896:"acea2065",903:"a4ad446a",913:"dd95d7ef",992:"62b70386"}[e]+".js",b.miniCssF=e=>"assets/css/styles.0350c134.css",b.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),b.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),a={},f="dashy:",b.l=(e,t,r,d)=>{if(a[e])a[e].push(t);else{var o,c;if(void 0!==r)for(var n=document.getElementsByTagName("script"),i=0;i{o.onerror=o.onload=null,clearTimeout(l);var f=a[e];if(delete a[e],o.parentNode&&o.parentNode.removeChild(o),f&&f.forEach((e=>e(r))),t)return t(r)},l=setTimeout(u.bind(null,void 0,{type:"timeout",target:o}),12e4);o.onerror=u.bind(null,o.onerror),o.onload=u.bind(null,o.onload),c&&document.head.appendChild(o)}},b.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},b.p="/",b.gca=function(e){return e={17896441:"401","9d9f8394":"13","1f391b9e":"61",e505b3a5:"87","393be207":"134","27d9d47d":"138","2140bf91":"146","0bfe2fe2":"151","0276dc25":"158",dea0f9ea:"228",f9f76190:"245","87124b4e":"278","52e2a80b":"306",f62d4472:"326","9b4185c1":"330","044d5aa6":"332","4d54d076":"459","4bdb0d83":"496","9beb87c2":"539",ccab7630:"546","9be9f402":"552","935f2afb":"581",ac5280f2:"586",a3713279:"588",c4f5d8e4:"634",f52ebc2d:"676","1f5fb0df":"695","1be78505":"714",ffd4f506:"778","1745c531":"790","72e14192":"814","625b1ed7":"831","25b7c3f2":"847","18ba09e8":"849",a63906ba:"903","68d50eef":"992"}[e]||e,b.p+b.u(e)},(()=>{var e={354:0,869:0};b.f.j=(t,r)=>{var a=b.o(e,t)?e[t]:void 0;if(0!==a)if(a)r.push(a[2]);else if(/^(354|869)$/.test(t))e[t]=0;else{var f=new Promise(((r,f)=>a=e[t]=[r,f]));r.push(a[2]=f);var d=b.p+b.u(t),o=new Error;b.l(d,(r=>{if(b.o(e,t)&&(0!==(a=e[t])&&(e[t]=void 0),a)){var f=r&&("load"===r.type?"missing":r.type),d=r&&r.target&&r.target.src;o.message="Loading chunk "+t+" failed.\n("+f+": "+d+")",o.name="ChunkLoadError",o.type=f,o.request=d,a[1](o)}}),"chunk-"+t,t)}},b.O.j=t=>0===e[t];var t=(t,r)=>{var a,f,d=r[0],o=r[1],c=r[2],n=0;if(d.some((t=>0!==e[t]))){for(a in o)b.o(o,a)&&(b.m[a]=o[a]);if(c)var i=c(b)}for(t&&t(r);n Alternate Views & Opening Methods | Dashy - +

Alternate Views & Opening Methods

Views#

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.

Default#

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
Demo

Workspace#

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
Workspace view demo

Minimal View#

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
Workspace view demo

Opening Methods#

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.

- + \ No newline at end of file diff --git a/docs/authentication/index.html b/docs/authentication/index.html index fdef6db0..31f16904 100644 --- a/docs/authentication/index.html +++ b/docs/authentication/index.html @@ -6,22 +6,22 @@ Authentication | Dashy - +
-

Authentication

[!IMPORTANT] +

Authentication

[!IMPORTANT] Dashy's built-in auth is not indented 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.

In cases where Dashy is only accessibly within your home network, and you just want to add a login page, then the built-in auth may be sufficient, but keep in mind that configuration can still be accessed.

Built-In Auth#

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.

[!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. -You can trigger a rebuild through the UI, under Config --> Rebuild, or by running yarn build in the root directory.

Setting Up Authentication#

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

Hash Password#

Dashy uses SHA-256 Hash, a 64-character string, which you can generate 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.

Logging In and Out#

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.

Enabling Guest Access#

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.

Granular Access#

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]

Permissions#

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.

Security#

With basic auth, all logic is happening on the client-side, which could mean a skilled user could manipulate the code to view parts of your configuration, including the hash. If the SHA-256 hash is of a common password, it may be possible to determine it, using a lookup table, in order to find the original password. Which can be used to manually generate the auth token, that can then be inserted into session storage, to become a valid logged in user. Therefore, you should always use a long, strong and unique password, and if you instance contains security-critical info and/ or is exposed directly to the internet, and alternative authentication method may be better. The purpose of the login page is merely to prevent immediate unauthorized access to your homepage.

⬆️ Back to Top


Keycloak#

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. +You can trigger a rebuild through the UI, under Config --> Rebuild, or by running yarn build in the root directory.

Setting Up Authentication#

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

Hash Password#

Dashy uses SHA-256 Hash, a 64-character string, which you can generate 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.

Logging In and Out#

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.

Enabling Guest Access#

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.

Granular Access#

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]

Permissions#

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.

Using Environment Variables for Passwords#

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 VUE_APP_, and you must set this variable before building the app.

For example:

  auth:    users:    - user: bob      password: VUE_APP_BOB

Just be sure to set VUE_APP_BOB='my super secret password' before build-time.

Adding HTTP Auth to Configuration#

If you'd also like to prevent direct visit access to your configuration file, you can set the ENABLE_HTTP_AUTH environmental variable.

Security#

With basic auth, all logic is happening on the client-side, which could mean a skilled user could manipulate the code to view parts of your configuration, including the hash. If the SHA-256 hash is of a common password, it may be possible to determine it, using a lookup table, in order to find the original password. Which can be used to manually generate the auth token, that can then be inserted into session storage, to become a valid logged in user. Therefore, you should always use a long, strong and unique password, and if you instance contains security-critical info and/ or is exposed directly to the internet, and alternative authentication method may be better. The purpose of the login page is merely to prevent immediate unauthorized access to your homepage.

⬆️ Back to Top


HTTP Auth#

If you'd like to protect all your config files from direct access, you can set the BASIC_AUTH_USERNAME and BASIC_AUTH_PASSWORD environmental variables. You'll then be prompted to enter these credentials when visiting Dashy.

Then, if you'd like your frontend to automatically log you in, without prompting you for credentials, then also specify VUE_APP_BASIC_AUTH_USERNAME and VUE_APP_BASIC_AUTH_PASSWORD. This is useful for when you're hosting Dashy on a private server, and you want to prevent unauthorized access to your config files, while still allowing the frontend to access them. Note that a rebuild is required for these changes to take effect.

⬆️ Back to Top


Keycloak#

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.

1. Deploy Keycloak#

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_USER=admin \  -e KEYCLOAK_PASSWORD=admin \  quay.io/keycloak/keycloak:15.0.2

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.

2. Setup Keycloak Users#

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.

  1. Head over to the admin console
  2. In the top-left corner there is a dropdown called 'Master', hover over it and then click 'Add Realm'
  3. Give your realm a name, and hit 'Create'

You can now create your first user.

  1. In the left-hand menu, click 'Users', then 'Add User'
  2. Fill in the form, including username and hit 'Save'
  3. Under the 'Credentials' tab, give the new user an initial password. They will be prompted to change this after first login

The last thing we need to do in the Keycloak admin console is to create a new client

  1. Within your new realm, navigate to 'Clients' on the left-hand side, then click 'Create' in the top-right
  2. Choose a 'Client ID', set 'Client Protocol' to 'openid-connect', and for 'Valid Redirect URIs' put a URL pattern to where you're hosting Dashy (if you're just testing locally, then * is fine), and do the same for the 'Web Origins' field
  3. Make note of your client-id, and click 'Save'

3. Enable Keycloak in Dashy Config File#

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:  ...  auth:    enableKeycloak: true    keycloak:      serverUrl: 'http://localhost:8081'      realm: 'alicia-homelab'      clientId: 'dashy'

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'.

4. Add groups and roles (Optional)#

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']

Depending on how you're hosting Dashy and Keycloak, you may also need to set some HTTP headers, to prevent a CORS error. This would typically be the Access-Control-Allow-Origin [URL-of Dashy] on your Keycloak instance. See the Setting Headers guide in the management docs for more info.

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.


Alternative Authentication Methods#

If you are self-hosting Dashy, and require secure authentication to prevent unauthorized access, then you can either use Keycloak, or one of the following options:

Authentication Server#

Authelia#

Authelia is an open-source full-featured authentication server, which can be self-hosted and either on bare metal, in a Docker container or in a Kubernetes cluster. It allows for fine-grained access control rules based on IP, path, users etc, and supports 2FA, simple password access or bypass policies for your domains.

  • git clone https://github.com/authelia/authelia.git
  • cd authelia/examples/compose/lite
  • Modify the users_database.yml the default username and password is authelia
  • Modify the configuration.yml and docker-compose.yml with your respective domains and secrets
  • docker-compose up -d

For more information, see the Authelia docs

VPN#

A catch-all solution to accessing services running from your home network remotely is to use a VPN. It means you do not need to worry about implementing complex authentication rules, or trusting the login implementation of individual applications. However it can be inconvenient to use on a day-to-day basis, and some public and corporate WiFi block VPN connections. Two popular VPN protocols are OpenVPN and WireGuard

IP-Based Access#

If you have a static IP or use a VPN to access your running services, then you can use conditional access to block access to Dashy from everyone except users of your pre-defined IP address. This feature is offered by most cloud providers, and supported by most web servers.

Apache#

In Apache, this is configured in your .htaccess file in Dashy's root folder, and should look something like:

Order Deny,AllowDeny from allAllow from [your-ip]

NGINX#

In NGINX you can specify control access rules for a given site in your nginx.conf or hosts file. For example:

server {    listen 8080;    server_name www.dashy.example.com;    location / {        root /path/to/dashy/;        passenger_enabled on;        allow [your-ip];        deny all;    }  }

Caddy#

In Caddy, Request Matchers can be used to filter requests

dashy.site {    @public_networks not remote_ip [your-ip]    respond @public_networks "Access denied" 403}

Web Server Authentication#

Most web servers make password protecting certain apps very easy. Note that you should also set up HTTPS and have a valid certificate in order for this to be secure.

Apache#

First crate a .htaccess file in Dashy's route directory. Specify the auth type and path to where you want to store the password file (usually the same folder). For example:

AuthType BasicAuthName "Please Sign into Dashy"AuthUserFile /path/dashy/.htpasswdrequire valid-user

Then create a .htpasswd file in the same directory. List users and their hashed passwords here, with one user on each line, and a colon between username and password (e.g. [username]:[hashed-password]). You will need to generate an MD5 hash of your desired password, this can be done with an online tool. Your file will look something like:

alicia:$apr1$jv0spemw$RzOX5/GgY69JMkgV6u16l0

NGINX#

NGINX has an authentication module which can be used to add passwords to given sites, and is fairly simple to set up. Similar to above, you will need to create a .htpasswd file. Then just enable auth and specify the path to that file, for example:

location / {  auth_basic "closed site";  auth_basic_user_file conf/htpasswd;}

Caddy#

Caddy has a basic-auth directive, where you specify a username and hash. The password hash needs to be base-64 encoded, the caddy hash-password command can help with this. For example:

basicauth /secret/* {    alicia JDJhJDEwJEVCNmdaNEg2Ti5iejRMYkF3MFZhZ3VtV3E1SzBWZEZ5Q3VWc0tzOEJwZE9TaFlZdEVkZDhX}

For more info about implementing a single sign on for all your apps with Caddy, see this tutorial

Lighttpd#

You can use the mod_auth module to secure your site with Lighttpd. Like with Apache, you need to first create a password file listing your usernames and hashed passwords, but in Lighttpd, it's usually called .lighttpdpassword.

Then in your lighttpd.conf file (usually in the /etc/lighttpd/ directory), load in the mod_auth module, and configure it's directives. For example:

server.modules += ( "mod_auth" )auth.debug = 2auth.backend = "plain"auth.backend.plain.userfile = "/home/lighttpd/.lighttpdpassword"
-$HTTP["host"] == "dashy.my-domain.net" {  server.document-root = "/home/lighttpd/dashy.my-domain.net/http"  server.errorlog = "/var/log/lighttpd/dashy.my-domain.net/error.log"  accesslog.filename = "/var/log/lighttpd/dashy.my-domain.net/access.log"  auth.require = (    "/docs/" => (      "method" => "basic",      "realm" => "Password protected area",      "require" => "user=alicia"    )  )}

Restart your web server for changes to take effect.

OAuth Services#

There are also authentication services, such as Ory.sh, Okta, Auth0, Firebase. Implementing one of these solutions would involve some changes to the Auth.js file, but should be fairly straightforward.

Static Site Hosting Providers#

If you are hosting Dashy on a cloud platform, you will probably find that it has built-in support for password protected access to web apps. For more info, see the relevant docs for your provider, for example: Netlify Password Protection, Cloudflare Access, AWS Cognito, Azure Authentication and Vercel Password Protection.

⬆️ Back to Top

- +$HTTP["host"] == "dashy.my-domain.net" { server.document-root = "/home/lighttpd/dashy.my-domain.net/http" server.errorlog = "/var/log/lighttpd/dashy.my-domain.net/error.log" accesslog.filename = "/var/log/lighttpd/dashy.my-domain.net/access.log" auth.require = ( "/docs/" => ( "method" => "basic", "realm" => "Password protected area", "require" => "user=alicia" ) )}

Restart your web server for changes to take effect.

OAuth Services#

There are also authentication services, such as Ory.sh, Okta, Auth0, Firebase. Implementing one of these solutions would involve some changes to the Auth.js file, but should be fairly straightforward.

Static Site Hosting Providers#

If you are hosting Dashy on a cloud platform, you will probably find that it has built-in support for password protected access to web apps. For more info, see the relevant docs for your provider, for example: Netlify Password Protection, Cloudflare Access, AWS Cognito, Azure Authentication and Vercel Password Protection.

⬆️ Back to Top

+ \ No newline at end of file diff --git a/docs/backup-restore/index.html b/docs/backup-restore/index.html index 0ba5457d..6d6c5f71 100644 --- a/docs/backup-restore/index.html +++ b/docs/backup-restore/index.html @@ -6,7 +6,7 @@ Cloud Backup and Restore | Dashy - + @@ -15,7 +15,7 @@ workers_dev = trueroute = "example.com/*"zone_id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"account_id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" kv_namespaces = [ { binding = "DASHY_CLOUD_BACKUP", id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" }]

Complete index.js#

  • 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

Commands#

  • wrangler dev - To start the wrangler development server
  • wrangler publish - To publish to your cloudflare account (first run wrangler login)

API#

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.

- + \ No newline at end of file diff --git a/docs/changelog/index.html b/docs/changelog/index.html index 5a7b2de2..7e243801 100644 --- a/docs/changelog/index.html +++ b/docs/changelog/index.html @@ -6,7 +6,7 @@ Changelog | Dashy - + @@ -16,7 +16,7 @@ Fixes #544 #555

Several other changes sinc When using SSL, the server can now redirect from HTTP to HTTPS. This is enabled by default when using SSL. #538 Section context menus are now accessible on mobile, and will no longer clip off the screen. #541 Italian translations have been added. #556

✨ 2.0.4 - Dynamic Config Loading PR #528

  • conf.yml is now loaded dynamically and the app now only needs a browser refresh on config change, not a full rebuild!

🐛 2.0.3 - Bug Fixes PR #488

  • Press enter to submit login form (Re: #483)
  • Allow disabling write to local storage and disk (Re: #485)
  • Fix malformed YAML from export config (Re: #482)
  • Allow global option for useProxy (Re: #486)
  • Look into arrow key navigation error (Re: #463)
  • Disallow displaying config (Re: #455)
  • Round values in Glances Alerts widget (Re: #454)
  • Create a CPU temp widget (Re: #452)
  • Add to docs: Keycloak in Kubernetes (Re: #479)
  • Add a widget for displaying images (Re: #487)

⬆️ 2.0.2 - Dependency Updates PR #471

  • Updates Alpine version for main Dockerfile
  • Updates node_modules to latest stable versions

🐛 2.0.1 - Fixes Section Height PR #462

  • Adds cutToHeight to config schema (Re: #461)
  • Removes the full-height CSS from colorful theme
  • Improved config validation warnings in JSON editor
  • Removes empty Keycloak block from appConfig editor
  • Adds typechecking to search and clear search for Safari

⚡️ 2.0.0 - Small Fixes and Docker Multi-Arch Build PR #451

  • Fixes full-height sections for mobile and Safari (Re: #432, #442)
  • Fixes empty section visible in search (Re: #447)
  • Fixes numbers omited from tag names (Re: #430)
  • Option for custom status code in status check (Re: #456, #448)
  • Adds @stuu3k's dashboard to showcase (Re: #446)
  • Switches recover and death count in Covid widget (Re: #148)
  • Improved contrast in light material theme
  • Adds new script to lint, test, build and publish a multi-architecture Docker image to various registries

💄 1.9.9 - Minor UI + Docs Updates PR #431

  • Improved theme support for widgets
  • Better widget layout in Workspace and Minimal views
  • Updates lots of the docs

✨ 1.9.8 - More Widgets and Widget Improvements PR #425

  • Fixes several minor widget issues raised by users
  • Adds several new widgets, for monitoring system
  • Better widget data requests and error handling
  • Implements widget support into Workspace view

🐛 1.9.7 - Minor UI Editor Bug fixes PR #416

  • Fixes unable to edit item bug (#415)
  • Fixes unable to add new app bug (#390)
  • Fixes nav links visibility (#389)

⚡️ 1.9.6 - Adds Proxy Support for Widget Requests PR #392

  • Refactors widget mixin to include data requests, so that code can be shared between widgets
  • Adds a Node endpoint for proxying requests server-side, used for APIs that are not CORS enabled
  • Adds option to config file for user to force proxying of requests
  • Writes a Netlify cloud function to support proxying when the app is hosted on Netlify

🐛 1.9.5 - Bug fixes and Minor Improvements PR #388

  • Adds icon.horse to supported favicon APIs
  • Fixes tile move bug, Re: #366
  • Fixes save items without title bug, Re: #377

✨ 1.9.4 - Widget Support PR #382

  • Adds support for dynamic content, through widgets
  • Adds 30+ pre-built widgets for general info and self-hosted services
  • Writes docs on widget usage

⚡️ 1.9.2 - Native SSL Support + Performance Improvements PR #326

  • Updates the server to use Express, removing serve-static, connect and body-parser
  • Adds native support for passing in self-signed SSL certificates and updates docs
  • Updates router to lazy-load additional pages (minimal, workspace, etc)
  • Changes default favicon API to allesedv, since faviconkit is down, and adds basic fallback
  • Updates GH action build scripts to fallback on context token when running on fork

💄 1.9.1 - Editor and Theming Fixes and Improvements PR #319

  • Bug fixes for interactive editor: #310, #311, #312
  • Adds option to modify text font through the UI
  • Adds two new themes: One Dark and Adventure
  • Theming stylesheet refactor, better inheritance

🐳 1.9.0 - Alpha of Dashy-Lite Docker Container PR #306

  • Create an Alpine-based container, that serves the built app up with plain NGINX, instead of Node.
  • This is much lighter, but doesn't currently support any of the server-side actions (like status-checks, and writing changes to disk)

✨ 1.8.9 - All New Interactive Config Editor PR #298

  • Builds a new UI-based config editor
  • Support for sections, items, app config and page info
  • Live preview, and undoing of local changes
  • Export config or write changes to disk through UI

✨ 1.8.8 - Improved Item Targets PR #292

  • Adds support for _top and _parent anchor targets on items, Re: #289
  • Adds appConfig.defaultOpeningMethod option to specify default target
  • Adds new icons to show items opening method on hover
  • Refactors target checking, updates item target docs and schema

⚡️ 1.8.7 - Bug Fixes and Improvements PR #273

  • Clean URLs without the hash, now using history-mode routing
  • New initial main example conf.yml
  • Minor UI style updates and fixes
  • Support for single section view
  • A new theme, soft-glow
  • Container security in management docs, and other things
  • Bug fixes, including missing Firefox favicon and fix custom icon paths with base_url

⚡️ 1.8.6 - Implementation of VueX PR: #271

  • New state management pattern, which should lead to a more organized code base long term, and will also make building out the new UI editor significantly easier to do in a clean and reliable way

💄 1.8.5 - Lots of Requested UI Improvements PR #261

  • Adds an option for landing URL in workspace, Re: #255
  • Switches to a new API for generative icons, Re: #163
  • Adds new tab functionality to Workspace, Re: #254
  • Remove CSS validation in style editor, Re: #259
  • Cap item description at 2 lines, Re: #250
  • Adds native support for common homelab icons, using dashboard-icons
  • Improves general responsiveness of home page sections positioning
  • Updates, fixes and adds a bunch of actions for easier repo management

✨ 1.8.4 - Custom Error Pages PR #257

  • Creates a 404 Not Found page
  • Routes any missing views to the 404 page

⚡️ 1.8.3 - Improved UX for Initial Load PR #238

  • Removes the old splash screen
  • Adds placeholder in the HTML index, which will usually be visible on initial load
  • Show progress bar on route switcher

✨ 1.8.2 - Serverless Functions for Netlify Instances PR #235

  • Previously when Dashy was deployed as a static site to Netlify, it was not possible to use several features, which required server-side code
  • This PR adds serverless cloud functions to provide most of this functionality

🩹 1.8.1 - Additional Languages, Bug Fix, and more PR #234

  • Merges 5 additional languages
  • Adds RickyCZ's dashboard to showcase
  • Fixes #323, and improves status indicator tooltip
  • Define constants for endpoints
  • Updates management, translations and readme docs

⚡️ 1.7.7 - Improved Error Handling PR #226

  • Errors can be viewed via the UI (Config menu --> App Info)
  • Service workers should be disabled by default
  • Also renames actions to be easier to maintain
  • Updates docs to include release schedule + merge strategy

✨ 1.7.6 - Adds Multi-Search Support with Bangs PR #224

  • Adds option for user to add custom search bangs, in order to specify search engine/ target app. Re: #206

🎨 1.7.5 - Improved Language Detection & UI PR #223

  • Makes the auto language detection algo smarter
  • Improves responsiveness for the language selector form

🌐 1.7.4 - Adds Spanish Translations PR #222

  • Adds Spanish language file, contributed by @lu4t

👷 1.7.3 - CI with GH Actions PR #212

  • Adds a series of GH actions for repository maintenance and administration
  • Auto-labels PR and issues, adds helpful info to PRs based on files changed
  • Lints, tests, builds and deploys the app when PR submitted
  • Checks app size, dependencies, security, accessibility etc when PR submitted
  • Closes incomplete, stale or spammy issues and leaves a comment

✨ 1.7.2 - Item Sort Options PR #203

  • Adds option for user to specify sortBy to order items within a section
  • Can sort by last clicked, most used, alphabetically and more
  • And fixes UI of the item tooltip and, if specified, will show Provider in the tooltip
  • Also improves error logging and console warning message

🐛 1.7.1 - Lots of Tiny Fixes and Improvements PR #200

  • Removes background in console art
  • Updates auto environmental variables
  • Icon image assets max height Force same Icon/Item Height #200
  • Adds an action to close spammy issues
  • Adds option to enable SRI integrity, plus refactos PWA into defaults
  • Updates privacy and security docs
  • Adds option for different favicon API for each app [FEATURE_REQUEST] Allow using different faviconApi for each items #196
  • Fixes loading of local SVG icons #199

🍻 1.7.0 - Documentation Website PR #190

  • Builds a quick website to host the docs. No code changes, but prepares for V1.7 release

✨ 1.6.9 - Web Search Feature PR #185

  • Adds ability to search the web directly from Dashy by pressing enter on the search bar
  • Adds configuration options, for setting default search engine, opening method and disabling
  • Adds text under search bar, implements into minimal view also

⚡️ 1.6.8 - Improved Loading Experience PR #183

  • During app initialization, show the build progress and status message
  • While requests are being made, show loader at top of screen
  • Also adds some UI improvements to Workspace view

⚡️ 1.6.7 - Option for non-SSL status checks plus minor things PR #182

  • Adds an option for user to use status checks with non-HTTPS services, Re: #181
  • Updates the .env template, plus the variables used in the server
  • Uses the v-cloak to hide text before it's finished loading
  • Fixed the parsing of the update-checker during build

⚡️ 1.6.6 - Improved Search & Shortcuts PR #175

  • Refactors the search algorithm to improve performance and code reusability
  • Updates search to ignore case, special characters and minor-typos
  • Adds the option for user to specify tags, which can be used for searching

✨ 1.6.5 - Adds support for Secure Authentication using Keycloak PR #174

  • Major restructure of auth config
  • Implements keycloak support, adds docs and updates schema

✨ 1.6.4 - Adds functionality for Granular Auth Control PR #171

  • Enables sections to be visible for all users except for those specified
  • Enables sections to be hidden from all users except for those specified
  • Enables sections to be hidden from guests, but visible to all authenticated users

⚡️ 1.6.3 - Dependency and Build File Updates PR #168

  • Removes any dependencies which are not 100% essential
  • Moves packages that are only used for building into devDependencies
  • Updates dependencies to latest version
  • Adds a .dockerignore, so that non-essential files are not included in the container
  • Updates deployment config files for Netlify, Heroku and GH actions
  • Made a brand new bug-report template, with input fields and validation!

✨ 1.6.2 - Support for Guest Access PR #167

  • Adds functionality for optional read-only guest access to dashboards with authentication
  • Can be enabled by setting appConfig.enableGuestAccess: true

💄 1.6.1 - Adds new Theme PR #166

  • Adds Dashy theme, for use in the dev dashboard

✨ 1.5.9 - New Minimal/ Startpage View PR #155

  • Adds a new view, called minimal view, designed to be like a light-weight startpage
  • Implemented all the required features (filtering, opening methods, icons, etc) into minimal view
  • Adds appConfig.startingView into schema, for specifying the initial default view to be loaded

✨ 1.5.8 - Multi-Tasking Support in Workspace View PR #146

  • Adds option to keep launched apps open in the background, to reduce friction when switching between websites, Re: #144
  • This can be enabled by setting appConfig.enableMultiTasking: true
  • Note that having many apps opened simultaneously, will have an impact on performance

✨ 1.5.7 - Adds Support for Material Design Icons PR #141

⚡️ 1.5.6 - Refactor + Couple of small things PR #135

  • The main Dockerfile now uses yarn.lock instead of package-lock.json
  • Adds a check to verify password is not empty in cloud backup screen
  • Improves responsiveness of config modals for mobile devices
  • Enables the user to use their own self-hosted Sentry instance
  • Removes the View Config tab of the Config menu, as not needed
  • Updates and fixes some typos in the readme

🌐 1.5.5 - Adds Missing Translations + Small UI Issues PR #129

  • Adds missing translations to several UI elements, Re: #126
  • Fixes login translations not being picked up on page load, Re: #127
  • Fixes small text overflow glitch in config icon, Re: #123
  • Several small UI improvements: height of config editor, scrollbar on theme dropdown, page height, white-on-white on material theme, etc
  • Adds an action to auto-assign reviewer based on ./.github/CODEOWNERS file

🐳 1.5.4 - Docker ARM Support PR #122

  • Adds a Dockerfile for arm64v8 and arm32v7, to support Raspberry Pi and other modern ARM-based devices
  • Sets up automated workflow to publish ARM containers to DockerHub after every new release
  • Adds documentation for running Dashy on RPi/ ARM-based devices, Re: #117

🩹 1.5.3 - UI Quick Fix PR #121

  • Downgrades and pins vue-material-tabs to 0.1.5, to prevent breaking changes. Fixes #118 p1
  • Sets auto-width for theme selector, so text doesn't wrap for long theme names. Fixes #119
  • Uses flex layout for config menu, so note doesn't overlap menu on small screens. Fixes #118 p2

🌐 1.5.2 - Adds Dutch Translations PR #120

  • Dutch language support, contributed by @evroon

🩹 1.5.1 - UI Quick Fix PR #116

  • Uses min-max width parent layout, to prevent longer languages (e.g. French) text overflow. Fixes #115

🔒 1.5.0 - Improve Robustness of Auth PR #113

  • Use both username + password for generating token, so that a change in either will log the user out
  • Prevent privilege escalation by disallowing a user from modifying their user type through the UI
  • Improve the isAuthenticated check, by taking account of empty users array

✨ 1.4.8 - Optional Crash Reports PR #112

  • Adds an optional, off by default method of getting crash reports
  • This can be enabled in appConfig.enableErrorReporting, and will not be used at all unless explicitly activated by user
  • This is needed for when a user raises a bug which is hard to fix
  • Also improves robustness of config accumulator, don't throw error when config is missing
  • Adds Privacy & Security docs

♻️ 1.4.7 - Refactor PR #110

  • Moves cloud sync dialog into the config menu, and removes icon on homepage
  • Fixes typo in Default theme name, Re #106
  • Spell checks readme
  • Updates the contributor CI action, that generates list of contributors + sponsors

📝 1.4.6 - Documentation Updates PR #108

  • Breaks many of the longer files into several more digestible articles
  • Writes repo pages including, Security, Code of Conduct, Legal, Updates license
  • Makes an automatically generated Credits page
  • Adds a contributing page, with several ways that users can help out
  • Implements this changelog, as requested in #87

🌐 1.4.5 - Adds German Translations PR #107

  • German language support, contributed by @Niklashere

✨ 1.4.4 - Adds Support for Logo Image PR #105

  • Adds option in config file for user to specify path to an image
  • If found, will display said image in the header

✨ 1.4.3 - Auto-Checks for Updates PR #101 and PR #102

  • Write a script to compare current version with git master version
  • Periodically checks for updates, and displays message to user
  • Enables user to disable update-checks in the config file
  • Checks not using vulnerable version on project-build

✨ 1.4.2 - Adds Multi-Language Support PR #99

  • Implements vue-i18n, sets object globally
  • Extracts all text to a single JSON file
  • Auto-detects users language, and applies, if availible
  • Builds a form to let user manually select their language
  • Lets users language be saved and read from local storage, or config file

✨ 1.4.1 - Adds Support for Custom Key Bindings PR #94

  • Adds new attribute under item for saving numeric key binding
  • Listens for keypress, and launches corresponding item, if found

✨ 1.4.0 - Builds a Custom Theme Configurator

  • Adds property to save custom theme variables
  • Builds UI form, with color pickers, a pallette and popup
  • Integrates the saving colors, and applying saved colors functionality

🔨 1.3.9 - Enable Custom Styesheet in Docker PR #92

  • Enables the user to pass a custom stylesheet in with Docker
  • Adds support for 1-Click deployment to Render.com

🌟 1.3.8 - Showcase #91

  • Adds @Shadowking001's screenshot to showcase

🌟 1.3.7 - Showcase PR #84

  • Adds @dtctek's screenshot to showcase

✨ 1.3.6 - Enables User to Hide Unwanted Components PR #78

  • Adds several additional options to the config, allowing the user to hide structural components that they don't need
  • Including hideHeading, hideNav, hideSearch, hideSettings, hideFooter, hideSplashScreen

✨ 1.3.5 - Adds Support for Emoji Icons PR #76

  • Enables user to use emojis for item and section icons
  • Adds a handler to convert Unicode, or Shortcode into an Emoji

🌟 1.3.4 - Showcase Addition PR #75

  • Adds @cerealconyogurt's screenshot to the showcase

💄 1.3.3 - UI Improvements PR #73

  • New style of Large item
  • 2 new color themes
  • Added CSS variables for search label and footer background
  • Improves process for auto-checking if font-awesome is needed
  • Silences non-critical warnings in production build
  • Adds new optional font-face for cyber punk
  • Shortens readme, and adds contribute links to showcase

⚡️ 1.3.0 - Custom Headers for Status Check PR #72

  • Enables user to pass custom headers to the status check endpoint
  • Enables user to use a different URL for the status check request

🌟 1.2.9 - Creates a Showcase Page PR #68

  • Adds a page in the docs for users to share their screenshots of their dashboard

✨ 1.2.8 - Adds Remember-Me Functionality into the Login Form PR #66

  • Adds a dropdown menu in the login form with various time intervals available
  • Adds appropriate expiry into session storage, in order to keep user logged in for their desired time interval

✨ 1.2.7 - Implements a Right-Click Context Menu #62

  • Built a context menu, showing all item opening methods, on right-click
  • Made a clickOutside directive, in order to close menu when user clicks away
  • Adds launching functionality, user can click to launch

⚡️ 1.2.6 - Make Font Assets Local PR #60

  • Downloaded font files to assets
  • Removed all calls to font CDN, replaced with local calls

🐛 1.2.5 - Small Fixes, and Efficiency Improvements PR #57

  • Adds correct license
  • Improves service workers, and adds serviceWorkerStatus local storage item
  • Adds missing statusCheck and statusCheckInterval docs into Configuring.md
  • Adds an About App page, containing info needed to raise a bug report
  • Adds TDLR license into main readme
  • Introduces app versioning
  • Adds safeguards into ConfigAccumalaror, to prevent error being thrown
  • Updates PR template
  • Improved Webpack build experience, with progress bar and completion notification
  • Adds new and improved icons for layout options
  • Make the Page Title into a home page link
  • Adds missing favicon, fixes #55
  • Adds assets to PWA manifest.json
  • Documents app commands in readme
  • Enable passing website as URL param to the workspace
  • Modified items, so that title text doesn't get shortened,

✨ 1.2.4 - Adds Support for Continuous Status Checking #52

  • Enables user to re-call the status check at a specified interval
  • Processes interval in ms, and updates the traffic light when required

🐛 1.2.3 - Bug Fix PR #49

  • Removes duplicate Docker env var, fixes #48

✨ 1.2.2 - Better Favicon Support

  • Enables user to force direct/ local favicon fetching
  • Adds support for additional favicon API, returning high-res app icons
  • Adds support for generative icons

🐛 1.2.1 - Bugfix #44

  • Fixes footer positioning on mobile, makes sticky, fixes #42

✨ 1.2.0 - Adds Writing Config to Disk from UI Functionality PR #43

  • Creates a new server endpoint for handling the backing up of a the file
  • Adds backup existing file functionality
  • Adds writing new file functionality
  • Does error checking, testing and adds some security parameters
  • Adds a radio button in the UI, so user chan choose save method
  • Process config within the UI, convert to YAML, and write changes to disk

🐛 1.1.8 - Bugfix #40

  • Status check tooltip was not visible in Material themes, raised in issue #39

✨ 1.1.7 - Adds Workspace View PR #38

  • Adds a new route, for the workspace view
  • Builds the sidebar, which displays the users apps
  • Loads the app into the workspace's main iframe when clicked
  • Adds some collapsing functionality, better styles, subtle animations and theme support

✨ 1.1.6 - Implements Status Indicators, and Monitoring Functionality PR #34

  • Wrote a Node endpoint for pinging the users desired services
  • Added status checking functionality in frontend
  • Build small traffic-light component to display status of users services
  • Adds animations, and handles errors
  • Writes docs, and tests code

✨ 1.1.5 - Adds Authentication / Login Functionality PR #32

  • Enables the user to protect their dashboard behind a login screen
  • Creates a Authentication handler to manage the hashing of passwords, and generation of a token
  • Build a quick login form, where user can input username and password
  • Adds a log out button

💄 1.1.4 - Support for Custom HTML Footer PR #30

  • Enables user to insert structure for the footer defined as HTML

🚀 1.1.3 - Adds Support for 1-Click Cloud Deployments PR #29

  • Support for 1-Click Deploy to Netlify
  • Support for 1-Click Deploy to Heroku

🔧 1.1.2 - Docker Efficiency Improvements PR #26

  • Writes a Node health check script, and implements into the Docker container
  • Changes default port in docker-compose, as 8080 is commonly used by other apps
  • Adds the 1-Click deploy with PWD into the readme
  • Updates dependencies
  • Adds a getting started guide to the docs
  • Adds splash screen for first load
  • Deleted unused assets
  • Makes linter run as a pre-commit hook
  • Fixes lint errors in server.js and validate-config.js

🐛 1.1.1 - Bug Fixes PR #20 + PR #21

  • Adds issue template
  • Bug fixes
    • Improves github PR and issue templates
    • Shortens readme file
    • Adds documentation in the docs folder
    • Fixes Layout tab not showing in portrait #19
    • Improves mobile performance for both the settings, config and backup pop-ups
    • Fixes issue where theme not applied on load when the settings are hidden
    • Adds minimum dimensions to modalsShortens readme file
    • Adds documentation in the docs folder
    • Adds minimum dimensions to modals

🚑️ 1.1.0 - Hotfix #18

  • Implementing the JSON validator had actually broken the entire JSON editor
  • Fixed it by remove explicit use of Ajv, and using a derivative instead

📝 1.0.5 - Documentation PR #16

  • Previously there was very little documentation, this release fixed that
  • Wrote specific docs for:
    • Getting Started
    • Configuring
    • Backup & Restore
    • Theming
    • Developing

✨ 1.0.0 - Implements Config Validation PR #13

  • Write a JSON schema for the conf.yml file
  • Wrote a validation script to compare users config against schema
  • Adds a formatter to print helpful messages about what needs fixing
  • Implements validation process into build script
  • Implements validation process into UI config configurator's validation

🔧 0.9.5 - Brand New Docker Container PR #12

  • With help from several users, a new container based on Alpine is released
  • A sample Docker Compose script is also written, and docs are updated
  • A 1-Click button for deploying to Play-with-Docker is added to the Readme

✨ 0.9.0 - Adds Hide Settings Functionality PR #11

  • Enables user to hide settings from UI
  • Users preference is saved in local storage
  • User can hide other structural elements of the UI from the config

💄 0.8.5 - Adds new Built-In Themes PR #9

  • Adds Minimal-Dark and Minimal-Light theme
  • Adds Material-Dark and Material-Light theme
  • Adds additional theme docs
  • Adds option for sections to have items too

✨ 0.8.0 - Implements Custom CSS Editor PR: #8

  • Adds a page in the config menu
  • Adds syntax highlighting, CSS validation and sanitization
  • Saves users CSS, and applies styles on page load

✨ 0.7.5 - Adds Cloud Backup and Restore Feature PR #6

  • Creates a form for entering backup ID and decryption password
  • Puts form in modal, and adds button to launch form, with custom icon
  • Implemented the cryptography stuff for end-to-end data encryption
  • Wrote and tested the backend, and deployed as a serverless function on CF workers
  • On the frontend, users input is encrypted, and passed to backend cloud function
  • Response from the backend is handles appropriately, and message displayed to the user
  • Implements the restoring from server functionality, with data integrity checks

✨ 0.7.0 - Support for Custom Nav Links PR #4

  • User can add custom nav bar links from the Config Settings menu
  • Better UI styling to the config menu
  • New icons inside buttons

✨ 0.6.5 - UI Config Editor PR #3

Adds the ability for the user to edit their configuration directly from the UI

  • Edit all section and item data using a rich JSON editor
  • Download/ backup conf.yml directly from the UI
  • Edit site meta data: title, description, footer, etc
  • Reset all locally stored data to the initial state
  • Also includes a new toast component, for subtle notifications

✨ 0.6.0 - Navbar, Footer and Background Image

  • Adds option for a custom full-size background image
  • Made footer customizable
  • Fixes error being thrown when navbar links are empty

⚡️ 0.5.5 - Improved Theming

  • Makes more specific color variables, which inherit base vars
  • Makes it possible for users to write their own theme
  • Fix some color edge cases
  • Adds docs for theming

✨ 0.5.0 - Theme Support

  • Converts all SCSS variables to CSS variables
  • Implements theme switching functionality
  • Adds a dropdown menu, enabling user to select theme
  • Adds an initial theme option to appConfig.theme
  • Saves selected theme to local storage
  • Wrote a ton of color themes

✨ 0.4.5 - Keyboard Navigation

  • Implements arrow key navigation

✨ 0.4.0 - Font Awesome Support

  • Adds support for Font-Awesome icons
  • Auto-loads font-awesome only when needed
  • Adds support for SVG icons

✨ 0.3.5 - Opening Method

  • Shows opening method on hover
  • Opening method can be specified in config, as item[n].target

🔨 0.3.0 - Docker

  • Writes a Dockerfile

🎨 0.2.5 - Code Quality, Docs and UI

  • Huge code quality overhaul, now uses AirBnB style ESLint
  • Adds in-code docs, removes unneeded code, moves reusable helpers into utils dir
  • Adds a readme, records a demo gif and adds some basic deployment docs
  • Removes dependencies which are not 100% necessary

✨ 0.2.0 - Collapsible Sections

  • Implements collapsing functionality, for less used or very long sections
  • Sections can read default state from section[n].collapsed within config
  • After change, state of each section is stored in local storage

⚡️ 0.1.5 - Search and Navigation

  • Improves instant search functionality
  • Implements keyboard navigation for selecting items
  • Launch selected item with enter, or Ctrl + Enter to open in new tab

🎉 0.1.0 - Init

Project started. Forked from Lissy93/Dash

- + \ No newline at end of file diff --git a/docs/code-of-conduct/index.html b/docs/code-of-conduct/index.html index 82e78acc..ff796057 100644 --- a/docs/code-of-conduct/index.html +++ b/docs/code-of-conduct/index.html @@ -6,7 +6,7 @@ Contributor Covenant Code of Conduct | Dashy - + @@ -61,7 +61,7 @@ version 2.0, available at enforcement ladder.

For answers to common questions about this code of conduct, see the FAQ at https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations.

- + \ No newline at end of file diff --git a/docs/configuring/index.html b/docs/configuring/index.html index 11685033..5f4b161a 100644 --- a/docs/configuring/index.html +++ b/docs/configuring/index.html @@ -6,7 +6,7 @@ Configuring | Dashy - + @@ -16,7 +16,7 @@ Since the auth is initiated in the main app entry point (for security), a rebuil You can trigger a rebuild through the UI, under Config --> Rebuild, or by running yarn build in the root directory.

[!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.

FieldTypeRequiredDescription
usersarrayOptionalAn 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.
enableKeycloakbooleanOptionalIf 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
keycloakobjectOptionalConfig options to point Dashy to your Keycloak server. Requires enableKeycloak: true. See auth.keycloak for more info
enableHeaderAuthbooleanOptionalIf 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
headerAuthobjectOptionalConfig options to point Dashy to your headers for authentication. Requires enableHeaderAuth: true. See auth.headerAuth for more info
enableGuestAccessbooleanOptionalWhen 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

appConfig.auth.users (optional)#

FieldTypeRequiredDescription
userstringRequiredUsername to log in with
hashstringRequiredA SHA-256 hashed password
typestringOptionalThe user type, either admin or normal

⬆️ Back to Top

appConfig.auth.keycloak (optional)#

FieldTypeRequiredDescription
serverUrlstringRequiredThe URL (or URL/ IP + Port) where your keycloak server is running
realmstringRequiredThe name of the realm (must already be created) that you want to use
clientIdstringRequiredThe Client ID of the client you created for use with Dashy
legacySupportbooleanOptionalIf using Keycloak 17 or older, then set this to true

⬆️ Back to Top

appConfig.auth.headerAuth (optional)#

FieldTypeRequiredDescription
userHeaderstringOptionalThe Header name which contains username (default: REMOTE_USER). Case insensitive
proxyWhitelistarrayRequiredAn array of Upstream proxy servers to expect authencticated requests from

⬆️ Back to Top

appConfig.webSearch (optional)#

FieldTypeRequiredDescription
disableWebSearchstringOptionalWeb search is enabled by default, but can be disabled by setting this property to true
searchEnginestringOptionalSet 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
customSearchEnginestringOptionalYou 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
openingMethodstringOptionalSet your preferred opening method for search results: newtab, sametab, workspace. Defaults to newtab
searchBangsobjectOptionalA 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=)

⬆️ Back to Top

appConfig.hideComponents (optional)#

FieldTypeRequiredDescription
hideHeadingbooleanOptionalIf set to true, the page title & sub-title will not be visible. Defaults to false
hideNavbooleanOptionalIf set to true, the navigation menu will not be visible. Defaults to false
hideSearchbooleanOptionalIf set to true, the search bar will not be visible. Defaults to false
hideSettingsbooleanOptionalIf set to true, the settings menu will be initially collapsed. Defaults to false
hideFooterbooleanOptionalIf set to true, the footer will not be visible. Defaults to false

⬆️ Back to Top

section#

FieldTypeRequiredDescription
namestringRequiredThe title for the section
iconstringOptionalAn single icon to be displayed next to the title. See section.icon
itemsarrayOptionalAn array of items to be displayed within the section. See item. Sections must include either 1 or more items, or 1 or more widgets.
widgetsarrayOptionalAn array of widgets to be displayed within the section. See widget
displayDataobjectOptionalMeta-data to optionally override display settings for a given section. See displayData

⬆️ Back to Top

section.item#

FieldTypeRequiredDescription
titlestringRequiredThe text to display/ title of a given item. Max length 18
descriptionstringOptionalAdditional info about an item, which is shown in the tooltip on hover, or visible on large tiles
urlstringRequiredThe URL / location of web address for when the item is clicked
iconstringOptionalThe icon for a given item. Can be a font-awesome icon, favicon, remote URL or local URL. See item.icon
targetstringOptionalThe 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
hotkeynumberOptionalGive frequently opened applications a numeric hotkey, between 0 - 9. You can then just press that key to launch that application.
tagsstring[]OptionalA list of tags, which can be used for improved search
statusCheckbooleanOptionalWhen 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
statusCheckUrlstringOptionalIf you've enabled statusCheck, and want to use a different URL to what is defined under the item, then specify it here
statusCheckHeadersobjectOptionalIf you're endpoint requires any specific headers for the status checking, then define them here
statusCheckAllowInsecurebooleanOptionalBy 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
statusCheckAcceptCodesstringOptionalIf 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
statusCheckMaxRedirectsnumberOptionalIf 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
colorstringOptionalAn 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
relstringOptionalThe value of the rel attribute for the link. Useful for specifying the relationship between the target link/document and Dashy. Defaults to noopener noreferrer
backgroundColorstringOptionalAn 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
providerstringOptionalThe name of the provider for a given service, useful for when including hosted apps. In some themes, this is visible under the item name
displayDataobjectOptionalMeta-data to optionally override display settings for a given item. See displayData

⬆️ Back to Top

item.displayData (optional)#

FieldTypeRequiredDescription
hideForUsersstring[]OptionalCurrent item will be visible to all users, except for those specified in this list
showForUsersstring[]OptionalCurrent item will be hidden from all users, except for those specified in this list
hideForGuestsbooleanOptionalCurrent item will be visible for logged in users, but not for guests (see appConfig.enableGuestAccess). Defaults to false
hideForKeycloakUsersobjectOptionalCurrent item will be visible to all keycloak users, except for those configured via these groups and roles. See hideForKeycloakUsers
showForKeycloakUsersobjectOptionalCurrent item will be hidden from all keycloak users, except for those configured via these groups and roles. See showForKeycloakUsers

⬆️ Back to Top

section.widgets (optional)#

FieldTypeRequiredDescription
typestringRequiredThe widget type. See Widget Docs for full list of supported widgets
optionsobjectOptionalSome widgets accept either optional or required additional options. Again, see the Widget Docs for full list of options
updateIntervalnumberOptionalYou can keep a widget constantly updated by specifying an update interval, in seconds. See Continuous Updates Docs for more info
useProxybooleanOptionalSome 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
timeoutnumberOptionalRequest timeout in milliseconds, defaults to ½ a second (500)
ignoreErrorsbooleanOptionalPrevent an error message being displayed, if a network request or something else fails. Useful for false-positives
labelstringOptionalAdd 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

section.displayData (optional)#

FieldTypeRequiredDescription
sortBystringOptionalThe 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
collapsedbooleanOptionalIf 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
cutToHeightbooleanOptionalBy default, sections will fill available space. Set this option to true to match section height with content height
rowsnumberOptionalHeight of the section, specified as the number of rows it should span vertically, e.g. 2. Defaults to 1. Max is 5.
colsnumberOptionalWidth of the section, specified as the number of columns the section should span horizontally, e.g. 2. Defaults to 1. Max is 5.
itemSizestringOptionalSpecify the size for items within this group, either small, medium or large. Note that this will override any settings specified through the UI
colorstringOptionalA custom accent color for the section, as a hex code or HTML color (e.g. #fff)
customStylesstringOptionalCustom CSS properties that should be applied to that section, e.g. border: 2px dashed #ff0000;
sectionLayoutstringOptionalSpecify which CSS layout will be used to responsively place items. Can be either auto (which uses flex layout), or grid. If grid is selected, then itemCountX and itemCountY may also be set. Defaults to auto
itemCountXnumberOptionalThe number of items to display per row / horizontally. If not set, it will be calculated automatically based on available space. Can only be set if sectionLayout is set to grid. Must be a whole number between 1 and 12
itemCountYnumberOptionalThe number of items to display per column / vertically. If not set, it will be calculated automatically based on available space. If itemCountX is set, then itemCountY can be calculated automatically. Can only be set if sectionLayout is set to grid. Must be a whole number between 1 and 12
hideForUsersstring[]OptionalCurrent section will be visible to all users, except for those specified in this list
showForUsersstring[]OptionalCurrent section will be hidden from all users, except for those specified in this list
hideForGuestsbooleanOptionalCurrent section will be visible for logged in users, but not for guests (see appConfig.enableGuestAccess). Defaults to false
hideForKeycloakUsersobjectOptionalCurrent section will be visible to all keycloak users, except for those configured via these groups and roles. See hideForKeycloakUsers
showForKeycloakUsersobjectOptionalCurrent section will be hidden from all keycloak users, except for those configured via these groups and roles. See showForKeycloakUsers

⬆️ Back to Top

section.icon and section.item.icon#

FieldTypeRequiredDescription
iconstringOptionalThe 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, and icons specified by URL

⬆️ Back to Top

section.displayData.hideForKeycloakUsers, section.displayData.showForKeycloakUsers, item.displayData.hideForKeycloakUsers and item.displayData.showForKeycloakUsers#

FieldTypeRequiredDescription
groupsstring[]OptionalCurrent Section or Item will be hidden or shown based on the user having any of the groups in this list
rolesstring[]OptionalCurrent Section or Item will be hidden or shown based on the user having any of the roles in this list

⬆️ Back to Top


Notes#

Editing Config through the UI#

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
Interactive Editor demo

JSON Editor
Config Editor demo

About YAML#

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.

Config Saving Methods#

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.

Preventing Changes#

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.

Example#

---pageInfo:  title: Home Labsections: # 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

- + \ No newline at end of file diff --git a/docs/contributing/index.html b/docs/contributing/index.html index e6733c0e..62e18fb7 100644 --- a/docs/contributing/index.html +++ b/docs/contributing/index.html @@ -6,7 +6,7 @@ Contributing | Dashy - + @@ -35,7 +35,7 @@ All content is located either in the ./README.md< Alicia Sykes's Website Alicia Sykes's Blog Alicia Sykes's PGP

If you like, you could also consider subscribing to my mailing list for occasional blog post updates.


Contributors#

For a full list of Dashy's contributors, see the Credits Page

Auto-generated contributors

Star-Gazers Over Time#

Stargazers

- + \ No newline at end of file diff --git a/docs/credits/index.html b/docs/credits/index.html index 0a6b2082..01d540b2 100644 --- a/docs/credits/index.html +++ b/docs/credits/index.html @@ -6,13 +6,13 @@ Credits | Dashy - +
-

Credits

Sponsors#

koconder
Vincent Koc
Admonstrator
Aaron Viehl
tbjers
Torgny Bjers
emlazzarin
Eddy Lazzarin
AnandChowdhary
Anand Chowdhary
shrippen
Null
bile0026
Zach Biles
UlisesGascon
Ulises Gascón
digitalarche
Digital Archeology
InDieTasten
Null
araguaci
Null
bmcgonag
Brian McGonagill
vlad-timofeev
Vlad Timofeev
helixzz
HeliXZz
patvdv
Patrick Van Der Veken
mryesiller
Göksel Yeşiller
allesauseinerhand
Null
forwardemail
Forward Email - Open-source & Privacy-focused Email Service (2023)
lamtrinhdev
LamTrinh.Dev
Bastii717
Null
M2TD
Null
frankdez93
Null
terminaltrove
Terminal Trove
NixyJuppie
Nixy
nrvo
Null

Contributors#

Lissy93
Alicia Sykes
liss-bot
Alicia Bot
marekful
Marcell Fülöp
EVOTk
EVOTk
snyk-bot
Snyk Bot
azerioxal
Kenneth Church
m42e
Matthias Bilger
CrazyWolf13
Tobias
pinarruiz
Alejandro Pinar Ruiz
imjimmeh
Null
kashif-se
Kashif Sohail
walkxcode
Walkx
josuablejeru
Josua Blejeru
evroon
Erik Vroon
jammo2k5
Jammo2k5
lordpansar
Magnus Sundström
Cereal916
Kristian Brasel
albcp
Alberto
altearius
Chris Nielsen
UrekD
UrekD
z3r0l1nk
Mihai
wozboz
Null
Totto16
Totto16
toddejohnson
Todd Johnson
remygrandin
Remygrandin
DimitriDR
Dimitri
Tracreed
David Alasow
aviolaris
Andreas Violaris
rubenandre
Rúben Silva
rtm516
Rtm516
zcq100
Null
onedr0p
ᗪєνιη ᗷυнʟ
stanly0726
Stanly0726
Bogyie
Bogyeong Kim
Tuzi555
Jakub Tuzar
berksmbl
Berk Sümbül
alucarddelta
Brent
k073l
Null
kt-alt
Null
mmihaly
Null
patrickheeney
Patrick Heeney
rokiden
Denis Kazimirov
zigotica
Sergi Meseguer
BySempron
Sergio
ssrangisetti
Null
itsmejoeeey
Joey Miller
hockeymikey
Null
thomaswienecke
Thomas Wienecke
deneor
Null
daentech
Dan Gilbert
moemoeq
CHAIYEON CHO
BOZG
Stephen Rigney
a-mnich
Alexander Mnich
alexdelprete
Alessandro Del Prete
alayham
Al-Ayham Saleh
sachahjkl
Sacha
shazzx
Shazz
ThinkSalat
Shawn Salat
royshreyaa
Null
Smexhy
Null
stavros-k
Stavros Kois
XenonR
Steffen Schmidt
StevKast
Steven Kast
AmadeusGraves
Ángel Fernández Sánchez
turnrye
Ryan Turner
rubjo
Null
PrynsTag
Prince Carl Velasco
PlusaN
Andrey
oka4shi
OKAMOTO Shigehiro
bubylou
Nicholas Malcolm
miclav
Michael Lavaire
imsakg
Mert Sefa AKGUN
ethan-hann
Ethan Hann
tazboyz16
Null
sur1v
José Ignacio
soaibsafi
Soaibuzzaman
pablomalo
Null
markxoe
Mark Oude Elberink
lxjv
Laker Turner
kxenoxx
Kxenox
jrobles98
Null
jnach
Jnach
imlonghao
Imlonghao
icy-comet
Aniket Teredesai
huangshaohuai
Null
flechaig
Null
dr460nf1r3
Nico
baifengheixi
Null
allozavrr
Null
XertDev
Xert
emiran-orange
Null
edugof
Eduardo Gomez
Dylan-Bs
Dylan Bersans
dougaldhub
Doug Lock
dkyeremeh
Desmond Kyeremeh
deepsourcebot
DeepSource Bot
DawidPietrykowski
Null
skaarj1989
David
clsty
Celestial.y
bskim45
Bumsoo Kim
BhasherBEL
Brieuc Dubois
BeginCI
Begin
BRAVO68WEB
Jyotirmoy Bandyopadhyaya [Bravo68]
AaronPorts
Artyom
alydemah
Aly Mohamed
5idereal
5idereal
0n1cOn3
0n1cOn3
maximemoreillon
Maxime Moreillon
Glitch3dPenguin
Max Kulik
markusdd
Markus Krause
asenov
Мирослав Асенов
luispabon
Luis Pabon
LeoColman
Leonardo Colman Lopes
KierenConnell
Kieren Connell
kieraneglin
Kieran
Singebob
Jeremy Chauvin
Hellhium
Jemy SCHNEPP
jjmung
JJ Munguia
b1thunt3r
Ishan Jain
Compunctus
Iaroslav Dronskii
nealian
Ian Neal
FraglyG
Hendrik Strydom
hubortje
Harald Töpfer
gbrown09
Garrett Brown
FormatToday
FormatToday

Helpful Users#

evotk
Evotk
shadowking001
LawrenceP.
urekd
Urekd
dylanbeme
DylanH
milesteg1
Milesteg1
lu4t
Lu4t

Bots#

liss-bot
Alicia Bot
snyk-bot
Snyk Bot
netlify
Netlify
viezly
Viezly
muse-dev
MuseDev
github-actions[bot]
github-actions[bot]
dependabot[bot]
dependabot[bot]

The above section is auto-generated, using contribute-list by @akhilmhdh.

Newest Stargazers#

Recent Star Gazers

Recent Forkers


Dependencies#

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. For a full breakdown of dependency licenses, please see Legal

Core#

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.

Utilities#

  • 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

Frontend Components#

Backup & Sync Server#

Although the app is purely frontend, there is an optional cloud sync feature, for off-site backup and restore. This is built as a serverless function on Cloudflare workers using KV and web crypto

External Services#

The 1-Click deploy demo uses Play-with-Docker Labs. Code is hosted on GitHub, Docker images are hosted on DockerHub, and the demos are hosted on Netlify.

Actions#

This repo makes heavy use of GitHub actions to run automated workflows. The following base actions are totally awesome, and have been extremely useful. Full credit to their respective authors


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! 💖

- +

Credits

Sponsors#

github
GitHub
koconder
Vincent Koc
Admonstrator
Aaron Viehl
tbjers
Torgny Bjers
emlazzarin
Eddy Lazzarin
AnandChowdhary
Anand Chowdhary
shrippen
Null
bile0026
Zach Biles
UlisesGascon
Ulises Gascón
digitalarche
Digital Archeology
InDieTasten
Null
araguaci
Null
bmcgonag
Brian McGonagill
vlad-timofeev
Vlad Timofeev
helixzz
HeliXZz
patvdv
Patrick Van Der Veken
plgonzalezrx8
Pedro Gonzalez
mryesiller
Göksel Yeşiller
allesauseinerhand
Null
forwardemail
Forward Email - Open-source & Privacy-focused Email Service (2023)
lamtrinhdev
LamTrinh.Dev
Bastii717
Null
getumbrel
Umbrel
M2TD
Null
frankdez93
Null
terminaltrove
Terminal Trove
NixyJuppie
Nixy
nrvo
Null

Contributors#

Lissy93
Alicia Sykes
liss-bot
Alicia Bot
marekful
Marcell Fülöp
EVOTk
EVOTk
snyk-bot
Snyk Bot
azerioxal
Kenneth Church
m42e
Matthias Bilger
CrazyWolf13
Tobias
pinarruiz
Alejandro Pinar Ruiz
imjimmeh
Null
kashif-se
Kashif Sohail
walkxcode
Walkx
josuablejeru
Josua Blejeru
evroon
Erik Vroon
jammo2k5
Jammo2k5
lordpansar
Magnus Sundström
Cereal916
Kristian Brasel
albcp
Alberto
altearius
Chris Nielsen
UrekD
UrekD
z3r0l1nk
Mihai
wozboz
Null
Totto16
Totto16
toddejohnson
Todd Johnson
remygrandin
Remygrandin
DimitriDR
Dimitri
Tracreed
David Alasow
aviolaris
Andreas Violaris
rubenandre
Rúben Silva
rtm516
Rtm516
zcq100
Null
onedr0p
ᗪєνιη ᗷυнʟ
stanly0726
Stanly0726
Bogyie
Bogyeong Kim
Tuzi555
Jakub Tuzar
berksmbl
Berk Sümbül
alucarddelta
Brent
k073l
Null
kt-alt
Null
mmihaly
Null
patrickheeney
Patrick Heeney
rokiden
Denis Kazimirov
zigotica
Sergi Meseguer
BySempron
Sergio
ssrangisetti
Null
itsmejoeeey
Joey Miller
hockeymikey
Null
thomaswienecke
Thomas Wienecke
deneor
Null
daentech
Dan Gilbert
moemoeq
CHAIYEON CHO
BOZG
Stephen Rigney
a-mnich
Alexander Mnich
alexdelprete
Alessandro Del Prete
alayham
Al-Ayham Saleh
sachahjkl
Sacha
shazzx
Shazz
ThinkSalat
Shawn Salat
royshreyaa
Null
Smexhy
Null
stavros-k
Stavros Kois
XenonR
Steffen Schmidt
StevKast
Steven Kast
AmadeusGraves
Ángel Fernández Sánchez
turnrye
Ryan Turner
rubjo
Null
PrynsTag
Prince Carl Velasco
PlusaN
Andrey
oka4shi
OKAMOTO Shigehiro
bubylou
Nicholas Malcolm
miclav
Michael Lavaire
imsakg
Mert Sefa AKGUN
ethan-hann
Ethan Hann
tazboyz16
Null
sur1v
José Ignacio
soaibsafi
Soaibuzzaman
pablomalo
Null
markxoe
Mark Oude Elberink
lxjv
Laker Turner
kxenoxx
Kxenox
jrobles98
Null
jnach
Jnach
imlonghao
Imlonghao
icy-comet
Aniket Teredesai
huangshaohuai
Null
flechaig
Null
dr460nf1r3
Nico
baifengheixi
Null
allozavrr
Null
XertDev
Xert
emiran-orange
Null
edugof
Eduardo Gomez
Dylan-Bs
Dylan Bersans
dougaldhub
Doug Lock
dkyeremeh
Desmond Kyeremeh
deepsourcebot
DeepSource Bot
DawidPietrykowski
Null
skaarj1989
David
clsty
Celestial.y
bskim45
Bumsoo Kim
BhasherBEL
Brieuc Dubois
BeginCI
Begin
BRAVO68WEB
Jyotirmoy Bandyopadhyaya [Bravo68]
AaronPorts
Artyom
alydemah
Aly Mohamed
5idereal
5idereal
0n1cOn3
0n1cOn3
maximemoreillon
Maxime Moreillon
Glitch3dPenguin
Max Kulik
markusdd
Markus Krause
asenov
Мирослав Асенов
luispabon
Luis Pabon
LeoColman
Leonardo Colman Lopes
KierenConnell
Kieren Connell
kieraneglin
Kieran
Singebob
Jeremy Chauvin
Hellhium
Jemy SCHNEPP
jjmung
JJ Munguia
b1thunt3r
Ishan Jain
Compunctus
Iaroslav Dronskii
nealian
Ian Neal
FraglyG
Hendrik Strydom
hubortje
Harald Töpfer
gbrown09
Garrett Brown
FormatToday
FormatToday

Helpful Users#

evotk
Evotk
shadowking001
LawrenceP.
urekd
Urekd
dylanbeme
DylanH
milesteg1
Milesteg1
lu4t
Lu4t

Bots#

liss-bot
Alicia Bot
snyk-bot
Snyk Bot
netlify
Netlify
viezly
Viezly
muse-dev
MuseDev
github-actions[bot]
github-actions[bot]
dependabot[bot]
dependabot[bot]

The above section is auto-generated, using contribute-list by @akhilmhdh.

Newest Stargazers#

Recent Star Gazers

Recent Forkers


Dependencies#

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. For a full breakdown of dependency licenses, please see Legal

Core#

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.

Utilities#

  • 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

Frontend Components#

Backup & Sync Server#

Although the app is purely frontend, there is an optional cloud sync feature, for off-site backup and restore. This is built as a serverless function on Cloudflare workers using KV and web crypto

External Services#

The 1-Click deploy demo uses Play-with-Docker Labs. Code is hosted on GitHub, Docker images are hosted on DockerHub, and the demos are hosted on Netlify.

Actions#

This repo makes heavy use of GitHub actions to run automated workflows. The following base actions are totally awesome, and have been extremely useful. Full credit to their respective authors


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! 💖

+ \ No newline at end of file diff --git a/docs/deployment/index.html b/docs/deployment/index.html index 53115199..8ee5b1bb 100644 --- a/docs/deployment/index.html +++ b/docs/deployment/index.html @@ -6,7 +6,7 @@ Deployment | Dashy - + @@ -15,7 +15,7 @@ Status: Docker Build StatusDocker PullsDocker StarsDocker Image SizeDocker Cloud Build

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 /root/my-local-conf.yml:/app/user-data/conf.yml \  --name my-dashboard \  --restart=always \  lissy93/dashy:latest

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 is
  • -v Specify volumes, to pass data from your host system to the container, in the format of [host-path]:[container-path], you can use this to pass your config file, directory of assets (like icons), custom CSS or web assets (like favicon.ico, manifest.json etc)
  • --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 This last option is the image the container should be built from, you can also use a specific version or architecture type, by replacing :latest with one of the tags

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

If you're deploying Dashy on a modern ARM-based board, such as a Raspberry Pi (2+), then you'll need to use one of Dashy's ARM images. Set the base image + tag to either lissy93/dashy:arm64v8 or lissy93/dashy:arm32v7, depending on your system architecture. You can also use the multi-arch image, which should work on all system architectures.

The image defaults to :latest, but you can instead specify a specific version, e.g. docker pull lissy93/dashy:release-1.5.0


Using Docker Compose#

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.

---version: "3.8"services:  dashy:    # To build from source, replace 'image: lissy93/dashy' with 'build: .'    # build: .    image: lissy93/dashy    container_name: Dashy    # Pass in your config file below, by specifying the path on your host machine    # volumes:      # - /root/my-config.yml:/app/user-data/conf.yml    ports:      - 4000:8080    # Set any environmental variables    environment:      - NODE_ENV=production    # Specify your user ID and group ID. You can find this by running `id -u` and `id -g`    #  - UID=1000    #  - GID=1000    # Specify restart policy    restart: unless-stopped    # Configure healthchecks    healthcheck:      test: ['CMD', 'node', '/app/services/healthcheck']      interval: 1m30s      timeout: 10s      retries: 3      start_period: 40s

You can use a different tag, by for example setting image: lissy93/dashy:arm64v8, or pull from GHCR instead by setting image: ghcr.io/lissy93/dashy.

If you are building from source, and would like to use one of the other Dockerfiles, then under services.dashy first set context: ., then specify the the path to the dockerfile, e.g. dockerfile: ./docker/Dockerfile-arm32v7


Kubernetes#

@vyrtualsynthese has written a Helm Chart for deploying with Kubernetes, available here


Unraid#

// TODO


Synology NAS#

Installing dashy is really simply and fast:

  1. Install Docker via Synology Package Center.

  2. 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.

  3. Go to Control Panel / Task Scheduler / Create / Scheduled Task / User-defined script.

  4. Once you click on User-defined script a new window will open.

  5. Follow the instructions below:

  6. General: In the Task field type in Install dashy. Uncheck "Enabled" option. Select root User.

  7. Schedule: Select Run on the following date then select "Do not repeat".

  8. 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/my-local-conf.yml:/app/user-data/conf.yml \  --name dashy \  --restart=always \  lissy93/dashy:latest

dashy should be up within 1-2min after you've started the install task procedure


Build from Source#

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

  1. Get Code: git clone https://github.com/Lissy93/dashy.git and cd dashy
  2. Configuration: Fill in you're settings in ./user-data/conf.yml
  3. Install dependencies: yarn
  4. Build: yarn build
  5. Run: yarn start

Deploy to cloud service#

If you don't have a home server, then fear not - Dashy can be deployed to pretty much any cloud provider. The above Docker and NPM guides will work exactly the same on a VPS, but I've also setup some 1-Click deploy links for 10+ of the most common cloud providers, to make things easier. Note that if your instance is exposed to the internet, it will be your responsibility to adequately secure it.

Some hosting providers required a bit of extra configuration, which was why I've made separate branches for deploying to those services (named: deploy_cloudflare, deploy_digital-ocean, deploy_platform-sh and deploy_render). If there's another cloud service which you'd like 1-click deployment to be supported for, feel free to raise an issue.

[!NOTE] If you use a static hosting provider, then status checks, writing new config changes to disk from the UI, and triggering a rebuild through the UI will not be available. This is because these features need endpoints provided by Dashy's local Node server. Everything else should work just the same though.

Netlify#

Deploy to Netlify

Netlify offers Git-based serverless cloud hosting for web applications. Their services are free to use for personal use, and they support deployment from both public and private repos, as well as direct file upload. The free plan also allows you to use your own custom domain or sub-domain, and is easy to setup.

To deploy Dashy to Netlify, use the following link

https://app.netlify.com/start/deploy?repository=https://github.com/lissy93/dashy

Heroku#

Deploy to Heroku

Heroku is a fully managed cloud platform as a service. You define app settings in a Procfile and app.json, which specifying how the app should be build and how the server should be started. Heroku costs a minimum of $5 for 1,000 dyno hours per month (there are around 720 hours in a month), and supports custom domains. Heroku's single-dyno service is not as quite performant as some other providers, and the app will have a short wake-up time when not visited for a while

To deploy Dashy to Heroku, use the following link

https://heroku.com/deploy?template=https://github.com/Lissy93/dashy

Vercel#

Deploy with Vercel

Vercel is a performance-focused platform for hosting static frontend apps. It comes bundled with some useful tools for monitoring and analyzing application performance and other metrics. Vercel is free for personal use, allows for custom domains and has very reasonable limits.

To deploy Dashy to Vercel, use the following link

https://vercel.com/new/project?template=https://github.com/lissy93/dashy

DigitalOcean#

Deploy to DO

DigitalOcean is a cloud service providing affordable developer-friendly virtual machines from $5/month. But they also have an app platform, where you can run web apps, static sites, APIs and background workers. CDN-backed static sites are free for personal use.

https://cloud.digitalocean.com/apps/new?repo=https://github.com/lissy93/dashy/tree/deploy_digital-ocean

Google Cloud Platform#

Run on Google Cloud

Cloud Run is a service offered by Google Cloud. It's a fully managed serverless platform, for developing and deploying highly scalable containerized applications. Similar to AWS and Azure, GCP offers a wide range of cloud services, which are billed on a pay‐per‐use basis, but Cloud Run has a free tier offering 180,000 vCPU-seconds, 360,000 GiB-seconds, and 2 million requests per month.

To deploy Dashy to GCP, use the following link

https://deploy.cloud.run/?git_repo=https://github.com/lissy93/dashy.git

Platform.sh#

Deploy to Platform.sh

Platform.sh is an end-to-end solution for developing and deploying applications. It is geared towards enterprise users with large teams, and focuses on allowing applications to scale up and down. Unlike the above providers, Platform.sh is not free, although you can deploy a test app to it without needing a payment method

To deploy Dashy to Platform.sh, use the following link

https://console.platform.sh/projects/create-project/?template=https://github.com/lissy93/dashy

Render#

Deploy to Render

Render is cloud provider that provides easy deployments for static sites, Docker apps, web services, databases and background workers. Render is great for developing applications, and very easy to use. Static sites are free, and services start at $7/month. Currently there are only 2 server locations - Oregon, USA and Frankfurt, Germany. For more info, see the Render Docs

To deploy Dashy to Render, use the following link

https://render.com/deploy?repo=https://github.com/lissy93/dashy/tree/deploy_render

Railway#

Deploy on Railway

Railway is a Platform as a Service (PaaS) that offers a complete platform for building and delivering programs to the backend of the cloud. You bring your code and Railway does the rest. Railway offers an extremely good developer experience and makes it effortless to deploy apps. Railway offers a free Trial Plan, and paid plans start at $5/month. Railway has 4 server locations: US-west, US-east, EU-west and ASIA-South-East.

https://railway.app/template/MtdjAQ

Scalingo#

Deploy on Scalingo

Scalingo is a scalable container-based cloud platform as a service. It's focus is on compliance and uptime, and is geared towards enterprise users. Scalingo is also not free, although they do have a 3-day free trial that does not require a payment method.

To deploy Dashy to Scalingo, use the following link

https://my.scalingo.com/deploy?source=https://github.com/lissy93/dashy#master

Play-with-Docker#

Try in PWD

Play with Docker is a community project by Marcos Liljedhal and Jonathan Leibiusky and sponsored by Docker, intended to provide a hands-on learning environment. Their labs let you quickly spin up a Docker container or stack, and test out the image in a temporary, sandboxed environment. There's no need to sign up, and it's completely free.

To run Dashy in PWD, use the following URL:

https://labs.play-with-docker.com/?stack=https://raw.githubusercontent.com/Lissy93/dashy/master/docker-compose.yml

Surge.sh#

Follow instructions below

Surge.sh is quick and easy static web publishing platform for frontend-apps. Surge supports password-protected projects. You can also add a custom domain and then force HTTPS by default and optionally set a custom SSL certificate

To deploy Dashy to Surge.sh, first clone and cd into Dashy, install dependencies, and then use the following commands

yarn add -g surgeyarn buildsurge ./dist

Easypanel#

Deploy to Easypanel

Easypanel it's a modern server control panel. You can use it to deploy Dashy on your own server.

To deploy Dashy to Easypanel, use the following link

https://easypanel.io/docs/templates/dashy

Hosting with CDN#

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, triggering a rebuild through the UI and application status checks. Everything else will work fine.


Requirements#

System Requirements#

Dashy works well on a Raspberry Pi (tested on Pi 3 and later), but should also run well on any system.

Docker#

Initial app build causes a spike in resource usage, but once the built app is running it is fairly steady. For this reason, Dashy works best with a minimum of 1GB of memory, and 1GB of disk space.

Bare Metal#

Minimum 526mb mem, 2GB disk space.

CDN / Cloud Deploy#

No specific requirements. The built application alone (without the Node server) is very light-weight, and can be handled smoothly by pretty much any CDN or cloud deployment service (see this list or natively supported cloud providers).

If you're using your own icons, or other assets, additional disk space will be required for those resources.

Browser Support#

JavaScript is required to run Dashy.

In terms of browser support, pretty much any browser released since 2018 should render content just fine. However, for Internet Explorer, only IE11+ is supported, yet performance here is still not optimal. The recommended browser is either a Chromium-based / Webkit browser (Chrome, Brave, Vivaldi, Edge, Yandex, etc), or Firefox or one of it's forks (FF-ESR, Tor, LibreWolf, etc). Recent versions of Safari and Opera are also supported, but with limited continuous testing.

- + \ No newline at end of file diff --git a/docs/developing/index.html b/docs/developing/index.html index 5af9f9c2..a3c0ed1f 100644 --- a/docs/developing/index.html +++ b/docs/developing/index.html @@ -6,7 +6,7 @@ Developing | Dashy - + @@ -14,7 +14,7 @@

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.

Setting up the Dev Environment#

Prerequisites#

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).

Running the Project#

  1. Get Code: git clone https://github.com/Lissy93/dashy.git
  2. Navigate into the directory: cd dashy
  3. Install dependencies: yarn
  4. 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.

Project Commands#

Basics#

  • yarn build - In the interest of speed, the application is pre-compiled, this means that the config file is read during build-time, and therefore the app needs to rebuilt for any new changes to take effect. Luckily this is very straight forward. Just run yarn build or docker exec -it [container-id] yarn build
  • yarn start - Starts a web server, and serves up the production site from ./dist (must run build command 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-and-start - Builds the app, runs checks and starts the production server. Commands are run in parallel, and so is faster than running them in independently. Uses the yarn build and yarn start commands
  • yarn build-watch - If you find yourself making frequent changes to your configuration, and do not want to have to keep manually rebuilding, then this option is for you. It will watch for changes to any files within the projects root, and then trigger a rebuild. Note that if you are developing new features, then yarn dev would be more appropriate, as it's significantly faster at recompiling (under 1 second), and has hot reloading, linting and testing integrated
  • 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
  • You can manage the app using the Vue-CLI Service, with npx vue-cli-service [command]. Or to start the Vue Management UI, run npx vue ui, and open http://localhost:8000

Environmental Variables#

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
  • VUE_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
  • VUE_APP_VERSION - Again, set automatically using package.json during build time
  • BACKUP_DIR - Directory for conf.yml backups

Environment Modes#

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.


Git Strategy#

Git Flow#

Like most Git repos, we are following the Github Flow standard.

  1. Create a branch (or fork if you don'd have write access)
  2. Code some awesome stuff 🧑‍💻
  3. Add, commit and push your changes to your branch/ fork
  4. Head over to GitHub and create a Pull Request
  5. Fill in the required sections in the template, and hit submit
  6. Follow up with any reviews on your code
  7. Merge 🎉

Git Branch Naming#

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

Commit Emojis#

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

PR Guidelines#

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!

Resources for Beginners#

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:

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).


App Info#

Style Guide#

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:


Application Structure#

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│  │  ├── EditSiteMeta.vue        # Form where the user can edit site meta data│  │  ├── JsonEditor.vue          # JSON editor, where the user can modify the main config file│  │  ╰── RebuildApp.vue          # A component allowing user to trigger a rebuild through the UI│  ├── 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.json         # 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│  ├── 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                  # TAuthentication page   ├── Minimal.vue                # The minimal view   ╰── Workspace.vue              # The workspace view with apps in sidebar

Visualisation of Source Directory#

File Breakdown


Development Tools#

Performance - Lighthouse#

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.

Dependencies - BundlePhobia#

BundlePhobia is a really useful app that lets you analyze the cost of adding any particular dependency to an application


Notes#

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.

- + \ No newline at end of file diff --git a/docs/development-guides/index.html b/docs/development-guides/index.html index 70effd18..d36d308b 100644 --- a/docs/development-guides/index.html +++ b/docs/development-guides/index.html @@ -6,7 +6,7 @@ Development Guides | Dashy - + @@ -23,7 +23,7 @@ Then, from within the root of Dashy's directory, start the server, by runni export default { mixins: [WidgetMixin], data() { return { results: null, }; }, computed: { endpoint() { return `${widgetApiEndpoints.myApi}/something`; }, }, methods: { fetchData() { this.makeRequest(this.endpoint).then(this.processData); }, processData(data) { // Do processing any here, and set component data this.results = data; }, },};</script> <style scoped lang="scss"></style>

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

Step 2 - Adding Functionality#

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/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 <template> to render results to the user. Again, how you do this will depend on the structure of your data, and what you want to display, but at it's simplest, it might look something like this:

<p class="results">{{ myResults }}</p>

Styling#

Styles can be written for your widget within the <style> block.

There are several color variables used by widgets, which extend from the base palette. Using these enables users to override colors to theme their dashboard, if they wish. The variables are: --widget-text-color, --widget-background-color and --widget-accent-color

<style scoped lang="scss">p.results {  color: var(--widget-text-color);}</style>

For examples of finished widget components, see the Widgets directory. Specifically, the XkcdComic.vue widget is quite minimal, so would make a good example, as will this example implementation.

Step 3 - Register#

Next, register your new widget in WidgetBase.vue. In this file, you'll need to add the following:

const COMPAT = {  ...  'example-widget': 'ExampleWidget',};

Here, the example-widget property name will be used to identify the widget when parsing the type property in a configuration file. The ExampleWidget string is used to dynamically import the widget, and therefore must match the widget's filename as it exists in the components/widgets folder.

Step 4 - Docs#

Finally, add some documentation for your widget in the Widget Docs, so that others know how to use it. Include the following information: Title, short description, screenshot, config options and some example YAML.

Summary: For a complete example of everything discussed here, see: 3da76ce


Respecting Config Permissions#

Any screen that displays part or all of the users config, must not be shown when the user has disabled viewing config.

This can be done by checking the allowViewConfig attribute of the permissions getter, in the store. First create a new computed property, like:

allowViewConfig() {  return this.$store.getters.permissions.allowViewConfig;},

Then wrap the part of your UI which displays config with: v-if="allowViewConfig"

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.

- + \ No newline at end of file diff --git a/docs/icons/index.html b/docs/icons/index.html index 74ecfb05..0d9481d7 100644 --- a/docs/icons/index.html +++ b/docs/icons/index.html @@ -6,13 +6,13 @@ Icons | Dashy - +

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.


Favicons#

Dashy can auto-fetch an icon for a given service, using it's favicon. Just set icon: favicon to use this feature.

Since different websites host their favicons at different paths, for the best results Dashy can use an API to resolve a websites icon.

The default favicon API is allesedv.com, but you can change this under appConfig.faviconApi. If you'd prefer not to use an API, just set this value to local. You can also use different APIs for individual items, by setting icon: favicon-[api], e.g. favicon-clearbit.

The following favicon APIs are supported:

  • allesedv - allesedv.com is a highly efficient IPv6-enabled service
  • iconhorse - Icon.Horse returns quality icons for any site, with caching for speed and fallbacks for sites without an icon
  • clearbit - Clearbit returns high-quality square logos from mainstream websites
  • faviconkit - faviconkit.com good quality icons and most sites supported (Note: down as of Nov '21)
  • besticon - BestIcon fetches websites icons from manifest
  • mcapi - MC-API fetches default website favicon, originally a Minecraft util
  • duckduckgo - Returns decent quality website icons, from DuckDuckGo search
  • google - Official Google favicon API service, good support for all sites, but poor quality
  • yandex - Lower quality icons, but useful in some regions where other services are blocked
  • local - Set to local to fetch the default icon at /favicon.ico instead of using an API

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, icon: https://monitoring.local/faviconx128.png- you can find this path using the browser dev tools.


Font Awesome#

You can use any Font Awesome Icon simply by specifying it's identifier. This is in the format of [category] [name] and can be found on the page for that icon on the Font Awesome site. For example: fas fa-rocket, fab fa-monero or fas fa-unicorn.

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: appConfig.fontAwesomeKey. This is usually a 10-digit string, for example 13014ae648.


Simple Icons#

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 website, then just set your icon the the simple icon slug, prefixed with si-.

For example:

sections:- name: Simple Icons Example  items:  - title: Portainer    icon: si-portainer  - title: FreeNAS    icon: si-freenas  - title: NextCloud    icon: si-nextcloud  - title: Home Assistant    icon: si-homeassistant

Generative Icons#

To uses a unique and programmatically generated icon for a given service just set 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 DiceBear (or Evatar for fallback), and use a hash of the services domain/ ip for entropy, so each domain will have a unique icon.


Emoji Icons#

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. 'U+1F680') or shortcode (e.g. ':rocket:'). You can find these codes for any emoji using Emojipedia (near the bottom of emoji each page), or for a quick reference to emoji shortcodes, check out emojis.ninja by @nomanoff.

For example, these will all render the same rocket (🚀) emoji: icon: ':rocket:' or icon: 'U+1F680' or icon: 🚀


Home-Lab Icons#

The dashboard-icons repo by @WalkxCode provides a comprehensive collection of 360+ high-quality PNG 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 hl-. See here for a full list of all available icons. Note that these are fetched and cached strait from GitHub, so if you require offline access, the Local Icons method may be a better option for you.

For example:

sections:- name: Home Lab Icons Example  items:  - title: AdGuard Home    icon: hl-adguardhome  - title: Long Horn    icon: hl-longhorn  - title: Nagios    icon: hl-nagios  - title: Whoogle Search    icon: hl-whooglesearch


Material Design Icons#

Dashy also supports 5000+ material-design-icons. To use these, first find the name/ slug for your icon here, and then prefix is with mdi-.

For example:

sections:- name: Material Design Icons Example  items:  - title: Alien Icon    icon: mdi-alien  - title: Fire Icon    icon: mdi-fire  - title: Dino Icon    icon: mdi-google-downasaur


Icons by URL#

You can also set an icon by passing in a valid URL pointing to the icons location. For example 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.


Local Icons#

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 ./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: -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 /app/user-data/item-icons/maltrail.png, then I would just set icon: maltrail.png.

You can also use sub-folders within the item-icons directory to keep things organized. You would then specify an icon with it's folder name slash image name. For example: networking/monit.png


Default Icon#

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 appConfig.defaultIcon.


No Icon#

If you don't wish for a given item or section to have an icon, just leave out the icon attribute.


Icon Collections and Resources#

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 .svg or .png) and paste it as your icon, or download and save the icons in /user-data/item-icons / mapped Docker volume. Full credit to the authors, please see the licenses for each service for usage and copyright information.

If you are a student, then you can get free access to premium icons on Icon Scout or Icons8 using the GitHub Student Pack.


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.

- + \ No newline at end of file diff --git a/docs/index.html b/docs/index.html index a8aed244..fb91c447 100644 --- a/docs/index.html +++ b/docs/index.html @@ -6,13 +6,13 @@ Dashy | Dashy - +
- + \ No newline at end of file diff --git a/docs/license/index.html b/docs/license/index.html index 7031d437..e2604e4b 100644 --- a/docs/license/index.html +++ b/docs/license/index.html @@ -6,7 +6,7 @@ license | Dashy - + @@ -16,7 +16,7 @@ Permission is hereby granted, free of charge, to any person obtaining a copyof this software and associated documentation files (the "Software"), to dealin the Software without restriction, including without limitation the rightsto use, copy, modify, merge, publish, distribute, sublicense, and/or sellcopies of the Software, and to permit persons to whom the Software isfurnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in allcopies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS ORIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THEAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHERLIABILITY, 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 THESOFTWARE.
- + \ No newline at end of file diff --git a/docs/management/index.html b/docs/management/index.html index 6673280b..3194738f 100644 --- a/docs/management/index.html +++ b/docs/management/index.html @@ -6,7 +6,7 @@ App Management | Dashy - + @@ -31,7 +31,7 @@ Similarly, never expose /var/run/docker.sock to other containers as For example: scp -r ./dist/* [username]@[server_ip]:/var/www/dashy/html

Apache#

Copy Dashy's dist folder to your apache server, sudo cp -r ./dashy/dist /var/www/html/dashy.

In your Apache config, /etc/apche2/apache2.conf add:

<Directory /var/www/html>    Options Indexes FollowSymLinks    AllowOverride All    Require all granted</Directory>
 <IfModule mod_rewrite.c>  RewriteEngine On  RewriteBase /  RewriteRule ^index\.html$ - [L]  RewriteCond %{REQUEST_FILENAME} !-f  RewriteCond %{REQUEST_FILENAME} !-d  RewriteRule . /index.html [L]</IfModule>

Add a .htaccess file within /var/www/html/dashy/.htaccess, and add:

Options -MultiViewsRewriteEngine OnRewriteCond %{REQUEST_FILENAME} !-fRewriteRule ^ index.html [QSA,L]

Then restart Apache, with sudo systemctl restart apache2

Caddy#

Caddy v2

try_files {path} /

Caddy v1

rewrite {  regexp .*  to {path} /}

Firebase Hosting#

Create a file names firebase.json, and populate it with something similar to:

{  "hosting": {    "public": "dist",    "rewrites": [      {        "source": "**",        "destination": "/index.html"      }    ]  }}

cPanel#

  1. Login to your WHM
  2. Open 'Feature Manager' on the left sidebar
  3. Under 'Manage feature list', click 'Edit'
  4. Find 'Application manager' in the list, enable it and hit 'Save'
  5. Log into your users cPanel account, and under 'Software' find 'Application Manager'
  6. Click 'Register Application', fill in the form using the path that Dashy is located, and choose a domain, and hit 'Save'
  7. The application should now show up in the list, click 'Ensure dependencies', and move the toggle switch to 'Enabled'
  8. If you need to change the port, click 'Add environmental variable', give it the name 'PORT', choose a port number and press 'Save'.
  9. Dashy should now be running at your selected path an on a given port

⬆️ Back to Top


Running a Modified Version of the App#

If you'd like to make any code changes to the app, and deploy your modified version, this section briefly explains how.

The first step is to fork the project on GitHub, and clone it to your local system. Next, install the dependencies (yarn), and start the development server (yarn dev) and visit 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 yarn build will build the app for production, and output the assets into ./dist which can then be deployed using a web server, CDN or the built-in Node server with yarn start. For more info on all of this, take a look at the Developing Docs. To build your own Docker container from the modified app, see Building your Own Container

⬆️ Back to Top


Building your Own Container#

Similar to above, you'll first need to fork and clone Dashy to your local system, and then install dependencies.

Then, either use Dashy's default Dockerfile as is, or modify it according to your needs.

To build and deploy locally, first build the app with: docker build -t dashy ., and then start the app with docker run -p 8080:8080 --name my-dashboard dashy. Or modify the docker-compose.yml file, replacing image: lissy93/dashy with build: . and run docker compose up.

Your container should now be running, and will appear in the list when you run docker container ls –a. If you'd like to enter the container, run docker exec -it [container-id] /bin/ash.

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. 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 7330339c..5dff4797 100644 --- a/docs/multi-language-support/index.html +++ b/docs/multi-language-support/index.html @@ -6,7 +6,7 @@ Internationalization | Dashy - + @@ -16,7 +16,7 @@ If your language is a specific dialect or regional translation, then use the Pos 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, 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.


Adding New Text to a Component#

If you're working on a new component, then any text that is displayed to the user should be extracted out of the component, and stored in the file. This also applies to any existing components, that might have been forgotten to be translated. Thankfully, everything is already setup, so this is a pretty easy job.

1. Add Translated Text#

Firstly, go to ./src/assets/locales/en.json, and either find the appropriate section, or create a new section. Lets say you're new component is called my-widget, you could add "my-widget": {} to store all your text as key-value-pairs. E.g.

"my-widget": {    "awesome-text": "I am some text, that will be seen by the user"}

Note that you must add English translations for all text. Missing languages are not a problem, as they will always fallback to English, but if the English is missing, then nothing can be displayed.

2. Use Text within Component#

Once your text is in the translation file, you can now use it within your component. There is a global $t function, with takes the key of your new translation, and returns the value. For example:

<p>{{ $t('my-widget.awesome-text') }}</p>

Note that the {{ }} just tells Vue that this is JavaScript/ dynamic. This will render: <p>I am some text, that will be seen by the user</p>

If you need to display text programmatically, from within the components JavaScript (e.g. in a toast popup), then use this.$t. For example: alert(this.$t('my-widget.awesome-text')).

You may also need to pass a variable to be displayed within a translation. Vue I18n supports Interpolations using mustache-like syntax.

For example, you would set your translation to:

{    "welcome-message": "Hello {name}!"}

And then pass that variable (name) within a JSON object as the second parameter on $t, like:

$t('welcome-message', { name: 'Alicia' })

Which will render:

Hello Alicia!

There are many other advanced features, including Pluralization, Datetime & Number Formatting, Message Support and more, all of which are outlined in the Vue-i18n Docs.

Basic Example#

Using the search bar as an example, this would look something like:

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",  },  ...}
- + \ No newline at end of file diff --git a/docs/pages-and-sections/index.html b/docs/pages-and-sections/index.html index bee82aa9..9044e3cb 100644 --- a/docs/pages-and-sections/index.html +++ b/docs/pages-and-sections/index.html @@ -6,14 +6,14 @@ Pages and Sections | Dashy - +

Pages and Sections

Multi-Page Support#

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.

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.

Using Local Sub-Pages#

To get started, create a new .yml config file for your sub-page, placing it within /app/public. Then within your primary conf.yml, choose a name, and specify the path to the new file.

For example:

pages:- name: Networking Services  path: 'networking.yml'- name: Work Stuff  path: 'work.yml'

If you're sub-page is located within /app/public, then you only need to specify the filename, but if it's anywhere else, then the full path is required.

Using Remote Sub-Pages#

Config files don't need to be local, you can store them anywhere, and data will be imported as sub-pages on page load.

For example:

pages:- name: Getting Started  path: 'https://snippet.host/tvcw/raw'- name: Homelab  path: 'https://snippet.host/tetp/raw'- name: Browser Startpage  path: 'https://snippet.host/zcom/raw'

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 snippet.host, which supports never-expiring CORS-enabled pastes, which can also be edited later.

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. The 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.

The following example shows creating a config, publishing it as a Gist, copying the URL to the raw file, and using it within your dashboard.

Public config in a gist demo

Restrictions#

Only top-level fields supported by sub-pages are pageInfo and sections. The appConfig and pages will always be inherited from your main 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.

Note that since page paths are required by the router, they are set at build-time, not run-time, and so a rebuild (happens automatically) is required for changes to page paths to take effect (this only applies to changes to the pages array, rebuild isn't required for editing page content).

Sub-Items#

A normal section will contain zero or more items, for example:

- 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, or for action buttons (coming soon).

Item groups may also have an optional title.

- name: Coding  icon: far fa-code  items:  - title: Normal Item 1  - title: Normal Item 2  - subItems:    - title: JavaScript      url: https://developer.mozilla.org      icon: si-javascript    - title: TypeScript      url: https://www.typescriptlang.org/docs      icon: si-typescript    - title: Svelt      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 cff01f38..a6f5b52d 100644 --- a/docs/privacy/index.html +++ b/docs/privacy/index.html @@ -6,14 +6,14 @@ Privacy & Security | Dashy - +

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, by keeping your local services organized and accessible from a single place. The management docs contains a though guide on the steps you can take to secure your homelab.

Dashy operates on the premise, that no external data requests should ever be made, unless explicitly enabled by the user. In the interest of transparency, the code is 100% open source and clearly documented throughout.

🔐 For privacy and security tips, check out another project of mine: Personal Security Checklist

Contents#


External Requests#

By default, Dashy will not make any external requests, unless you configure it to. Some features (which are off by default) do require internat access, and this section outlines those features, the services used, and links to their privacy policies.

The following section outlines all network requests that are made when certain features are enabled.

Icons#

Font Awesome#

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 appConfig.enableFontAwesome to true / false. Requests are made directly to Font-Awesome CDN, for more info, see the Font Awesome Privacy Policy.

Material Design Icons#

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 appConfig.enableMaterialDesignIcons to true / false. Requests are made directly to Material-Design-Icons CDN, for more info, see the Material Design Icons Website.

Favicon Fetching#

If an item's icon is set to favicon, then it will be auto-fetched from the corresponding URL. Since not all websites have their icon located at /favicon.ico, and if they do, it's often very low resolution (like 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.

The default favicon API is allesedv.com, but this can be changed by setting appConfig.faviconApi to an alternate source (iconhorse, clearbit, faviconkit, besticon, duckduckgo, google and allesedv are supported). If you do not want to use any API, then you can set this property to local, and the favicon will be fetched from the default path. For hosted services, this will still incur an external request.

Generative Icons#

If an item has the icon set to generative, then an external request it made to 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 Dicebear Privacy Policy

As a fallback, if Dicebear fails, then Evatar is used.

Other Icons#

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 ./public/item-icons (which can be mounted as a volume if you're using Docker).

Web Assets#

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.


Features#

Status Checking#

The status checking feature allows you to ping your apps/ services to check if they are currently operational.

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.

Update Checks#

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 appConfig.disableUpdateChecks: true

Cloud Backup#

Dashy has an optional End-to-End encrypted cloud backup feature. No data is ever transmitted unless you actively enable this feature through the UI.

All data is encrypted before being sent to the backend. 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.

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.

Web Search#

Dashy has a primitive 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.

This feature can be disabled under appConfig, with webSearch: { disableWebSearch: true }

Anonymous Error Reporting#

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. 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).

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 ae automatically deleted after 14 days. For more about privacy and security, see the Sentry Docs.

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 appConfig.enableErrorReporting: true.

If you need to monitor bugs yourself, then you can self-host your own Sentry Server, and use it by setting appConfig.sentryDsn to your Sentry instances Data Source Name, then just enable error reporting in Dashy.


Themes#

Certain themes may use external assets (such as fonts or images). Currently, this only applies the Adventure theme.


Widgets#

Dashy supports 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.


Browser Storage#

In order for user preferences to be persisted between sessions, certain data needs to be stored in the browsers local storage. No personal info is kept here, none of this data can be accessed by other domains, and no data is ever sent to any server without your prior consent.

You can view and delete stored data by opening up the dev tools: F12 --> Application --> Storage.

The following section outlines all data that is stored in the browsers, as cookies, session storage or local storage.

Cookies#

Cookies will expire after their pre-defined lifetime

  • AUTH_TOKEN - A unique token, generated from a hash of users credentials, to verify they are authenticated. Only used when auth is enabled.

Session Storage#

Session storage is deleted when the current session ends (tab / window is closed)

  • SW_STATUS - The current status of any service workers
  • ERROR_LOG - List of recent errors

Local Storage#

Local storage is persisted between sessions, and only deleted when manually removed

  • LANGUAGE - The locale to show app text in
  • HIDE_INFO_NOTIFICATION - Set to true once user dismissed welcome message, so that it's not shown again
  • LAYOUT_ORIENTATION - Preferred section layout, either horizontal, vertical or auto
  • COLLAPSE_STATE - Remembers which sections are collapsed
  • ICON_SIZE - Size of items, either small, medium or large
  • THEME - Users applied theme
  • CUSTOM_COLORS - Any color modifications made to a given theme
  • BACKUP_ID - If a backup has been made, the ID is stored here
  • BACKUP_HASH - A unique hash of the previous backups meta data
  • HIDE_SETTINGS - Lets user hide or show the settings menu
  • USERNAME - If user logged in, store username. Only used to show welcome message, not used for auth
  • CONF_SECTIONS - Array of sections, only used when user applies changes locally
  • PAGE_INFO - Config page info, only used when user applies changes locally
  • APP_CONFIG - App config, only used when user applies changes locally
  • MOST_USED - If smart sort is used to order items by most used, store open count
  • LAST_USED - If smart sort is used to order items by last used, store timestamps

Deleting Stored Data#

You can manually view and delete session storage, local storage and cookies at anytime. Fist open your browsers developer tools (usually 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.


Dependencies#

As with most web projects, Dashy relies on several dependencies. For links to each, and a breakdown of their licenses, please see Legal.

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 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.

Note that packages listed under devDependencies section are only used for building the project, and are not included in the production environment.


Securing your Environment#

Running your self-hosted applications in individual, containerized environments (such as containers or VMs) helps keep them isolated, and prevent an exploit in one service effecting another.

If you're running Dashy in a container, see Management Docs --> Container Security for step-by-step security guide.

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 and employ monitoring for all your self-hosted apps. A couple of things that you should look at include:

  • Use SSL for securing traffic in transit
  • Configure authentication to prevent unauthorized access
  • Keep your system, software and Dashy up-to-date
  • Ensure your server is appropriately secured
  • Manage users and SSH correctly
  • Enable and configure firewall rules
  • Implement security, malware and traffic scanning
  • Setup malicious traffic detection
  • Understand the Docker attack fronts, and follow Docker Security Best Practices

This is covered in more detail in App Management.


Security Features#

Subresource Integrity#

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.

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 INTEGRITY environmental variable to true.

SSL#

Native SSL support is enabled, for setup instructions, see the Management Docs

Authentication#

Dashy supports both basic auth, as well as server-based SSO using Keycloak. Full details of which, along with alternate authentication methods can be found in the 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 or another server-side method.


Disabling Features#

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.

  • To disable smart-sort (uses local storage), set appConfig.disableSmartSort: true
  • To disable update checks (makes external request to GH), set appConfig.disableUpdateChecks: true
  • To disable web search (redirect to external / internal content), set appConfig.disableWebSearch: true
  • To keep status checks disabled (external/ internal requests), set appConfig.statusCheck: false
  • To keep font-awesome icons disabled (external requests), set appConfig.enableFontAwesome: false
  • To keep error reporting disabled (external requests and data collection), set appConfig.enableErrorReporting: false
  • To keep the service worker disabled (stores cache of app in browser data), set appConfig.enableServiceWorker: false

Reporting a Security Issue#

If you think you've found a critical issue with Dashy, please send an email to security@mail.alicia.omg.lol. You can encrypt it, using 0688 F8D3 4587 D954 E9E5 1FB8 FEDB 68F5 5C02 83A7. You should receive a response within 48 hours. For more information, see SECURITY.md.

All non-critical issues can be raised as a ticket.

Please include the following information:

  • Type of issue (e.g. buffer overflow, SQL injection, cross-site scripting, etc.)
  • Full paths of source file(s) related to the manifestation of the issue
  • The location of the affected source code (tag/branch/commit or direct URL)
  • Any special configuration required to reproduce the issue
  • Step-by-step instructions to reproduce the issue
  • Proof-of-concept or exploit code (if possible)
  • Impact of the issue, including how an attacker might exploit the issue
- + \ No newline at end of file diff --git a/docs/quick-start/index.html b/docs/quick-start/index.html index 7648f52b..dcd0b921 100644 --- a/docs/quick-start/index.html +++ b/docs/quick-start/index.html @@ -6,16 +6,17 @@ Quick Start | Dashy - +

Quick Start

Welcome to Dashy! So glad you're here 😊 In a couple of minutes, you'll have your new dashboard up and running 🚀

TLDR; Run docker run -p 8080:8080 lissy93/dashy, then open http://localhost:8080


1. Prerequisites#

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 Docker Documentation. -If 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 Deployment Docs.


2. Installation#

To pull the latest image, and build and start the app run:

docker run -d \  -p 8080:8080 \  -v ~/my-conf.yml:/app/user-data/conf.yml \  --name my-dashboard \  --restart=always \  lissy93/dashy:latest

Either replace the -v path to point to your config file, or leave it out. For a full list of available options, then see Dashy with Docker Docs. If you'd prefer to use Docker Compose, then see Dashy with Docker Compose Docs. Alternate registries, architectures and pinned versions are also supported.

Your dashboard should now be up and running at http://localhost:8080 (or your servers IP address/ domain, and the port that you chose). The first time you build, it may take a few minutes.


3. Configure#

Now that you've got Dashy running, you are going to want to set it up with your own content. +If 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 Deployment Docs.


2. Installation#

To pull the latest image, and build and start the app run:

docker run -d \  -p 8080:8080 \  -v ~/my-conf.yml:/app/user-data/conf.yml \  --name my-dashboard \  --restart=always \  lissy93/dashy:latest

Either replace the -v path to point to your config file, or leave it out. For a full list of available options, then see Dashy with Docker Docs. If you'd prefer to use Docker Compose, then see Dashy with Docker Compose Docs. Alternate registries, architectures and pinned versions are also supported.

Your dashboard should now be up and running at http://localhost:8080 (or your servers IP address/ domain, and the port that you chose). The first time you build, it may take a few minutes.


3. User Data Directory#

Your config file should be placed inside user-data/ (in Docker, that's /app/user-data/).

This directory can also contain some optional assets you wish to use within your dashboard, like icons, fonts, styles, scripts, etc.

Any files placed here will be served up to the root of the domain, and override the contents of public/. +For example, if you had user-data/favicon.ico this would be accessible at http://my-dashy-instance.local/favicon.ico

Example Files in user-data:

  • conf.yml - This is the only file that is compulsary, it's your main Dashy config
  • **.yml - Include more config files, if you'd like to have multiple pages, see Multi-page support for docs
  • favicon.ico - The default favicon, shown in the browser's tab title
  • initialization.html - Static HTML page displayed before the app has finished compiling, see public/initialization.html
  • robots.txt - Search engine crawl rules, override this if you want your dashboard to be indexable
  • manifest.json - PWA configuration file, for installing Dashy on mobile devices
  • index.html - The main index page which initializes the client-side app, copy it from /public/index.html
  • **.html - Write your own HTML pages, and access them at http://my-dashy-instance.local/my-page.html
  • fonts/ - Custom fonts (be sure to include the ones already in public/fonts
  • item-icons/ - To use your own icons for items on your dashboard, see Icons --> Local Icons
  • web-icons/ - Override Dashy logo
  • widget-resources/ - Fonts, icons and assets for custom widgets

4. Configure#

Now that you've got Dashy running, you are going to want to set it up with your own content. Config is written in YAML Format, and saved in /user-data/conf.yml. -The format on the config file is pretty straight forward. There are three root attributes:

  • pageInfo - Dashboard meta data, like title, description, nav bar links and footer text
  • appConfig - Dashboard settings, like themes, authentication, language and customization
  • sections - An array of sections, each including an array of items

You can view a full list of all available config options in the Configuring Docs.

pageInfo:  title: Home Labsections: # An array of sections- name: Example Section  icon: far fa-rocket  items:  - title: GitHub    description: Dashy source code and docs    icon: fab fa-github    url: https://github.com/Lissy93/dashy  - title: Issues    description: View open issues, or raise a new one    icon: fas fa-bug    url: https://github.com/Lissy93/dashy/issues- name: 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/

Notes:

  • You can use a Docker volume to pass a config file from your host system to the container
    • E.g. -v ./host-system/my-local-conf.yml:/app/user-data/conf.yml
  • It's also possible to edit your config directly through the UI, and changes will be saved in this file
  • Check your config against Dashy's schema, with docker exec -it [container-id] yarn validate-config
  • You might find it helpful to look at some examples, a collection of which can be found here
  • After editing your config, the app will rebuild in the background, which may take a minute

4. Further Customisation#

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 Management Docs.

You might also want to check out the docs for specific features you'd like to use:

  • Authentication - Setting up authentication to protect your dashboard
  • Alternate Views - Using the startpage and workspace view
  • Backup & Restore - Guide to Dashy's cloud sync feature
  • Icons - Outline of all available icon types for sections and items
  • Localisation - How to change language, or add your own
  • Status Indicators - Using Dashy to monitor uptime and status of your apps
  • Search & Shortcuts - Using instant filter, web search and custom hotkeys
  • Theming - Complete guide to applying, writing and modifying themes and styles

5. Final Note#

If you need any help or support in getting Dashy running, head over to the Discussions page. If you think you've found a bug, please do raise it so it can be fixed. For contact options, see the Support Page.

If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the Contributing Page. Huge thanks to everyone who has already helped out!

Enjoy your dashboard :)


Alternative Deployment Method 1 - From Source#

You can also easily run the app on your system without Docker. For this Git, Node.js, and Yarn are required.

git clone https://github.com/Lissy93/dashy.git && cd dashyyarn # Install dependenciesyarn build # Build the appyarn start # Start the app

Then edit ./user-data/conf.yml and rebuild the app with yarn build


Alternative Deployment Method 2 - Netlify#

Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many other cloud providers). All you need it a GitHub account.

  1. Fork Dashy's repository on GitHub
  2. Log in to Netlify with GitHub
  3. Click "New site from Git" and select your forked repo, then click Deploy!
  4. You can then edit the config in ./user-data/conf.yml in your repo, and Netlify will rebuild the app

Alternative Deployment Method 3 - Cloud Services#

Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:

- +The format on the config file is pretty straight forward. There are three root attributes:

  • pageInfo - Dashboard meta data, like title, description, nav bar links and footer text
  • appConfig - Dashboard settings, like themes, authentication, language and customization
  • sections - An array of sections, each including an array of items
  • pages - Have multiples pages in your dashboard

You can view a full list of all available config options in the Configuring Docs.

pageInfo:  title: Home Labsections: # An array of sections- name: Example Section  icon: far fa-rocket  items:  - title: GitHub    description: Dashy source code and docs    icon: fab fa-github    url: https://github.com/Lissy93/dashy  - title: Issues    description: View open issues, or raise a new one    icon: fas fa-bug    url: https://github.com/Lissy93/dashy/issues- name: 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/

Notes:

  • You can use a Docker volume to pass a config file from your host system to the container
    • E.g. -v ./host-system/my-local-conf.yml:/app/user-data/conf.yml
  • It's also possible to edit your config directly through the UI, and changes will be saved in this file
  • Check your config against Dashy's schema, with docker exec -it [container-id] yarn validate-config
  • You might find it helpful to look at some examples, a collection of which can be found here
  • It's also possible to load a remote config, e.g. from a GitHub Gist

5. Further Customisation#

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 Management Docs.

You might also want to check out the docs for specific features you'd like to use:

  • Authentication - Setting up authentication to protect your dashboard
  • Alternate Views - Using the startpage and workspace view
  • Backup & Restore - Guide to Dashy's cloud sync feature
  • Icons - Outline of all available icon types for sections and items
  • Localisation - How to change language, or add your own
  • Status Indicators - Using Dashy to monitor uptime and status of your apps
  • Search & Shortcuts - Using instant filter, web search and custom hotkeys
  • Theming - Complete guide to applying, writing and modifying themes and styles

6. Final Note#

If you need any help or support in getting Dashy running, head over to the Discussions page. If you think you've found a bug, please do raise it so it can be fixed. For contact options, see the Support Page.

If you're enjoying Dashy, and have a few minutes to spare, please do take a moment to look at the Contributing Page. Huge thanks to everyone who has already helped out!

Enjoy your dashboard :)


Alternative Deployment Method 1 - From Source#

You can also easily run the app on your system without Docker. For this Git, Node.js, and Yarn are required.

git clone https://github.com/Lissy93/dashy.git && cd dashyyarn # Install dependenciesyarn build # Build the appyarn start # Start the app

Then edit ./user-data/conf.yml


Alternative Deployment Method 2 - Netlify#

Don't have a server? No problem! You can run Dashy for free on Netlify (as well as many other cloud providers). All you need it a GitHub account.

  1. Fork Dashy's repository on GitHub
  2. Log in to Netlify with GitHub
  3. Click "New site from Git" and select your forked repo, then click Deploy!
  4. You can then edit the config in ./user-data/conf.yml in your repo, and Netlify will rebuild the app

Alternative Deployment Method 3 - Cloud Services#

Dashy supports 1-Click deployments on several popular cloud platforms. To spin up a new instance, just click a link below:

+ \ No newline at end of file diff --git a/docs/readme/index.html b/docs/readme/index.html index 1773116c..14837b06 100644 --- a/docs/readme/index.html +++ b/docs/readme/index.html @@ -6,13 +6,13 @@ ![Dashy Docs](https://i.ibb.co/4mdNf7M/heading-docs.png) | Dashy - +

Dashy Docs

Running Dashy#

  • Quick Start - TLDR guide on getting Dashy up and running
  • Deployment - Full guide on deploying Dashy either locally or online
  • Configuring - Complete list of all available options in the config file
  • App Management - Managing your app, updating, security, web server configuration, etc
  • Troubleshooting - Common errors and problems, and how to fix them

Development and Contributing#

  • Developing - Running Dashy development server locally, and general workflow
  • Development Guides - Common development tasks, to help new contributors
  • Contributing - How you can help keep Dashy alive
  • Showcase - See how others are using Dashy, and share your dashboard
  • Credits - List of people and projects that have made Dashy possible
  • Release Workflow - Info about releases, CI and automated tasks

Feature Docs#

  • Authentication - Guide to setting up authentication to protect your dashboard
  • Alternate Views - Outline of available pages / views and item opening methods
  • Backup & Restore - Guide to backing up config with Dashy's cloud sync feature
  • Icons - Outline of all available icon types for sections and items, with examples
  • Language Switching - Details on how to switch language, or add a new locale
  • Pages and Sections - Multi-page support, sections, items and sub-items
  • Status Indicators - Using Dashy to monitor uptime and status of your apps
  • Searching & Shortcuts - Searching, launching methods + keyboard shortcuts
  • Theming - Complete guide to applying, writing and modifying themes + styles
  • Widgets - List of all dynamic content widgets, with usage guides and examples

Misc#

- + \ No newline at end of file diff --git a/docs/release-workflow/index.html b/docs/release-workflow/index.html index 0b83b514..878440f8 100644 --- a/docs/release-workflow/index.html +++ b/docs/release-workflow/index.html @@ -6,14 +6,14 @@ Releases and Workflows | Dashy - +

Releases and Workflows

Release Schedule#

We're using Semantic Versioning, to indicate major, minor and patch versions. You can find the current version number in the readme, and check your apps version under the config menu. The version number is pulled from the package.json file.

Typically there is a new major release every 2 weeks, usually on Sunday, and you can view these under the Releases Page. Each new version will also have a corresponding tag on GitHub, and each major release will also result in the creation of a new tag on DockerHub, so that you can fix your container to a certain version.

For a full breakdown of each change, you can view the Changelog. Each new feature or significant change needs to be submitted through a pull request, which makes it easy to review and track these changes, and roll back if needed.


Deployment Process#

All changes and new features are submitted as pull requests, which can then be tested, reviewed and (hopefully) merged into the master branch. Every time there is a change in the major version number, a new release is published. This usually happens every 2 weeks, on a Sunday.

When a PR is opened:

  • The feature branch is built, and deployed as a Netlify instance. This can be accessed at: https://deploy-preview-[pr-number]--dashy-dev.netlify.app, and this URL as well as a link to the build logs are added as a comment on the PR by Netlify bot
  • Depending on what files were modified, the bot may also add a comment to remind the author of useful info
  • A series of checks will run on the new code, using GH Actions, and prevent merging if they fail. This includes: linting, testing, code quality and complexity checking, security scanning and a spell check
  • If a new dependency was added, liss-bot will comment with a summary of those changes, as well as the cost of the module, version, and any security concerns. If the bundle size has increased, this will also be added as a comment

After the PR is merged:

  • The app is build, and deployed to: https://dev.dashy.to
  • A new tag in GitHub is created, using the apps version number (from the package.json)
  • The Docker container is built, and published under the :latest tag on DockerHub and GHCR

When a new major version is released:

  • A new GitHub release is created and published, under new versions tag, with info from the changelog
  • The container is built and published under a new tag will be created on DockerHub, called :release-[version]
  • An announcement is opened in GitHub discussions, outlining the main changes, where users can comment and ask questions

Netlify Status


Git Strategy#

Git Flow#

Like most Git repos, we are following the Github Flow standard.

  1. Create a branch (or fork if you don'd have write access)
  2. Code some awesome stuff, then add and commit your changes
  3. Create a Pull Request, complete the checklist and ensure the build succeeds
  4. Follow up with any reviews on your code
  5. Merge 🎉

Git Branch Naming#

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

Commit Emojis#

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

PR Guidelines#

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!

Automated Workflows#

Dashy makes heavy use of GitHub Actions to fully automate the checking, testing, building, deploying of the project, as well as administration tasks like management of issues, tags, releases and documentation. The following section outlines each workflow, along with a link the the action file, current status and short description. A lot of these automations were made possible using community actions contributed to GH marketplace by some amazing people.

Code Processing#

ActionDescription
Code Linter
code-linter.yml
After a pull request is created, all new code changes will be linted, and the CI will fail with a helpful message if the code has any formatting inconsistencies
Code Spell Check
code-spell-check.yml
After a PR submitted, all auto-fixable spelling errors will be detected, then Liss-Bot will create a separate PR to propose the fixes
Dependency Update Summary
dependency-updates-summary.yml
After a PR is submitted, if any of the dependencies are modified, then Liss-Bot will add a comment, explaining which packages have been added, removed, updated or downgraded, as well as other helpful info
Get Size
get-size.yml
Adds comment to PR if the size of the built + bundled application has changed compared to the previous version
Security Scan
security-scanning.yml
Uses Snyk to scan the code and dependencies after a PR. Will add a comment and cause the build to fail if a new vulnerability or potential issue is present

Releases#

ActionDescription
Create Tag
auto-tag-pr.yml
Whenever the version indicated in package.json is updates, a new GitHub tag will be created for that point in time
Build App
build-app.yml
After changes are merged into the master branch, the app will be build, with output pushed to the dev-demo branch
Cache Artifacts
cache-artifacts.yml
After build, returned files will be cached for future actions for that commit
Docker Publish
docker-publish.yml
After PR is merged, the multi-architecture Docker container will be built, and then published to GHCR

Issue Management#

ActionDescription
Close Incomplete Issues
close-incomplete-issues.yml
Issues which do not match any of the issue templates will be closed, and a comment posted explaining why
Close Stale Issues
close-stale-issues.yml
Issues which have not been updated for 6 weeks will have a comment posted to them. If the author does not reply within the next week, then the issue will be marked as stale and closed. The original author may still reopen the issue at any time
Close Potential Spam Issues
issue-spam-control.yml
Auto-closes issues, and adds a comment if it was submitted by a user who hasn't yet interacted with the repo, is new to GitHub and has not starred the repository. The comment will advise them to check their issue is complete, and then allow them to reopen it
Issue Translator
issue-translator.yml
Auto-translates any comments and issues that were written in any language other than English, and posts the translation as a comment below
Label Sponsors
label-sponsors.yml
Adds a special label to any issues or pull requests raised by users who are sponsoring the project via GitHub, so that they can get priority support
LGTM Comment
lgtm-comment.yml
When a PR review contains the words LGTM (looks good to me), the Liss-Bot will reply with a random celebratory or thumbs up GIF, just as a bit of fun
Mind your Language
mind-your-language.yml
Replies to any comment (on issue or PR) that contains profanities, offensive or inappropriate language with a polite note reminding the user of the code of conduct
Release Notifier
release-commenter.yml
Once a release has been published which fixes an issue, a comment will be added to the relevant issues informing the user who raised it that it was fixed in the current release
Update Issue after Merge
update-issue-after-pr.yml
After a PR which fixes an issue is merged, Liss-Bot will add a comment to said issue based on the git commit message
Auto Add Comment Based on Tag
add-comment-from-tag.yml
Will add comment with useful info to certain issues, based on the tag applied

PR Management#

ActionDescription
PR Commenter
pr-commenter.yml
Adds comment with helpful info to pull requests, based on which files have been changes
Issue from Todo Code
raise-issue-from-todo.yml
When a todo note is found in the code after a PR, then Liss-Bot will automatically raise an issue, so that the todo can be addressed/ implemented. The issue will be closed once the todo has been implemented or removed

Documentation & Reports#

ActionDescription
Generate Credits
generate-credits.yml
Generates a report, including contributors, collaborators, sponsors, bots and helpful users. Will then insert a markdown table with thanks to these GitHub users and links to their profiles into the Credits page, as well as a summary of sponsors and top contributors into the main readme
Wiki Sync
wiki-sync.yml
Generates and publishes the repositories wiki page using the markdown files within the docs directory

- + \ No newline at end of file diff --git a/docs/searching/index.html b/docs/searching/index.html index 1d628e2d..ae504f50 100644 --- a/docs/searching/index.html +++ b/docs/searching/index.html @@ -6,7 +6,7 @@ Keyboard Shortcuts | Dashy - + @@ -14,7 +14,7 @@

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.

Navigating#

You can navigate through your items or search results using the keyboard. You can use Tab to cycle through results, and Shift + Tab to go backwards. Or use the arrow keys, , , and .

Launching Apps#

You can launch a elected app by hitting Enter. This will open the app using your default opening method, specified in target (either newtab, sametab, modal, top or workspace). You can also use Alt + Enter to open the app in a pop-up modal, or Ctrl + 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.

Tags#

By default, items are filtered by the title attribute, as well as the hostname (extracted from url), the provider and description. If you need to find results based on text which isn't included in these attributes, then you can add tags to a given item.

  items:  - title: Plex    description: Media library    icon: favicon    url: https://plex.lab.local    tags: [ movies, videos, music ]  - title: FreshRSS    description: RSS Reader    icon: favicon    url: https://freshrss.lab.local    tags: [ news, updates, blogs ]
 

In the above example, Plex will be visible when searching for 'movies', and FreshRSS with 'news'

Custom Hotkeys#

For apps that you use regularly, you can set a custom keybinding. Use the hotkey parameter on a certain item to specify a numeric key, between 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.

- title: Bookstack  icon: far fa-books  url: https://bookstack.lab.local/  hotkey: 2- title: Git Tea  icon: fab fa-git  url: https://git.lab.local/  target: workspace  hotkey: 3

In the above example, pressing 2 will launch Bookstack. Or hitting 3 will open Git in the workspace view.

Web Search#

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 . Web search options are configured under appConfig.webSearch.

Setting Search Engine#

Set your default search engine using the webSearch.searchEngine property. This defaults to DuckDuckGo. Search engine must be referenced by their key, the following providers are supported:

Using Custom Search Engine#

You can also use a custom search engine, that isn't included in the above list (like a self-hosted instance of Whoogle or Searx). Set searchEngine: custom, and then specify the URL (plus query params) to you're search engine under customSearchEngine.

For example:

appConfig:  webSearch:    searchEngine: custom    customSearchEngine: 'https://searx.local/search?q='

Setting Opening Method#

In a similar way to opening apps, you can specify where you would like search results to be opened. This is done under the openingMethod attribute, and can be set to either newtab, sametab or workspace. By default results are opened in a new tab.

Using Bangs#

An insanely useful feature of DDG is 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.

This is done under the 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.

For example:

appConfig:  webSearch:    searchEngine: 'duckduckgo'    openingMethod: 'newtab'    searchBangs:      /r: reddit      /w: wikipedia      /s: https://whoogle.local/search?q=      /a: https://www.amazon.co.uk/s?k=      ':wolf': wolframalpha      ':so': stackoverflow      ':git': github

Note that bangs begging with ! or : must be surrounded them in quotes

Disabling Web Search#

Web search can be disabled, by setting disableWebSearch, for example:

appConfig:  webSearch: { disableWebSearch: true }

Clearing Search#

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.

- + \ No newline at end of file diff --git a/docs/showcase/index.html b/docs/showcase/index.html index 8fa3a623..5cfb84df 100644 --- a/docs/showcase/index.html +++ b/docs/showcase/index.html @@ -6,7 +6,7 @@ *Dashy Showcase* 🌟 | Dashy - + @@ -18,7 +18,7 @@ Dashy is awesome!

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

- + \ No newline at end of file diff --git a/docs/showcase/readme/index.html b/docs/showcase/readme/index.html index b2aeb4b5..763adf49 100644 --- a/docs/showcase/readme/index.html +++ b/docs/showcase/readme/index.html @@ -6,13 +6,13 @@ readme | Dashy - + - + \ No newline at end of file diff --git a/docs/status-indicators/index.html b/docs/status-indicators/index.html index 4c9c0226..0584308d 100644 --- a/docs/status-indicators/index.html +++ b/docs/status-indicators/index.html @@ -6,14 +6,14 @@ Status Indicators | Dashy - +

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#

By default, this feature is off. If you do not want this feature, just don't add the statusCheck to your conf.yml file, then no requests will be made.

To enable status checks, you can either turn it on for all items, by setting appConfig.statusCheck: true, like:

appConfig:  statusCheck: true

Or you can enable/ disable it on a per-item basis, with the item[n].statusCheck attribute

sections:- name: Firewall  items:  - title: OPNsense    description: Firewall Central Management    icon: networking/opnsense.png    url: https://192.168.1.1    statusCheck: false  - title: MalTrail    description: Malicious traffic detection system    icon: networking/maltrail.png    url: http://192.168.1.1:8338    statusCheck: true  - title: Ntopng    description: Network traffic probe and network use monitor    icon: networking/ntop.png    url: http://192.168.1.1:3001    statusCheck: true

Continuous Checking#

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 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.

The following example, will instruct Dashy to continuously check the status of your services every 20 seconds.

appConfig:  statusCheck: true  statusCheckInterval: 20

Using a Different Endpoint#

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.

You can set the statusCheckUrl property on any given item in order to do this. The status checker will then ping that endpoint, instead of the apps main url property.

Setting Custom Headers#

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 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. For example, statusCheckHeaders: { 'X-Custom-Header': 'foobar' }

Disabling Security#

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 statusCheckAllowInsecure: true

Allowing Alternative Status Codes#

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 statusCheckAcceptCodes for a given item. For example, statusCheckAcceptCodes: '403,418'

Troubleshooting Failing Status Checks#

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:

If your service requires requests to include any authorization in the headers, then use the statusCheckHeaders property, as described above.

If you are still having issues, it may be because your target application is blocking requests from Dashy's IP. This is a CORS error, and can be fixed by setting the headers on your target app, to include:

Access-Control-Allow-Origin: https://location-of-dashy/Vary: Origin

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 statusCheckAllowInsecure to true for a given item.

If your service is online, but responds with a status code that is not in the 2xx range, then you can use statusCheckAcceptCodes to set an accepted status code.

If you get an error, like 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 https:// (or whatever protocol) before the URL, and ensure that if needed, you've specified the port.

Running Dashy in HOST network mode, instead of BRIDGE will allow status check access to other services in HOST mode. For more info, see #445.

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.

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 statusCheckUrl to your custom page.

For further troubleshooting, use an application like Postman to diagnose the issue. Set the parameter to GET, and then make a call to: https://[url-of-dashy]/status-check/?&url=[service-url]. Where the service URL must have first been encoded (e.g. with encodeURIComponent() or urlencoder.io)

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.

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.

- + \ No newline at end of file diff --git a/docs/theming/index.html b/docs/theming/index.html index 5c1c20e8..1b26d9fd 100644 --- a/docs/theming/index.html +++ b/docs/theming/index.html @@ -6,14 +6,14 @@ Theming | Dashy - +

Theming

By default Dashy comes with 25+ built-in themes, which can be applied from the dropdown menu in the UI.

Built-in Themes

You can also add your own themes, apply custom styles, and modify colors.

You can customize Dashy by writing your own CSS, which can be loaded either as an external stylesheet, set directly through the UI, or specified in the config file. Most styling options can be set through CSS variables, which are outlined below.

The following content requires that you have a basic understanding of CSS. If you're just beginning, you may find this article helpful.

How Theme-Switching Works#

The theme switching is done by simply changing the data-theme attribute on the root DOM element, which can then be targeted by CSS. First off, in order for the theme to show up in the theme switcher, it needs to be added to the config file, under appConfig.cssThemes, either as a string, or an array of strings for multiple themes. For example:

appConfig:  cssThemes: ['tiger', 'another-theme']

You can now create a block to target you're theme with html[data-theme='my-theme']{} and set some styles. The easiest method is by setting CSS variables, but you can also directly override elements by their selector. As an example, see the built-in CSS themes.

html[data-theme='tiger'] {  --primary: #f58233;  --background: #0b1021;}

Finally, from the UI use the theme dropdown menu to select your new theme, and your styles will be applied.

You can also set appConfig.theme to pre-select a default theme, which will be applied immediately after deployment.

Modifying Theme Colors#

Themes can be modified either through the UI, using the color picker menu (to the right of the theme dropdown), or directly in the config file, under appConfig.customColors. Here you can specify the value for any of the available CSS variables.

Example Themes

By default, any color modifications made to the current theme through the UI will only be applied locally. If you need these settings to be set globally, then click the 'Export' button, to get the color codes and variable names, which can then be backed up, or saved in your config file.

Custom colors are saved relative the the base theme selected. So if you switch themes after setting custom colors, then you're settings will no longer be applied. You're changes are not lost though, and switching back to the original theme will see your styles reapplied.

If these values are specified in your conf.yml file, then it will look something like the below example. Note that in YAML, values or keys which contain special characters, must be wrapped in quotes.

appConfig:  customColors:    oblivion:      primary: '#75efff'      background: '#2a3647'    dracula:      primary: '#8be9fd'

Adding your own Theme#

User-defined styles and custom themes should be defined in ./src/styles/user-defined-themes.scss. If you're using Docker, you can pass your own stylesheet in using the --volume flag. E.g. v ./my-themes.scss:/app/src/styles/user-defined-themes.scss. Don't forget to pass your theme name into appConfig.cssThemes so that it shows up on the theme-switcher dropdown.

Setting Custom CSS in the UI#

Custom CSS can be developed, tested and applied directly through the UI. Although you will need to make note of your changes to apply them across instances.

This can be done from the Config menu (spanner icon in the top-right), under the Custom Styles tab. This is then associated with appConfig.customCss in local storage. Styles can also be directly applied to this attribute in the config file, but this may get messy very quickly if you have a lot of CSS.

Page-Specific Styles#

If you've got multiple pages within your dashboard, you can choose to target certain styles to specific pages. The top-most element within <body> will have a class name specific to the current sub-page. This is usually the page's name, all lowercase, with dashes instead of spaces, but you can easily check this yourself within the dev tools.

For example, if the pages name was "CFT Toolbox", and you wanted to target .items, you would do:

.cft-toolbox .item { border: 4px solid yellow; }

Loading External Stylesheets#

The URI of a stylesheet, either local or hosted on a remote CDN can be passed into the config file. The attribute appConfig.externalStyleSheet accepts either a string, or an array of strings. You can also pass custom font stylesheets here, they must be in a CSS format (for example, https://fonts.googleapis.com/css2?family=Cutive+Mono). This is handled in ThemeHelper.js.

For example:

appConfig:  externalStyleSheet: 'https://example.com/my-stylesheet.css'
appConfig:  externalStyleSheet: ['/themes/my-theme-1.css', '/themes/my-theme-2.css']

Hard-Coding Section or Item Colors#

Some UI components have a color option, that can be set in the config file, to force the color of a given item or section no matter what theme is selected. These colors should be expressed as hex codes (e.g. #fff) or HTML colors (e.g. red). The following attributes are supported:

  • section.color - Custom color for a given section
  • item.color - Font and icon color for a given item
  • item.backgroundColor - Background color for a given icon

Typography#

Essential fonts bundled within the app are located within ./src/assets/fonts/. All optional fonts that are used by themes are stored in ./public/fonts/, if you want to add your own font, this is where you should put it. As with assets, if you're using Docker then using a volume to link a directory on your host system with this path within the container will make management much easier.

Fonts which are not being used by the current theme are not fetched on page load. They are instead only loaded into the application if and when they are required. So having multiple themes with various typefaces shouldn't have any negative impact on performance.

Full credit to the typographers behind each of the included fonts. Specifically: Matt McInerney, Christian Robertson, Haley Fiege, Peter Hull, Cyreal and the legendary Vernon Adams

CSS Variables#

All colors as well as other variable values (such as borders, border-radius, shadows) are specified as CSS variables. This makes theming the application easy, as you only need to change a given color or value in one place. You can find all variables in color-palette.scss and the themes which make use of these color variables are specified in color-themes.scss

CSS variables are simple to use. You define them like: --background: #fff; and use them like: body { background-color: var(--background); }. For more information, see this guide on using CSS Variables.

You can determine the variable used by any given element, and visualize changes using the browser developer tools (Usually opened with F12, or Options --> More --> Developer Tools). Under the elements tab, click the Element Selector icon (usually top-left corner), you will then be able to select any DOM element on the page by hovering and clicking it. In the CSS panel you will see all styles assigned to that given element, including CSS variables. Click a variable to see it's parent value, and for color attributes, click the color square to modify the color. For more information, see this getting started guide, and these articles on selecting elements and inspecting and modifying colors.

Top-Level Variables#

These are all that are required to create a theme. All other variables inherit their values from these variables, and can optionally be overridden.

  • --primary - Application primary color. Used for title, text, accents, and other features
  • --background - Application background color
  • --background-darker - Secondary background color (usually darker), used for navigation bar, section fill, footer etc
  • --curve-factor - The border radius used globally throughout the application. Specified in px, defaults to 5px
  • --dimming-factor - Inactive elements have slight transparency. This can be between 0 (invisible) and 1 (normal), defaults to 0.7

Targeted Color Variables#

You can target specific elements on the UI with these variables. All are optional, since by default, they inherit their values from above

  • --heading-text-color - Text color for web page heading and sub-heading. Defaults to --primary
  • --nav-link-text-color - The text color for links displayed in the navigation bar. Defaults to --primary
  • --nav-link-background-color - The background color for links displayed in the navigation bar
  • --nav-link-text-color-hover - The text color when a navigation bar link is hovered over. Defaults to --primary
  • --nav-link-background-color-hover - The background color for nav bar links when hovered over
  • --nav-link-border-color - The border color for nav bar links. Defaults to transparent
  • --nav-link-border-color-hover - The border color for nav bar links when hovered over. Defaults to --primary
  • --search-container-background - Background for the container containing the search bar. Defaults to --background-darker
  • --search-field-background - Fill color for the search bar. Defaults to --background
  • --settings-background - The background for the quick settings. Defaults to --background
  • --settings-text-color - The text and icon color for quick settings. Defaults to --primary
  • --footer-text-color - Color for text within the footer. Defaults to --medium-grey
  • --footer-text-color-link - Color for any hyperlinks within the footer. Defaults to --primary
  • --item-text-color - The text and icon color for items. Defaults to --primary
  • --item-group-outer-background - The background color for the outer part of a section (including section head). Defaults to --primary
  • --item-group-background - The background color for the inner part of item groups. Defaults to #0b1021cc (semi-transparent black)
  • --item-group-heading-text-color - The text color for section headings. Defaults to --item-group-background;
  • --item-group-heading-text-color-hover - The text color for section headings, when hovered. Defaults to --background
  • --config-code-background - Background color for the JSON editor in the config menu. Defaults to #fff (white)
  • --config-code-color - Text color for the non-highlighted code within the JSON editor. Defaults to --background
  • --config-settings-color - The background for the config/ settings pop-up modal. Defaults to --primary
  • --config-settings-background - The text color for text within the settings container. Defaults to --background-darker
  • --scroll-bar-color - Color of the scroll bar thumb. Defaults to --primary
  • --scroll-bar-background Color of the scroll bar blank space. Defaults to --background-darker
  • --highlight-background Fill color for text highlighting. Defaults to --primary
  • --highlight-color Text color for selected/ highlighted text. Defaults to --background
  • --toast-background - Background color for the toast info popup. Defaults to --primary
  • --toast-color - Text, icon and border color in the toast info popup. Defaults to --background
  • --welcome-popup-background - Background for the info pop-up shown on first load. Defaults to --background-darker
  • --welcome-popup-text-color - Text color for the welcome pop-up. Defaults to --primary
  • --side-bar-background - Background color of the sidebar used in the workspace view. Defaults to --background-darker
  • --side-bar-color - Color of icons and text within the sidebar. Defaults to --primary
  • --status-check-tooltip-background - Background color for status check tooltips. Defaults to --background-darker
  • --status-check-tooltip-color - Text color for the status check tooltips. Defaults to --primary
  • --code-editor-color - Text color used within raw code editors. Defaults to --black
  • --code-editor-background - Background color for raw code editors. Defaults to --white
  • --context-menu-color - Text color for right-click context menu over items. Defaults to --primary
  • --context-menu-background - Background color of right-click context menu. Defaults to --background
  • --context-menu-secondary-color - Border and outline color for context menu. Defaults to --background-darker

Non-Color Variables#

  • --outline-color - Used to outline focused or selected elements
  • --curve-factor-navbar - The border radius of the navbar. Usually this is greater than --curve-factor
  • --scroll-bar-width - Width of horizontal and vertical scroll bars. E.g. 8px
  • --item-group-padding - Inner padding of sections, determines the width of outline. E.g. 5px
  • --item-shadow - Shadow for items. E.g. 1px 1px 2px #130f23
  • --item-hover-shadow - Shadow for items when hovered over. E.g. 1px 2px 4px #373737
  • --item-icon-transform - A transform property, to modify item icons. E.g. drop-shadow(2px 4px 6px var(--transparent-50)) saturate(0.65)
  • --item-icon-transform-hover - Same as above, but applied when an item is hovered over. E.g. drop-shadow(4px 8px 3px var(--transparent-50)) saturate(2)
  • --item-group-shadow - The shadow for an item group/ section. Defaults to --item-shadow
  • --settings-container-shadow - A shadow property for the settings container. E.g. none

Action Colors#

These colors represent intent, and so are not often changed, but you can do so if you wish

  • --info - Information color, usually blue / #04e4f4
  • --success - Success color, usually green / #20e253
  • --warning - Warning color, usually yellow / #f6f000
  • --danger - Error/ danger color, usually red / #f80363
  • --neutral - Neutral color, usually grey / #272f4d
  • --white - Just white / #fff
  • --black - Just black / #000
- + \ No newline at end of file diff --git a/docs/troubleshooting/index.html b/docs/troubleshooting/index.html index 30140a18..d9781c14 100644 --- a/docs/troubleshooting/index.html +++ b/docs/troubleshooting/index.html @@ -6,7 +6,7 @@ Troubleshooting | Dashy - + @@ -34,7 +34,7 @@ This will not affect your config file. But be sure that you keep a backup of you Then describe the issue, briefly explaining the steps to reproduce, expected outcome and actual outcome.

Step 4 - Provide Supporting Info#

Where relevant please also include:

  • A screenshot of the issue
  • The relevant parts of your config file
  • Logs
    • If client-side issue, then include the browser logs (see how)
    • If server-side / during deployment, include the terminal output

Take care to redact any personal info, (like IP addresses, auth hashes or API keys).

Step 5 - Fix Released#

A maintainer will aim to respond within 48 hours. The timeframe for resolving your issue, will vary depending on severity of the bug and the complexity of the fix. You will be notified on your ticket, when a fix has been released.

Finally, be sure to remain respectful to other users and project maintainers, in line with the Contributor Covenant Code of Conduct.


How-To Open Browser Console#

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.

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 F12. You can right-click the console, and select Save As to download the log.

To open dev tools, and jump straight to the console:

  • Win / Linux: Ctrl + Shift + J
  • MacOS: Cmd + Option + J

For more detailed walk through, see this article.


Git Contributions not Displaying#

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 Credits Page, then this is likely a git config issue.

These statistics are generated using the username / email associated with commits. This info needs to be setup on your local machine using git config.

Run the following commands (replacing name + email with your info):

  • git config --global user.name "John Doe"
  • 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

- + \ No newline at end of file diff --git a/docs/widgets/index.html b/docs/widgets/index.html index 61c354aa..b7c00194 100644 --- a/docs/widgets/index.html +++ b/docs/widgets/index.html @@ -6,7 +6,7 @@ Widgets | Dashy - + @@ -25,7 +25,7 @@ Instead, for secrets you should use environmental vairables.

You can do th To specify the chart height, set chartHeight to an integer (in px), defaults to 300. For example:

- type: gl-load-history  options:    hostname: http://192.168.130.2:61208    chartColors: ['#9b5de5', '#f15bb5', '#00bbf9', '#00f5d4']    chartHeight: 450

Language Translations#

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.

However, any hard-coded content is translatable, and all dates and times will display in your local format.

For more info about multi-language support, see the Internationalization Docs.


Widget UI Options#

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.

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.

All 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 displayData.cols: 2 within the parent section. You can collapse a widget (by clicking the section title), and collapse state will be saved locally.

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 VS Code Server, or just SSH into your box and edit the conf.yml file directly.


Build your own Widget#

Widgets are built in a modular fashion, making it easy for anyone to create their own custom components.

For a full tutorial on creating your own widget, you can follow this guide, or take a look at here for a code example.

Alternatively, for displaying simple data, you could also just use the either the iframe, embed, data feed or API response widgets.


Requesting a Widget#

Suggestions for widget ideas are welcome. But there is no guarantee that I will build your widget idea.

Please only request widgets for services that:

  • Have a publicly accessible API
  • Are CORS and HTTPS enabled
  • Are free to use, or have a free plan
  • Allow for use in their Terms of Service
  • Would be useful for other users

You can suggest a widget 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.

For services that are not officially supported, it is likely still possible to display data using either the iframe, embed or API response widgets. For more advanced features, like charts and action buttons, you could also build your own widget, using this tutorial, it's fairly straight forward, and you can use an existing widget (or this example) as a template.


Troubleshooting Widget Errors#

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 open the browser console to see more details.

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.

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.

CORS Errors#

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: Cross-Origin Resource Sharing.

There are several ways to fix a CORS error:

Option 1 - Ensure Correct Protocol#

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.

Option 2 - Set Headers#

If you have control over the destination (e.g. for a self-hosted service), then you can simply apply the correct headers. Add the Access-Control-Allow-Origin header, with the value of either * to allow requests from anywhere, or more securely, the host of where Dashy is served from. For example:

Access-Control-Allow-Origin: https://url-of-dashy.local

or

Access-Control-Allow-Origin: *

Option 3 - Proxying Request#

You can route requests through Dashy's built-in CORS proxy. Instructions and more details can be found 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.

Just add the useProxy: true option to the failing widget.

Option 4 - Use a plugin#

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.

- + \ No newline at end of file diff --git a/index.html b/index.html index e1464584..aae8485d 100644 --- a/index.html +++ b/index.html @@ -6,13 +6,13 @@ Dashy | Dashy - +

Dashy

The Ultimate Homepage for your Homelab

Dashy is an open source, highly customizable, easy to use, privacy-respecting dashboard app.Keep Reading...
Feature List

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

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
Screenshot Coming Soon

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

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

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

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
Screenshot Coming Soon

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

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

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
- + \ No newline at end of file diff --git a/markdown-page/index.html b/markdown-page/index.html index 293b6175..2100939a 100644 --- a/markdown-page/index.html +++ b/markdown-page/index.html @@ -6,13 +6,13 @@ Markdown page example | Dashy - + - + \ No newline at end of file diff --git a/search/index.html b/search/index.html index 349c3f85..faef0c75 100644 --- a/search/index.html +++ b/search/index.html @@ -6,13 +6,13 @@ Search the documentation | Dashy - + - + \ No newline at end of file diff --git a/survey/index.html b/survey/index.html index 5794ebfa..406ac67f 100644 --- a/survey/index.html +++ b/survey/index.html @@ -6,13 +6,13 @@ Dashy Survey | Dashy - + - + \ No newline at end of file