Compare commits

..

34 Commits

Author SHA1 Message Date
github-actions[bot]
af1695687a New Crowdin translations by GitHub Action (#3589)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-06-05 19:38:39 -07:00
shamoon
f0d7cf3ce6 Security: Sanitize widget api keys from response
Closes https://github.com/gethomepage/homepage/security/advisories/GHSA-cjgf-vhj6-8cx4
2024-06-05 19:38:26 -07:00
Heng-Yi Wu
36e77e1fe3 Fix: coinmarketcap widget dropdown not clickable (#3591) 2024-06-05 00:56:28 -07:00
Aldy J
35dd7ec230 Enhancement: support multiple instance annotation for Kubernetes Automatic Service Discovery (#3590) 2024-06-04 21:29:13 -07:00
shamoon
15cc1d98c5 Fix: omada widget broken in v0.9.1 (#3587) 2024-06-04 07:28:45 -07:00
Liran Vaknin
9803ef70c6 Fix: openwrt widget handle null id in json rpc responses and cpu load representation (#3576) 2024-06-04 00:11:11 -07:00
shamoon
ccc27142ef Fix: caddy widget broken in 0.9.1 (#3579) 2024-06-04 00:05:06 -07:00
shamoon
9a20982406 Lint wg-easy 2024-06-04 00:03:48 -07:00
Conner Hnatiuk
67a69a5878 Fix: wg-easy threshold not properly computed (#3574) 2024-06-03 14:52:58 -07:00
shamoon
19c3ac0d7e Enforce method 2024-06-03 13:17:23 -07:00
shamoon
67a9f4983c Fix unifi with no type passing and lint 2024-06-03 11:19:54 -07:00
github-actions[bot]
ec79f3042a New Crowdin translations by GitHub Action (#3567)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-06-03 09:39:38 -07:00
shamoon
f13144ceb8 Remove widget type passing 2024-06-03 09:20:26 -07:00
shamoon
d42271acf6 Fix: calendar integration broken in v0.9.0 (#3566) 2024-06-03 09:19:20 -07:00
shamoon
fb58a66f3b Fix: include cpu in allowed glances endpoints (#3565) 2024-06-03 09:04:21 -07:00
shamoon
c96e6703d3 More path traversal fixes 2024-06-03 09:04:21 -07:00
github-actions[bot]
8a4c808ee2 New Crowdin translations by GitHub Action (#3432)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-06-03 09:04:21 -07:00
shamoon
52cce0ee21 Merge pull request from GHSA-24m5-7vjx-9x37
* Restrict emby endpoints and proxy segments

* Dont allow path traversal in segments

* Restrict qbittorrent proxy endpoints

* Restrict npm proxy endpoints

* Restrict flood proxy endpoints

* Restrict tdarr proxy endpoints

* Restrict xteve proxy endpoints

* Restrict transmission proxy endpoints

* disallow non-mapped endpoints

this change drops all requests that have un-mapped endpoint queries

allowedEndpoints is added as a method to pass proxy requests via a regex on the endpoint

most widgets with custom proxies use either no endpoint, or a static one

Co-Authored-By: Ben Phelps <ben@phelps.io>
2024-06-03 09:04:21 -07:00
Ben Phelps
8823b04291 switch DO logo
darker one looks nicer
2024-06-02 16:45:06 +03:00
dependabot[bot]
bb2da35d09 Chore(deps-dev): Bump eslint-plugin-react-hooks from 4.6.0 to 4.6.2 (#3555)
Bumps [eslint-plugin-react-hooks](https://github.com/facebook/react/tree/HEAD/packages/eslint-plugin-react-hooks) from 4.6.0 to 4.6.2.
- [Release notes](https://github.com/facebook/react/releases)
- [Changelog](https://github.com/facebook/react/blob/main/packages/eslint-plugin-react-hooks/CHANGELOG.md)
- [Commits](https://github.com/facebook/react/commits/HEAD/packages/eslint-plugin-react-hooks)

---
updated-dependencies:
- dependency-name: eslint-plugin-react-hooks
  dependency-type: direct:development
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-06-01 09:58:39 -07:00
dependabot[bot]
fa633a705d Chore(deps-dev): Bump eslint-plugin-react from 7.34.1 to 7.34.2 (#3554)
Bumps [eslint-plugin-react](https://github.com/jsx-eslint/eslint-plugin-react) from 7.34.1 to 7.34.2.
- [Release notes](https://github.com/jsx-eslint/eslint-plugin-react/releases)
- [Changelog](https://github.com/jsx-eslint/eslint-plugin-react/blob/master/CHANGELOG.md)
- [Commits](https://github.com/jsx-eslint/eslint-plugin-react/compare/v7.34.1...v7.34.2)

---
updated-dependencies:
- dependency-name: eslint-plugin-react
  dependency-type: direct:development
  update-type: version-update:semver-patch
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-06-01 09:53:27 -07:00
vycdev
48170fe899 Enhancement: separate Internet search settings for quicklaunch (#3541)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-05-31 21:24:42 -07:00
Simon
a53b042fec Fix: remove comma in Caddy and EVCC widgetS (#3551) 2024-05-31 18:01:27 -07:00
dependabot[bot]
c267eeabf8 Chore(deps): Bump crowdin/github-action from 1 to 2 (#3537)
Bumps [crowdin/github-action](https://github.com/crowdin/github-action) from 1 to 2.
- [Release notes](https://github.com/crowdin/github-action/releases)
- [Commits](https://github.com/crowdin/github-action/compare/v1...v2)

---
updated-dependencies:
- dependency-name: crowdin/github-action
  dependency-type: direct:production
  update-type: version-update:semver-major
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-05-29 10:44:23 -07:00
quietsy
c6770d233c Feature: SWAG dashboard widget (#3523) 2024-05-27 22:00:52 -07:00
Ben Phelps
0a75c831a6 Documentation: DO Credits (#3505)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-05-22 06:19:48 -07:00
Nick Disiere
4d76443846 Fix: correct icon in the longhorn widget (#3509) 2024-05-21 13:06:59 -07:00
shamoon
97d7ae21e4 Fix: handle some status cases with non-existent k8s pods (#3489) 2024-05-17 09:01:43 -07:00
Conner Hnatiuk
6ab6d6fd3a Feature: Wg-Easy Widget (#3476)
---------

Co-authored-by: ConnerWithAnE <46903591+ConnerWithAnE@users.noreply.github.com>
Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-05-16 22:26:12 -07:00
Jesus Lopez
1144f4dfa0 Fix: allow exclamation to open quicklaunch (#3475) 2024-05-16 18:09:50 -07:00
shamoon
a9ad2a2146 Improve k8s not found pod status 2024-05-16 08:28:12 -07:00
shamoon
43ebd6d0c5 Fix: handle ghostfolio v2.79.0 breaking API changes (#3471) 2024-05-16 08:07:33 -07:00
zinsmeik
d20ab844d6 Documentation: correct weatherapi example (#3436) 2024-05-10 06:23:54 -07:00
Noah S. Roberts
857ac1f7dc Documentation: use generic url in channels dvr widget docs (#3434) 2024-05-10 00:56:30 -07:00
106 changed files with 1533 additions and 743 deletions

View File

@@ -19,7 +19,7 @@ jobs:
- name: Checkout - name: Checkout
uses: actions/checkout@v4 uses: actions/checkout@v4
- name: crowdin action - name: crowdin action
uses: crowdin/github-action@v1 uses: crowdin/github-action@v2
with: with:
upload_translations: false upload_translations: false
download_translations: true download_translations: true

View File

@@ -25,6 +25,13 @@
<a href="https://paypal.me/phelpsben" title="Donate"><img alt="GitHub Sponsors" src="https://img.shields.io/github/sponsors/benphelps"></a> <a href="https://paypal.me/phelpsben" title="Donate"><img alt="GitHub Sponsors" src="https://img.shields.io/github/sponsors/benphelps"></a>
</p> </p>
<p align="center">
<a href="https://www.digitalocean.com/?refcode=df14bcb7c016&utm_campaign=Referral_Invite&utm_medium=Referral_Program&utm_source=badge"><img src="https://web-platforms.sfo2.cdn.digitaloceanspaces.com/WWW/Badge%201.svg" alt="DigitalOcean Referral Badge" /></a>
</p>
<p align="center">
<em>Homepage builds are kindly powered by DigitalOcean.</em>
</p>
# Features # Features
With features like quick search, bookmarks, weather support, a wide range of integrations and widgets, an elegant and modern design, and a focus on performance, Homepage is your ideal start to the day and a handy companion throughout it. With features like quick search, bookmarks, weather support, a wide range of integrations and widgets, an elegant and modern design, and a focus on performance, Homepage is your ideal start to the day and a handy companion throughout it.

View File

@@ -98,6 +98,8 @@ When the Kubernetes cluster connection has been properly configured, this servic
If you are using multiple instances of homepage, an `instance` annotation can be specified to limit services to a specific instance. If no instance is provided, the service will be visible on all instances. If you are using multiple instances of homepage, an `instance` annotation can be specified to limit services to a specific instance. If no instance is provided, the service will be visible on all instances.
If you have a single service that needs to be shown on multiple specific instances of homepage (but not on all of them), the service can be annotated by multiple `instance.name` annotations, where `name` can be the names of your specific multiple homepage instances. For example, a service that is annotated with `gethomepage.dev/instance.public: ""` and `gethomepage.dev/instance.internal: ""` will be shown on `public` and `internal` homepage instances.
### Traefik IngressRoute support ### Traefik IngressRoute support
Homepage can also read ingresses defined using the Traefik IngressRoute custom resource definition. Due to the complex nature of Traefik routing rules, it is required for the `gethomepage.dev/href` annotation to be set: Homepage can also read ingresses defined using the Traefik IngressRoute custom resource definition. Due to the complex nature of Traefik routing rules, it is required for the `gethomepage.dev/href` annotation to be set:

View File

@@ -363,7 +363,7 @@ providers:
You can then pass `provider` instead of `apiKey` in your widget configuration. You can then pass `provider` instead of `apiKey` in your widget configuration.
```yaml ```yaml
- weather: - weatherapi:
latitude: 50.449684 latitude: 50.449684
longitude: 30.525026 longitude: 30.525026
provider: weatherapi provider: weatherapi
@@ -377,9 +377,10 @@ You can use the 'Quick Launch' feature to search services, perform a web search
There are a few optional settings for the Quick Launch feature: There are a few optional settings for the Quick Launch feature:
- `searchDescriptions`: which lets you control whether item descriptions are included in searches. This is off by default. When enabled, results that match the item name will be placed above those that only match the description. - `searchDescriptions`: which lets you control whether item descriptions are included in searches. This is false by default. When enabled, results that match the item name will be placed above those that only match the description.
- `hideInternetSearch`: disable automatically including the currently-selected web search (e.g. from the widget) as a Quick Launch option. This is false by default, enabling the feature. - `hideInternetSearch`: disable automatically including the currently-selected web search (e.g. from the widget) as a Quick Launch option. This is false by default, enabling the feature.
- `showSearchSuggestions`: shows search suggestions for the internet search. This value will be inherited from the search widget if it is not specified. If it is not specified there either, it will default to false. - `showSearchSuggestions`: show search suggestions for the internet search. If this is not specified then the setting will be inherited from the search widget. If it is not specified there either, it will default to false. For custom providers the `suggestionUrl` needs to be set in order for this to work.
- `provider`: search engine provider. If none is specified it will try to use the provider set for the Search Widget, if neither are present then internet search will be disabled.
- `hideVisitURL`: disable detecting and offering an option to open URLs. This is false by default, enabling the feature. - `hideVisitURL`: disable detecting and offering an option to open URLs. This is false by default, enabling the feature.
```yaml ```yaml
@@ -388,6 +389,17 @@ quicklaunch:
hideInternetSearch: true hideInternetSearch: true
showSearchSuggestions: true showSearchSuggestions: true
hideVisitURL: true hideVisitURL: true
provider: google # google, duckduckgo, bing, baidu, brave or custom
```
or for a custom search:
```yaml
quicklaunch:
provider: custom
url: https://www.ecosia.org/search?q=
target: _blank
suggestionUrl: https://ac.ecosia.org/autocomplete?type=list&q=
``` ```
## Homepage Version ## Homepage Version

View File

@@ -17,3 +17,10 @@ hide:
A modern, <em>fully static, fast</em>, secure <em>fully proxied</em>, highly customizable application dashboard with integrations for over 100 services and translations into multiple languages. Easily configured via YAML files or through docker label discovery. A modern, <em>fully static, fast</em>, secure <em>fully proxied</em>, highly customizable application dashboard with integrations for over 100 services and translations into multiple languages. Easily configured via YAML files or through docker label discovery.
![Alt text](assets/homepage_demo.png) ![Alt text](assets/homepage_demo.png)
<p align="center">
<a href="https://www.digitalocean.com/?refcode=df14bcb7c016&utm_campaign=Referral_Invite&utm_medium=Referral_Program&utm_source=badge"><img src="https://web-platforms.sfo2.cdn.digitaloceanspaces.com/WWW/Badge%203.svg" alt="DigitalOcean Referral Badge" /></a>
</p>
<p align="center">
<em>Homepage builds are kindly powered by DigitalOcean.</em>
</p>

View File

@@ -8,5 +8,5 @@ Learn more about [Channels DVR Server](https://getchannels.com/dvr-server/).
```yaml ```yaml
widget: widget:
type: channelsdvrserver type: channelsdvrserver
url: http://192.168.1.55:8089 url: http://server.host.or.ip:port
``` ```

View File

@@ -0,0 +1,14 @@
---
title: SWAG Dashboard
description: SWAG Dashboard Widget Configuration
---
Learn more about [SWAG Dashboard](https://github.com/linuxserver/docker-mods/tree/swag-dashboard).
Allowed fields: `["proxied", "auth", "outdated", "banned"]`.
```yaml
widget:
type: swagdashboard
url: http://swagdashboard.host.or.ip:adminport # default port is 81
```

View File

@@ -0,0 +1,20 @@
---
title: Wg-Easy
description: Wg-Easy Widget Configuration
---
Learn more about [Wg-Easy](https://github.com/wg-easy/wg-easy).
Allowed fields: `["connected", "enabled", "disabled", "total"]`.
Note: by default `["connected", "enabled", "total"]` are displayed.
To detect if a device is connected the time since the last handshake is queried. `threshold` is the time to wait in minutes since the last handshake to consider a device connected. Default is 2 minutes.
```yaml
widget:
type: wgeasy
url: http://wg.easy.or.ip
password: yourwgeasypassword
threshold: 2 # optional
```

View File

@@ -129,6 +129,7 @@ nav:
- widgets/services/sonarr.md - widgets/services/sonarr.md
- widgets/services/speedtest-tracker.md - widgets/services/speedtest-tracker.md
- widgets/services/stash.md - widgets/services/stash.md
- widgets/services/swagdashboard.md
- widgets/services/syncthing-relay-server.md - widgets/services/syncthing-relay-server.md
- widgets/services/tailscale.md - widgets/services/tailscale.md
- widgets/services/tandoor.md - widgets/services/tandoor.md

161
package-lock.json generated
View File

@@ -50,8 +50,8 @@
"eslint-plugin-import": "^2.29.1", "eslint-plugin-import": "^2.29.1",
"eslint-plugin-jsx-a11y": "^6.8.0", "eslint-plugin-jsx-a11y": "^6.8.0",
"eslint-plugin-prettier": "^4.2.1", "eslint-plugin-prettier": "^4.2.1",
"eslint-plugin-react": "^7.34.1", "eslint-plugin-react": "^7.34.2",
"eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-react-hooks": "^4.6.2",
"postcss": "^8.4.38", "postcss": "^8.4.38",
"prettier": "^3.2.5", "prettier": "^3.2.5",
"tailwind-scrollbar": "^3.0.5", "tailwind-scrollbar": "^3.0.5",
@@ -1052,15 +1052,16 @@
} }
}, },
"node_modules/array-includes": { "node_modules/array-includes": {
"version": "3.1.7", "version": "3.1.8",
"resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.7.tgz", "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz",
"integrity": "sha512-dlcsNBIiWhPkHdOEEKnehA+RNUWDc4UqFtnIXU4uuYDPtA4LDkr7qip2p0VvFAEXNDr0yWZ9PJyIRiGjRLQzwQ==", "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.2", "call-bind": "^1.0.7",
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1", "es-abstract": "^1.23.2",
"get-intrinsic": "^1.2.1", "es-object-atoms": "^1.0.0",
"get-intrinsic": "^1.2.4",
"is-string": "^1.0.7" "is-string": "^1.0.7"
}, },
"engines": { "engines": {
@@ -2470,14 +2471,14 @@
} }
}, },
"node_modules/es-iterator-helpers": { "node_modules/es-iterator-helpers": {
"version": "1.0.18", "version": "1.0.19",
"resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.18.tgz", "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.19.tgz",
"integrity": "sha512-scxAJaewsahbqTYrGKJihhViaM6DDZDDoucfvzNbK0pOren1g/daDQ3IAhzn+1G14rBG7w+i5N+qul60++zlKA==", "integrity": "sha512-zoMwbCcH5hwUkKJkT8kDIBZSz9I6mVG//+lDCinLCGov4+r7NIy0ld8o03M0cJxl2spVf6ESYVS6/gpIfq1FFw==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.7", "call-bind": "^1.0.7",
"define-properties": "^1.2.1", "define-properties": "^1.2.1",
"es-abstract": "^1.23.0", "es-abstract": "^1.23.3",
"es-errors": "^1.3.0", "es-errors": "^1.3.0",
"es-set-tostringtag": "^2.0.3", "es-set-tostringtag": "^2.0.3",
"function-bind": "^1.1.2", "function-bind": "^1.1.2",
@@ -2875,29 +2876,29 @@
} }
}, },
"node_modules/eslint-plugin-react": { "node_modules/eslint-plugin-react": {
"version": "7.34.1", "version": "7.34.2",
"resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.34.1.tgz", "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.34.2.tgz",
"integrity": "sha512-N97CxlouPT1AHt8Jn0mhhN2RrADlUAsk1/atcT2KyA/l9Q/E6ll7OIGwNumFmWfZ9skV3XXccYS19h80rHtgkw==", "integrity": "sha512-2HCmrU+/JNigDN6tg55cRDKCQWicYAPB38JGSFDQt95jDm8rrvSUo7YPkOIm5l6ts1j1zCvysNcasvfTMQzUOw==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"array-includes": "^3.1.7", "array-includes": "^3.1.8",
"array.prototype.findlast": "^1.2.4", "array.prototype.findlast": "^1.2.5",
"array.prototype.flatmap": "^1.3.2", "array.prototype.flatmap": "^1.3.2",
"array.prototype.toreversed": "^1.1.2", "array.prototype.toreversed": "^1.1.2",
"array.prototype.tosorted": "^1.1.3", "array.prototype.tosorted": "^1.1.3",
"doctrine": "^2.1.0", "doctrine": "^2.1.0",
"es-iterator-helpers": "^1.0.17", "es-iterator-helpers": "^1.0.19",
"estraverse": "^5.3.0", "estraverse": "^5.3.0",
"jsx-ast-utils": "^2.4.1 || ^3.0.0", "jsx-ast-utils": "^2.4.1 || ^3.0.0",
"minimatch": "^3.1.2", "minimatch": "^3.1.2",
"object.entries": "^1.1.7", "object.entries": "^1.1.8",
"object.fromentries": "^2.0.7", "object.fromentries": "^2.0.8",
"object.hasown": "^1.1.3", "object.hasown": "^1.1.4",
"object.values": "^1.1.7", "object.values": "^1.2.0",
"prop-types": "^15.8.1", "prop-types": "^15.8.1",
"resolve": "^2.0.0-next.5", "resolve": "^2.0.0-next.5",
"semver": "^6.3.1", "semver": "^6.3.1",
"string.prototype.matchall": "^4.0.10" "string.prototype.matchall": "^4.0.11"
}, },
"engines": { "engines": {
"node": ">=4" "node": ">=4"
@@ -2907,9 +2908,9 @@
} }
}, },
"node_modules/eslint-plugin-react-hooks": { "node_modules/eslint-plugin-react-hooks": {
"version": "4.6.0", "version": "4.6.2",
"resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.0.tgz", "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-4.6.2.tgz",
"integrity": "sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==", "integrity": "sha512-QzliNJq4GinDBcD8gPB5v0wh6g8q3SUi6EFF0x8N/BL9PoVs0atuGc47ozMRyOWAKdwaZ5OnbOEa3WR+dSGKuQ==",
"dev": true, "dev": true,
"engines": { "engines": {
"node": ">=10" "node": ">=10"
@@ -5108,28 +5109,29 @@
} }
}, },
"node_modules/object.entries": { "node_modules/object.entries": {
"version": "1.1.7", "version": "1.1.8",
"resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.7.tgz", "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.8.tgz",
"integrity": "sha512-jCBs/0plmPsOnrKAfFQXRG2NFjlhZgjjcBLSmTnEhU8U6vVTsVe8ANeQJCHTl3gSsI4J+0emOoCgoKlmQPMgmA==", "integrity": "sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.2", "call-bind": "^1.0.7",
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1" "es-object-atoms": "^1.0.0"
}, },
"engines": { "engines": {
"node": ">= 0.4" "node": ">= 0.4"
} }
}, },
"node_modules/object.fromentries": { "node_modules/object.fromentries": {
"version": "2.0.7", "version": "2.0.8",
"resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.7.tgz", "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz",
"integrity": "sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==", "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.2", "call-bind": "^1.0.7",
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1" "es-abstract": "^1.23.2",
"es-object-atoms": "^1.0.0"
}, },
"engines": { "engines": {
"node": ">= 0.4" "node": ">= 0.4"
@@ -5151,27 +5153,31 @@
} }
}, },
"node_modules/object.hasown": { "node_modules/object.hasown": {
"version": "1.1.3", "version": "1.1.4",
"resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.3.tgz", "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.4.tgz",
"integrity": "sha512-fFI4VcYpRHvSLXxP7yiZOMAd331cPfd2p7PFDVbgUsYOfCT3tICVqXWngbjr4m49OvsBwUBQ6O2uQoJvy3RexA==", "integrity": "sha512-FZ9LZt9/RHzGySlBARE3VF+gE26TxR38SdmqOqliuTnl9wrKulaQs+4dee1V+Io8VfxqzAfHu6YuRgUy8OHoTg==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1" "es-abstract": "^1.23.2",
"es-object-atoms": "^1.0.0"
},
"engines": {
"node": ">= 0.4"
}, },
"funding": { "funding": {
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"
} }
}, },
"node_modules/object.values": { "node_modules/object.values": {
"version": "1.1.7", "version": "1.2.0",
"resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.7.tgz", "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.0.tgz",
"integrity": "sha512-aU6xnDFYT3x17e/f0IiiwlGPTy2jzMySGfUB4fq6z7CV8l85CWHDk5ErhyhpfDHhrOMwGFhSQkhMGHaIotA6Ng==", "integrity": "sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.2", "call-bind": "^1.0.7",
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1" "es-object-atoms": "^1.0.0"
}, },
"engines": { "engines": {
"node": ">= 0.4" "node": ">= 0.4"
@@ -6202,14 +6208,15 @@
} }
}, },
"node_modules/set-function-name": { "node_modules/set-function-name": {
"version": "2.0.1", "version": "2.0.2",
"resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.1.tgz", "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz",
"integrity": "sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA==", "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"define-data-property": "^1.0.1", "define-data-property": "^1.1.4",
"es-errors": "^1.3.0",
"functions-have-names": "^1.2.3", "functions-have-names": "^1.2.3",
"has-property-descriptors": "^1.0.0" "has-property-descriptors": "^1.0.2"
}, },
"engines": { "engines": {
"node": ">= 0.4" "node": ">= 0.4"
@@ -6256,14 +6263,18 @@
} }
}, },
"node_modules/side-channel": { "node_modules/side-channel": {
"version": "1.0.4", "version": "1.0.6",
"resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz",
"integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.0", "call-bind": "^1.0.7",
"get-intrinsic": "^1.0.2", "es-errors": "^1.3.0",
"object-inspect": "^1.9.0" "get-intrinsic": "^1.2.4",
"object-inspect": "^1.13.1"
},
"engines": {
"node": ">= 0.4"
}, },
"funding": { "funding": {
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"
@@ -6484,20 +6495,26 @@
} }
}, },
"node_modules/string.prototype.matchall": { "node_modules/string.prototype.matchall": {
"version": "4.0.10", "version": "4.0.11",
"resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz", "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.11.tgz",
"integrity": "sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==", "integrity": "sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==",
"dev": true, "dev": true,
"dependencies": { "dependencies": {
"call-bind": "^1.0.2", "call-bind": "^1.0.7",
"define-properties": "^1.2.0", "define-properties": "^1.2.1",
"es-abstract": "^1.22.1", "es-abstract": "^1.23.2",
"get-intrinsic": "^1.2.1", "es-errors": "^1.3.0",
"es-object-atoms": "^1.0.0",
"get-intrinsic": "^1.2.4",
"gopd": "^1.0.1",
"has-symbols": "^1.0.3", "has-symbols": "^1.0.3",
"internal-slot": "^1.0.5", "internal-slot": "^1.0.7",
"regexp.prototype.flags": "^1.5.0", "regexp.prototype.flags": "^1.5.2",
"set-function-name": "^2.0.0", "set-function-name": "^2.0.2",
"side-channel": "^1.0.4" "side-channel": "^1.0.6"
},
"engines": {
"node": ">= 0.4"
}, },
"funding": { "funding": {
"url": "https://github.com/sponsors/ljharb" "url": "https://github.com/sponsors/ljharb"

View File

@@ -52,8 +52,8 @@
"eslint-plugin-import": "^2.29.1", "eslint-plugin-import": "^2.29.1",
"eslint-plugin-jsx-a11y": "^6.8.0", "eslint-plugin-jsx-a11y": "^6.8.0",
"eslint-plugin-prettier": "^4.2.1", "eslint-plugin-prettier": "^4.2.1",
"eslint-plugin-react": "^7.34.1", "eslint-plugin-react": "^7.34.2",
"eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-react-hooks": "^4.6.2",
"postcss": "^8.4.38", "postcss": "^8.4.38",
"prettier": "^3.2.5", "prettier": "^3.2.5",
"tailwind-scrollbar": "^3.0.5", "tailwind-scrollbar": "^3.0.5",

444
pnpm-lock.yaml generated
View File

@@ -116,7 +116,7 @@ devDependencies:
version: 8.57.0 version: 8.57.0
eslint-config-airbnb: eslint-config-airbnb:
specifier: ^19.0.4 specifier: ^19.0.4
version: 19.0.4(eslint-plugin-import@2.29.1)(eslint-plugin-jsx-a11y@6.8.0)(eslint-plugin-react-hooks@4.6.0)(eslint-plugin-react@7.34.1)(eslint@8.57.0) version: 19.0.4(eslint-plugin-import@2.29.1)(eslint-plugin-jsx-a11y@6.8.0)(eslint-plugin-react-hooks@4.6.2)(eslint-plugin-react@7.34.2)(eslint@8.57.0)
eslint-config-next: eslint-config-next:
specifier: ^14.2.3 specifier: ^14.2.3
version: 14.2.3(eslint@8.57.0)(typescript@5.4.5) version: 14.2.3(eslint@8.57.0)(typescript@5.4.5)
@@ -133,11 +133,11 @@ devDependencies:
specifier: ^4.2.1 specifier: ^4.2.1
version: 4.2.1(eslint-config-prettier@9.1.0)(eslint@8.57.0)(prettier@3.2.5) version: 4.2.1(eslint-config-prettier@9.1.0)(eslint@8.57.0)(prettier@3.2.5)
eslint-plugin-react: eslint-plugin-react:
specifier: ^7.34.1 specifier: ^7.34.2
version: 7.34.1(eslint@8.57.0) version: 7.34.2(eslint@8.57.0)
eslint-plugin-react-hooks: eslint-plugin-react-hooks:
specifier: ^4.6.0 specifier: ^4.6.2
version: 4.6.0(eslint@8.57.0) version: 4.6.2(eslint@8.57.0)
postcss: postcss:
specifier: ^8.4.38 specifier: ^8.4.38
version: 8.4.38 version: 8.4.38
@@ -737,13 +737,6 @@ packages:
dequal: 2.0.3 dequal: 2.0.3
dev: true dev: true
/array-buffer-byte-length@1.0.0:
resolution: {integrity: sha512-LPuwb2P+NrQw3XhxGc36+XSvuBPopovXYTR9Ew++Du9Yb/bx5AzBfrIsBoj0EZUifjQU+sHL21sseZ3jerWO/A==}
dependencies:
call-bind: 1.0.5
is-array-buffer: 3.0.2
dev: true
/array-buffer-byte-length@1.0.1: /array-buffer-byte-length@1.0.1:
resolution: {integrity: sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==} resolution: {integrity: sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -763,6 +756,18 @@ packages:
is-string: 1.0.7 is-string: 1.0.7
dev: true dev: true
/array-includes@3.1.8:
resolution: {integrity: sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.7
define-properties: 1.2.1
es-abstract: 1.23.3
es-object-atoms: 1.0.0
get-intrinsic: 1.2.4
is-string: 1.0.7
dev: true
/array-union@2.1.0: /array-union@2.1.0:
resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==} resolution: {integrity: sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==}
engines: {node: '>=8'} engines: {node: '>=8'}
@@ -805,44 +810,31 @@ packages:
resolution: {integrity: sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==} resolution: {integrity: sha512-Ewyx0c9PmpcsByhSW4r+9zDU7sGjFc86qf/kKtuSCRdhfbk0SNLLkaT5qvcHnRGgc5NP/ly/y+qkXkqONX54CQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
define-properties: 1.2.1 define-properties: 1.2.1
es-abstract: 1.22.3 es-abstract: 1.23.3
es-shim-unscopables: 1.0.2 es-shim-unscopables: 1.0.2
dev: true dev: true
/array.prototype.toreversed@1.1.2: /array.prototype.toreversed@1.1.2:
resolution: {integrity: sha512-wwDCoT4Ck4Cz7sLtgUmzR5UV3YF5mFHUlbChCzZBQZ+0m2cl/DH3tKgvphv1nKgFsJ48oCSg6p91q2Vm0I/ZMA==} resolution: {integrity: sha512-wwDCoT4Ck4Cz7sLtgUmzR5UV3YF5mFHUlbChCzZBQZ+0m2cl/DH3tKgvphv1nKgFsJ48oCSg6p91q2Vm0I/ZMA==}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
define-properties: 1.2.1 define-properties: 1.2.1
es-abstract: 1.22.3 es-abstract: 1.23.3
es-shim-unscopables: 1.0.2 es-shim-unscopables: 1.0.2
dev: true dev: true
/array.prototype.tosorted@1.1.3: /array.prototype.tosorted@1.1.3:
resolution: {integrity: sha512-/DdH4TiTmOKzyQbp/eadcCVexiCb36xJg7HshYOYJnNZFDj33GEv0P7GxsynpShhq4OLYJzbGcBDkLsDt7MnNg==} resolution: {integrity: sha512-/DdH4TiTmOKzyQbp/eadcCVexiCb36xJg7HshYOYJnNZFDj33GEv0P7GxsynpShhq4OLYJzbGcBDkLsDt7MnNg==}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
define-properties: 1.2.1 define-properties: 1.2.1
es-abstract: 1.22.3 es-abstract: 1.23.3
es-errors: 1.3.0 es-errors: 1.3.0
es-shim-unscopables: 1.0.2 es-shim-unscopables: 1.0.2
dev: true dev: true
/arraybuffer.prototype.slice@1.0.2:
resolution: {integrity: sha512-yMBKppFur/fbHu9/6USUe03bZ4knMYiwFBcyiaXB8Go0qNehwX6inYPzK9U0NeQvGxKthcmHcaR8P5MStSRBAw==}
engines: {node: '>= 0.4'}
dependencies:
array-buffer-byte-length: 1.0.0
call-bind: 1.0.5
define-properties: 1.2.1
es-abstract: 1.22.3
get-intrinsic: 1.2.2
is-array-buffer: 3.0.2
is-shared-array-buffer: 1.0.2
dev: true
/arraybuffer.prototype.slice@1.0.3: /arraybuffer.prototype.slice@1.0.3:
resolution: {integrity: sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==} resolution: {integrity: sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -908,11 +900,6 @@ packages:
postcss-value-parser: 4.2.0 postcss-value-parser: 4.2.0
dev: true dev: true
/available-typed-arrays@1.0.5:
resolution: {integrity: sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==}
engines: {node: '>= 0.4'}
dev: true
/available-typed-arrays@1.0.7: /available-typed-arrays@1.0.7:
resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -1058,8 +1045,8 @@ packages:
resolution: {integrity: sha512-C3nQxfFZxFRVoJoGKKI8y3MOEo129NQ+FgQ08iye+Mk4zNZZGdjfs06bVTr+DBSlA66Q2VEcMki/cUCP4SercQ==} resolution: {integrity: sha512-C3nQxfFZxFRVoJoGKKI8y3MOEo129NQ+FgQ08iye+Mk4zNZZGdjfs06bVTr+DBSlA66Q2VEcMki/cUCP4SercQ==}
dependencies: dependencies:
function-bind: 1.1.2 function-bind: 1.1.2
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
set-function-length: 1.2.0 set-function-length: 1.2.2
dev: true dev: true
/call-bind@1.0.7: /call-bind@1.0.7:
@@ -1435,15 +1422,6 @@ packages:
engines: {node: '>=10'} engines: {node: '>=10'}
dev: false dev: false
/define-data-property@1.1.1:
resolution: {integrity: sha512-E7uGkTzkk1d0ByLeSc6ZsFS79Axg+m1P/VsgYsxHgiuc3tFSj+MjMIwe90FC4lOAZzNBdY7kkO2P2wKdsQ1vgQ==}
engines: {node: '>= 0.4'}
dependencies:
get-intrinsic: 1.2.2
gopd: 1.0.1
has-property-descriptors: 1.0.1
dev: true
/define-data-property@1.1.4: /define-data-property@1.1.4:
resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -1457,8 +1435,8 @@ packages:
resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==} resolution: {integrity: sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
define-data-property: 1.1.1 define-data-property: 1.1.4
has-property-descriptors: 1.0.1 has-property-descriptors: 1.0.2
object-keys: 1.1.1 object-keys: 1.1.1
dev: true dev: true
@@ -1613,45 +1591,45 @@ packages:
resolution: {integrity: sha512-eiiY8HQeYfYH2Con2berK+To6GrK2RxbPawDkGq4UiCQQfZHb6wX9qQqkbpPqaxQFcl8d9QzZqo0tGE0VcrdwA==} resolution: {integrity: sha512-eiiY8HQeYfYH2Con2berK+To6GrK2RxbPawDkGq4UiCQQfZHb6wX9qQqkbpPqaxQFcl8d9QzZqo0tGE0VcrdwA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
array-buffer-byte-length: 1.0.0 array-buffer-byte-length: 1.0.1
arraybuffer.prototype.slice: 1.0.2 arraybuffer.prototype.slice: 1.0.3
available-typed-arrays: 1.0.5 available-typed-arrays: 1.0.7
call-bind: 1.0.5 call-bind: 1.0.7
es-set-tostringtag: 2.0.2 es-set-tostringtag: 2.0.3
es-to-primitive: 1.2.1 es-to-primitive: 1.2.1
function.prototype.name: 1.1.6 function.prototype.name: 1.1.6
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
get-symbol-description: 1.0.0 get-symbol-description: 1.0.2
globalthis: 1.0.3 globalthis: 1.0.3
gopd: 1.0.1 gopd: 1.0.1
has-property-descriptors: 1.0.1 has-property-descriptors: 1.0.2
has-proto: 1.0.1 has-proto: 1.0.3
has-symbols: 1.0.3 has-symbols: 1.0.3
hasown: 2.0.0 hasown: 2.0.2
internal-slot: 1.0.6 internal-slot: 1.0.7
is-array-buffer: 3.0.2 is-array-buffer: 3.0.4
is-callable: 1.2.7 is-callable: 1.2.7
is-negative-zero: 2.0.2 is-negative-zero: 2.0.3
is-regex: 1.1.4 is-regex: 1.1.4
is-shared-array-buffer: 1.0.2 is-shared-array-buffer: 1.0.3
is-string: 1.0.7 is-string: 1.0.7
is-typed-array: 1.1.12 is-typed-array: 1.1.13
is-weakref: 1.0.2 is-weakref: 1.0.2
object-inspect: 1.13.1 object-inspect: 1.13.1
object-keys: 1.1.1 object-keys: 1.1.1
object.assign: 4.1.5 object.assign: 4.1.5
regexp.prototype.flags: 1.5.1 regexp.prototype.flags: 1.5.2
safe-array-concat: 1.1.0 safe-array-concat: 1.1.2
safe-regex-test: 1.0.2 safe-regex-test: 1.0.3
string.prototype.trim: 1.2.8 string.prototype.trim: 1.2.9
string.prototype.trimend: 1.0.7 string.prototype.trimend: 1.0.8
string.prototype.trimstart: 1.0.7 string.prototype.trimstart: 1.0.8
typed-array-buffer: 1.0.0 typed-array-buffer: 1.0.2
typed-array-byte-length: 1.0.0 typed-array-byte-length: 1.0.1
typed-array-byte-offset: 1.0.0 typed-array-byte-offset: 1.0.2
typed-array-length: 1.0.4 typed-array-length: 1.0.6
unbox-primitive: 1.0.2 unbox-primitive: 1.0.2
which-typed-array: 1.1.13 which-typed-array: 1.1.15
dev: true dev: true
/es-abstract@1.23.3: /es-abstract@1.23.3:
@@ -1737,8 +1715,8 @@ packages:
safe-array-concat: 1.1.0 safe-array-concat: 1.1.0
dev: true dev: true
/es-iterator-helpers@1.0.18: /es-iterator-helpers@1.0.19:
resolution: {integrity: sha512-scxAJaewsahbqTYrGKJihhViaM6DDZDDoucfvzNbK0pOren1g/daDQ3IAhzn+1G14rBG7w+i5N+qul60++zlKA==} resolution: {integrity: sha512-zoMwbCcH5hwUkKJkT8kDIBZSz9I6mVG//+lDCinLCGov4+r7NIy0ld8o03M0cJxl2spVf6ESYVS6/gpIfq1FFw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
call-bind: 1.0.7 call-bind: 1.0.7
@@ -1768,9 +1746,9 @@ packages:
resolution: {integrity: sha512-BuDyupZt65P9D2D2vA/zqcI3G5xRsklm5N3xCwuiy+/vKy8i0ifdsQP1sLgO4tZDSCaQUSnmC48khknGMV3D2Q==} resolution: {integrity: sha512-BuDyupZt65P9D2D2vA/zqcI3G5xRsklm5N3xCwuiy+/vKy8i0ifdsQP1sLgO4tZDSCaQUSnmC48khknGMV3D2Q==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
hasown: 2.0.0 hasown: 2.0.2
dev: true dev: true
/es-set-tostringtag@2.0.3: /es-set-tostringtag@2.0.3:
@@ -1785,7 +1763,7 @@ packages:
/es-shim-unscopables@1.0.2: /es-shim-unscopables@1.0.2:
resolution: {integrity: sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==} resolution: {integrity: sha512-J3yBRXCzDu4ULnQwxyToo/OjdMx6akgVC7K6few0a7F/0wLtmKKN7I73AH5T2836UuXRqN7Qg+IIUw/+YJksRw==}
dependencies: dependencies:
hasown: 2.0.0 hasown: 2.0.2
dev: true dev: true
/es-to-primitive@1.2.1: /es-to-primitive@1.2.1:
@@ -1822,7 +1800,7 @@ packages:
semver: 6.3.1 semver: 6.3.1
dev: true dev: true
/eslint-config-airbnb@19.0.4(eslint-plugin-import@2.29.1)(eslint-plugin-jsx-a11y@6.8.0)(eslint-plugin-react-hooks@4.6.0)(eslint-plugin-react@7.34.1)(eslint@8.57.0): /eslint-config-airbnb@19.0.4(eslint-plugin-import@2.29.1)(eslint-plugin-jsx-a11y@6.8.0)(eslint-plugin-react-hooks@4.6.2)(eslint-plugin-react@7.34.2)(eslint@8.57.0):
resolution: {integrity: sha512-T75QYQVQX57jiNgpF9r1KegMICE94VYwoFQyMGhrvc+lB8YF2E/M/PYDaQe1AJcWaEgqLE+ErXV1Og/+6Vyzew==} resolution: {integrity: sha512-T75QYQVQX57jiNgpF9r1KegMICE94VYwoFQyMGhrvc+lB8YF2E/M/PYDaQe1AJcWaEgqLE+ErXV1Og/+6Vyzew==}
engines: {node: ^10.12.0 || ^12.22.0 || ^14.17.0 || >=16.0.0} engines: {node: ^10.12.0 || ^12.22.0 || ^14.17.0 || >=16.0.0}
peerDependencies: peerDependencies:
@@ -1836,8 +1814,8 @@ packages:
eslint-config-airbnb-base: 15.0.0(eslint-plugin-import@2.29.1)(eslint@8.57.0) eslint-config-airbnb-base: 15.0.0(eslint-plugin-import@2.29.1)(eslint@8.57.0)
eslint-plugin-import: 2.29.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) eslint-plugin-import: 2.29.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0)
eslint-plugin-jsx-a11y: 6.8.0(eslint@8.57.0) eslint-plugin-jsx-a11y: 6.8.0(eslint@8.57.0)
eslint-plugin-react: 7.34.1(eslint@8.57.0) eslint-plugin-react: 7.34.2(eslint@8.57.0)
eslint-plugin-react-hooks: 4.6.0(eslint@8.57.0) eslint-plugin-react-hooks: 4.6.2(eslint@8.57.0)
object.assign: 4.1.5 object.assign: 4.1.5
object.entries: 1.1.7 object.entries: 1.1.7
dev: true dev: true
@@ -1859,8 +1837,8 @@ packages:
eslint-import-resolver-typescript: 3.6.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.29.1)(eslint@8.57.0) eslint-import-resolver-typescript: 3.6.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-node@0.3.9)(eslint-plugin-import@2.29.1)(eslint@8.57.0)
eslint-plugin-import: 2.29.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0) eslint-plugin-import: 2.29.1(@typescript-eslint/parser@5.62.0)(eslint-import-resolver-typescript@3.6.1)(eslint@8.57.0)
eslint-plugin-jsx-a11y: 6.8.0(eslint@8.57.0) eslint-plugin-jsx-a11y: 6.8.0(eslint@8.57.0)
eslint-plugin-react: 7.34.1(eslint@8.57.0) eslint-plugin-react: 7.34.2(eslint@8.57.0)
eslint-plugin-react-hooks: 4.6.0(eslint@8.57.0) eslint-plugin-react-hooks: 4.6.2(eslint@8.57.0)
typescript: 5.4.5 typescript: 5.4.5
transitivePeerDependencies: transitivePeerDependencies:
- eslint-import-resolver-webpack - eslint-import-resolver-webpack
@@ -2016,8 +1994,8 @@ packages:
prettier-linter-helpers: 1.0.0 prettier-linter-helpers: 1.0.0
dev: true dev: true
/eslint-plugin-react-hooks@4.6.0(eslint@8.57.0): /eslint-plugin-react-hooks@4.6.2(eslint@8.57.0):
resolution: {integrity: sha512-oFc7Itz9Qxh2x4gNHStv3BqJq54ExXmfC+a1NjAta66IAN87Wu0R/QArgIS9qKzX3dXKPI9H5crl9QchNMY9+g==} resolution: {integrity: sha512-QzliNJq4GinDBcD8gPB5v0wh6g8q3SUi6EFF0x8N/BL9PoVs0atuGc47ozMRyOWAKdwaZ5OnbOEa3WR+dSGKuQ==}
engines: {node: '>=10'} engines: {node: '>=10'}
peerDependencies: peerDependencies:
eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 eslint: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0
@@ -2025,31 +2003,31 @@ packages:
eslint: 8.57.0 eslint: 8.57.0
dev: true dev: true
/eslint-plugin-react@7.34.1(eslint@8.57.0): /eslint-plugin-react@7.34.2(eslint@8.57.0):
resolution: {integrity: sha512-N97CxlouPT1AHt8Jn0mhhN2RrADlUAsk1/atcT2KyA/l9Q/E6ll7OIGwNumFmWfZ9skV3XXccYS19h80rHtgkw==} resolution: {integrity: sha512-2HCmrU+/JNigDN6tg55cRDKCQWicYAPB38JGSFDQt95jDm8rrvSUo7YPkOIm5l6ts1j1zCvysNcasvfTMQzUOw==}
engines: {node: '>=4'} engines: {node: '>=4'}
peerDependencies: peerDependencies:
eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8 eslint: ^3 || ^4 || ^5 || ^6 || ^7 || ^8
dependencies: dependencies:
array-includes: 3.1.7 array-includes: 3.1.8
array.prototype.findlast: 1.2.5 array.prototype.findlast: 1.2.5
array.prototype.flatmap: 1.3.2 array.prototype.flatmap: 1.3.2
array.prototype.toreversed: 1.1.2 array.prototype.toreversed: 1.1.2
array.prototype.tosorted: 1.1.3 array.prototype.tosorted: 1.1.3
doctrine: 2.1.0 doctrine: 2.1.0
es-iterator-helpers: 1.0.18 es-iterator-helpers: 1.0.19
eslint: 8.57.0 eslint: 8.57.0
estraverse: 5.3.0 estraverse: 5.3.0
jsx-ast-utils: 3.3.5 jsx-ast-utils: 3.3.5
minimatch: 3.1.2 minimatch: 3.1.2
object.entries: 1.1.7 object.entries: 1.1.8
object.fromentries: 2.0.7 object.fromentries: 2.0.8
object.hasown: 1.1.3 object.hasown: 1.1.4
object.values: 1.1.7 object.values: 1.2.0
prop-types: 15.8.1 prop-types: 15.8.1
resolve: 2.0.0-next.5 resolve: 2.0.0-next.5
semver: 6.3.1 semver: 6.3.1
string.prototype.matchall: 4.0.10 string.prototype.matchall: 4.0.11
dev: true dev: true
/eslint-scope@7.2.2: /eslint-scope@7.2.2:
@@ -2373,9 +2351,9 @@ packages:
resolution: {integrity: sha512-0gSo4ml/0j98Y3lngkFEot/zhiCeWsbYIlZ+uZOVgzLyLaUw7wxUL+nCTP0XJvJg1AXulJRI3UJi8GsbDuxdGA==} resolution: {integrity: sha512-0gSo4ml/0j98Y3lngkFEot/zhiCeWsbYIlZ+uZOVgzLyLaUw7wxUL+nCTP0XJvJg1AXulJRI3UJi8GsbDuxdGA==}
dependencies: dependencies:
function-bind: 1.1.2 function-bind: 1.1.2
has-proto: 1.0.1 has-proto: 1.0.3
has-symbols: 1.0.3 has-symbols: 1.0.3
hasown: 2.0.0 hasown: 2.0.2
dev: true dev: true
/get-intrinsic@1.2.4: /get-intrinsic@1.2.4:
@@ -2386,7 +2364,7 @@ packages:
function-bind: 1.1.2 function-bind: 1.1.2
has-proto: 1.0.3 has-proto: 1.0.3
has-symbols: 1.0.3 has-symbols: 1.0.3
hasown: 2.0.0 hasown: 2.0.2
dev: true dev: true
/get-stream@6.0.1: /get-stream@6.0.1:
@@ -2394,14 +2372,6 @@ packages:
engines: {node: '>=10'} engines: {node: '>=10'}
dev: false dev: false
/get-symbol-description@1.0.0:
resolution: {integrity: sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.5
get-intrinsic: 1.2.2
dev: true
/get-symbol-description@1.0.2: /get-symbol-description@1.0.2:
resolution: {integrity: sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==} resolution: {integrity: sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -2542,7 +2512,7 @@ packages:
/has-property-descriptors@1.0.1: /has-property-descriptors@1.0.1:
resolution: {integrity: sha512-VsX8eaIewvas0xnvinAe9bw4WfIeODpGYikiWYLH+dma0Jw6KHYqWiWfhQlgOVK8D6PvjubK5Uc4P0iIhIcNVg==} resolution: {integrity: sha512-VsX8eaIewvas0xnvinAe9bw4WfIeODpGYikiWYLH+dma0Jw6KHYqWiWfhQlgOVK8D6PvjubK5Uc4P0iIhIcNVg==}
dependencies: dependencies:
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
dev: true dev: true
/has-property-descriptors@1.0.2: /has-property-descriptors@1.0.2:
@@ -2566,13 +2536,6 @@ packages:
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dev: true dev: true
/has-tostringtag@1.0.0:
resolution: {integrity: sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==}
engines: {node: '>= 0.4'}
dependencies:
has-symbols: 1.0.3
dev: true
/has-tostringtag@1.0.2: /has-tostringtag@1.0.2:
resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==} resolution: {integrity: sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -2714,8 +2677,8 @@ packages:
resolution: {integrity: sha512-Xj6dv+PsbtwyPpEflsejS+oIZxmMlV44zAhG479uYu89MsjcYOhCFnNyKrkJrihbsiasQyY0afoCl/9BLR65bg==} resolution: {integrity: sha512-Xj6dv+PsbtwyPpEflsejS+oIZxmMlV44zAhG479uYu89MsjcYOhCFnNyKrkJrihbsiasQyY0afoCl/9BLR65bg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
hasown: 2.0.0 hasown: 2.0.2
side-channel: 1.0.4 side-channel: 1.0.4
dev: true dev: true
@@ -2724,7 +2687,7 @@ packages:
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
es-errors: 1.3.0 es-errors: 1.3.0
hasown: 2.0.0 hasown: 2.0.2
side-channel: 1.0.4 side-channel: 1.0.4
dev: true dev: true
@@ -2738,14 +2701,6 @@ packages:
engines: {node: '>= 0.10'} engines: {node: '>= 0.10'}
dev: false dev: false
/is-array-buffer@3.0.2:
resolution: {integrity: sha512-y+FyyR/w8vfIRq4eQcM1EYgSTnmHXPqaF+IgzgraytCFq5Xh8lllDVmAZolPJiZttZLeFSINPYMaEJ7/vWUa1w==}
dependencies:
call-bind: 1.0.5
get-intrinsic: 1.2.2
is-typed-array: 1.1.12
dev: true
/is-array-buffer@3.0.4: /is-array-buffer@3.0.4:
resolution: {integrity: sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==} resolution: {integrity: sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -2762,7 +2717,7 @@ packages:
resolution: {integrity: sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==} resolution: {integrity: sha512-Y1JXKrfykRJGdlDwdKlLpLyMIiWqWvuSd17TvZk68PLAOGOoF4Xyav1z0Xhoi+gCYjZVeC5SI+hYFOfvXmGRCA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-bigint@1.0.4: /is-bigint@1.0.4:
@@ -2783,7 +2738,7 @@ packages:
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
call-bind: 1.0.7 call-bind: 1.0.7
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-callable@1.2.7: /is-callable@1.2.7:
@@ -2807,7 +2762,7 @@ packages:
resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==} resolution: {integrity: sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-extglob@2.1.1: /is-extglob@2.1.1:
@@ -2830,7 +2785,7 @@ packages:
resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==} resolution: {integrity: sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-glob@4.0.3: /is-glob@4.0.3:
@@ -2844,11 +2799,6 @@ packages:
resolution: {integrity: sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==} resolution: {integrity: sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==}
dev: true dev: true
/is-negative-zero@2.0.2:
resolution: {integrity: sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==}
engines: {node: '>= 0.4'}
dev: true
/is-negative-zero@2.0.3: /is-negative-zero@2.0.3:
resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==} resolution: {integrity: sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -2858,7 +2808,7 @@ packages:
resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==} resolution: {integrity: sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-number@7.0.0: /is-number@7.0.0:
@@ -2876,19 +2826,13 @@ packages:
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
call-bind: 1.0.7 call-bind: 1.0.7
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-set@2.0.2: /is-set@2.0.2:
resolution: {integrity: sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==} resolution: {integrity: sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==}
dev: true dev: true
/is-shared-array-buffer@1.0.2:
resolution: {integrity: sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==}
dependencies:
call-bind: 1.0.5
dev: true
/is-shared-array-buffer@1.0.3: /is-shared-array-buffer@1.0.3:
resolution: {integrity: sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==} resolution: {integrity: sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -2905,7 +2849,7 @@ packages:
resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==} resolution: {integrity: sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
dev: true dev: true
/is-symbol@1.0.4: /is-symbol@1.0.4:
@@ -2915,13 +2859,6 @@ packages:
has-symbols: 1.0.3 has-symbols: 1.0.3
dev: true dev: true
/is-typed-array@1.1.12:
resolution: {integrity: sha512-Z14TF2JNG8Lss5/HMqt0//T9JeHXttXy5pH/DBU4vi98ozO2btxzq9MwYDZYnKwU8nRsz/+GVFVRDq3DkVuSPg==}
engines: {node: '>= 0.4'}
dependencies:
which-typed-array: 1.1.13
dev: true
/is-typed-array@1.1.13: /is-typed-array@1.1.13:
resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==} resolution: {integrity: sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -3068,10 +3005,10 @@ packages:
resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==} resolution: {integrity: sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ==}
engines: {node: '>=4.0'} engines: {node: '>=4.0'}
dependencies: dependencies:
array-includes: 3.1.7 array-includes: 3.1.8
array.prototype.flat: 1.3.2 array.prototype.flat: 1.3.2
object.assign: 4.1.5 object.assign: 4.1.5
object.values: 1.1.7 object.values: 1.2.0
dev: true dev: true
/keyv@4.5.4: /keyv@4.5.4:
@@ -3476,6 +3413,15 @@ packages:
es-abstract: 1.22.3 es-abstract: 1.22.3
dev: true dev: true
/object.entries@1.1.8:
resolution: {integrity: sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.7
define-properties: 1.2.1
es-object-atoms: 1.0.0
dev: true
/object.fromentries@2.0.7: /object.fromentries@2.0.7:
resolution: {integrity: sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==} resolution: {integrity: sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -3485,6 +3431,16 @@ packages:
es-abstract: 1.22.3 es-abstract: 1.22.3
dev: true dev: true
/object.fromentries@2.0.8:
resolution: {integrity: sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.7
define-properties: 1.2.1
es-abstract: 1.23.3
es-object-atoms: 1.0.0
dev: true
/object.groupby@1.0.1: /object.groupby@1.0.1:
resolution: {integrity: sha512-HqaQtqLnp/8Bn4GL16cj+CUYbnpe1bh0TtEaWvybszDG4tgxCJuRpV8VGuvNaI1fAnI4lUJzDG55MXcOH4JZcQ==} resolution: {integrity: sha512-HqaQtqLnp/8Bn4GL16cj+CUYbnpe1bh0TtEaWvybszDG4tgxCJuRpV8VGuvNaI1fAnI4lUJzDG55MXcOH4JZcQ==}
dependencies: dependencies:
@@ -3494,11 +3450,13 @@ packages:
get-intrinsic: 1.2.2 get-intrinsic: 1.2.2
dev: true dev: true
/object.hasown@1.1.3: /object.hasown@1.1.4:
resolution: {integrity: sha512-fFI4VcYpRHvSLXxP7yiZOMAd331cPfd2p7PFDVbgUsYOfCT3tICVqXWngbjr4m49OvsBwUBQ6O2uQoJvy3RexA==} resolution: {integrity: sha512-FZ9LZt9/RHzGySlBARE3VF+gE26TxR38SdmqOqliuTnl9wrKulaQs+4dee1V+Io8VfxqzAfHu6YuRgUy8OHoTg==}
engines: {node: '>= 0.4'}
dependencies: dependencies:
define-properties: 1.2.1 define-properties: 1.2.1
es-abstract: 1.22.3 es-abstract: 1.23.3
es-object-atoms: 1.0.0
dev: true dev: true
/object.values@1.1.7: /object.values@1.1.7:
@@ -3510,6 +3468,15 @@ packages:
es-abstract: 1.22.3 es-abstract: 1.22.3
dev: true dev: true
/object.values@1.2.0:
resolution: {integrity: sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.7
define-properties: 1.2.1
es-object-atoms: 1.0.0
dev: true
/oidc-token-hash@5.0.3: /oidc-token-hash@5.0.3:
resolution: {integrity: sha512-IF4PcGgzAr6XXSff26Sk/+P4KZFJVuHAJZj3wgO3vX2bMdNVp/QXTP3P7CEm9V1IdG8lDLY3HhiqpsE/nOwpPw==} resolution: {integrity: sha512-IF4PcGgzAr6XXSff26Sk/+P4KZFJVuHAJZj3wgO3vX2bMdNVp/QXTP3P7CEm9V1IdG8lDLY3HhiqpsE/nOwpPw==}
engines: {node: ^10.13.0 || >=12.0.0} engines: {node: ^10.13.0 || >=12.0.0}
@@ -3987,15 +3954,6 @@ packages:
/regenerator-runtime@0.14.1: /regenerator-runtime@0.14.1:
resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==} resolution: {integrity: sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==}
/regexp.prototype.flags@1.5.1:
resolution: {integrity: sha512-sy6TXMN+hnP/wMy+ISxg3krXx7BAtWVO4UouuCN/ziM9UEne0euamVNafDfvC83bRNr95y0V5iijeDQFUNpvrg==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.5
define-properties: 1.2.1
set-function-name: 2.0.1
dev: true
/regexp.prototype.flags@1.5.2: /regexp.prototype.flags@1.5.2:
resolution: {integrity: sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==} resolution: {integrity: sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4003,7 +3961,7 @@ packages:
call-bind: 1.0.7 call-bind: 1.0.7
define-properties: 1.2.1 define-properties: 1.2.1
es-errors: 1.3.0 es-errors: 1.3.0
set-function-name: 2.0.1 set-function-name: 2.0.2
dev: true dev: true
/request@2.88.2: /request@2.88.2:
@@ -4105,8 +4063,8 @@ packages:
resolution: {integrity: sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg==} resolution: {integrity: sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg==}
engines: {node: '>=0.4'} engines: {node: '>=0.4'}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
has-symbols: 1.0.3 has-symbols: 1.0.3
isarray: 2.0.5 isarray: 2.0.5
dev: true dev: true
@@ -4129,15 +4087,6 @@ packages:
resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==}
dev: false dev: false
/safe-regex-test@1.0.2:
resolution: {integrity: sha512-83S9w6eFq12BBIJYvjMux6/dkirb8+4zJRA9cxNBVb7Wq5fJBW+Xze48WqR8pxua7bDuAaaAxtVVd4Idjp1dBQ==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.5
get-intrinsic: 1.2.2
is-regex: 1.1.4
dev: true
/safe-regex-test@1.0.3: /safe-regex-test@1.0.3:
resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==} resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4190,17 +4139,6 @@ packages:
lru-cache: 6.0.0 lru-cache: 6.0.0
dev: true dev: true
/set-function-length@1.2.0:
resolution: {integrity: sha512-4DBHDoyHlM1IRPGYcoxexgh67y4ueR53FKV1yyxwFMY7aCqcN/38M1+SwZ/qJQ8iLv7+ck385ot4CcisOAPT9w==}
engines: {node: '>= 0.4'}
dependencies:
define-data-property: 1.1.1
function-bind: 1.1.2
get-intrinsic: 1.2.2
gopd: 1.0.1
has-property-descriptors: 1.0.1
dev: true
/set-function-length@1.2.2: /set-function-length@1.2.2:
resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==} resolution: {integrity: sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4217,9 +4155,19 @@ packages:
resolution: {integrity: sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA==} resolution: {integrity: sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
define-data-property: 1.1.1 define-data-property: 1.1.4
functions-have-names: 1.2.3 functions-have-names: 1.2.3
has-property-descriptors: 1.0.1 has-property-descriptors: 1.0.2
dev: true
/set-function-name@2.0.2:
resolution: {integrity: sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==}
engines: {node: '>= 0.4'}
dependencies:
define-data-property: 1.1.4
es-errors: 1.3.0
functions-have-names: 1.2.3
has-property-descriptors: 1.0.2
dev: true dev: true
/setprototypeof@1.2.0: /setprototypeof@1.2.0:
@@ -4249,8 +4197,18 @@ packages:
/side-channel@1.0.4: /side-channel@1.0.4:
resolution: {integrity: sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==} resolution: {integrity: sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
get-intrinsic: 1.2.2 get-intrinsic: 1.2.4
object-inspect: 1.13.1
dev: true
/side-channel@1.0.6:
resolution: {integrity: sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.7
es-errors: 1.3.0
get-intrinsic: 1.2.4
object-inspect: 1.13.1 object-inspect: 1.13.1
dev: true dev: true
@@ -4355,27 +4313,22 @@ packages:
strip-ansi: 7.1.0 strip-ansi: 7.1.0
dev: true dev: true
/string.prototype.matchall@4.0.10: /string.prototype.matchall@4.0.11:
resolution: {integrity: sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==} resolution: {integrity: sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==}
dependencies:
call-bind: 1.0.5
define-properties: 1.2.1
es-abstract: 1.22.3
get-intrinsic: 1.2.2
has-symbols: 1.0.3
internal-slot: 1.0.6
regexp.prototype.flags: 1.5.1
set-function-name: 2.0.1
side-channel: 1.0.4
dev: true
/string.prototype.trim@1.2.8:
resolution: {integrity: sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
call-bind: 1.0.5 call-bind: 1.0.7
define-properties: 1.2.1 define-properties: 1.2.1
es-abstract: 1.22.3 es-abstract: 1.23.3
es-errors: 1.3.0
es-object-atoms: 1.0.0
get-intrinsic: 1.2.4
gopd: 1.0.1
has-symbols: 1.0.3
internal-slot: 1.0.7
regexp.prototype.flags: 1.5.2
set-function-name: 2.0.2
side-channel: 1.0.6
dev: true dev: true
/string.prototype.trim@1.2.9: /string.prototype.trim@1.2.9:
@@ -4388,14 +4341,6 @@ packages:
es-object-atoms: 1.0.0 es-object-atoms: 1.0.0
dev: true dev: true
/string.prototype.trimend@1.0.7:
resolution: {integrity: sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA==}
dependencies:
call-bind: 1.0.5
define-properties: 1.2.1
es-abstract: 1.22.3
dev: true
/string.prototype.trimend@1.0.8: /string.prototype.trimend@1.0.8:
resolution: {integrity: sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==} resolution: {integrity: sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==}
dependencies: dependencies:
@@ -4404,14 +4349,6 @@ packages:
es-object-atoms: 1.0.0 es-object-atoms: 1.0.0
dev: true dev: true
/string.prototype.trimstart@1.0.7:
resolution: {integrity: sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg==}
dependencies:
call-bind: 1.0.5
define-properties: 1.2.1
es-abstract: 1.22.3
dev: true
/string.prototype.trimstart@1.0.8: /string.prototype.trimstart@1.0.8:
resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==} resolution: {integrity: sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4728,15 +4665,6 @@ packages:
engines: {node: '>=10'} engines: {node: '>=10'}
dev: true dev: true
/typed-array-buffer@1.0.0:
resolution: {integrity: sha512-Y8KTSIglk9OZEr8zywiIHG/kmQ7KWyjseXs1CbSo8vC42w7hg2HgYTxSWwP0+is7bWDc1H+Fo026CpHFwm8tkw==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.5
get-intrinsic: 1.2.2
is-typed-array: 1.1.12
dev: true
/typed-array-buffer@1.0.2: /typed-array-buffer@1.0.2:
resolution: {integrity: sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==} resolution: {integrity: sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4746,16 +4674,6 @@ packages:
is-typed-array: 1.1.13 is-typed-array: 1.1.13
dev: true dev: true
/typed-array-byte-length@1.0.0:
resolution: {integrity: sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==}
engines: {node: '>= 0.4'}
dependencies:
call-bind: 1.0.5
for-each: 0.3.3
has-proto: 1.0.1
is-typed-array: 1.1.12
dev: true
/typed-array-byte-length@1.0.1: /typed-array-byte-length@1.0.1:
resolution: {integrity: sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==} resolution: {integrity: sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4767,17 +4685,6 @@ packages:
is-typed-array: 1.1.13 is-typed-array: 1.1.13
dev: true dev: true
/typed-array-byte-offset@1.0.0:
resolution: {integrity: sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==}
engines: {node: '>= 0.4'}
dependencies:
available-typed-arrays: 1.0.5
call-bind: 1.0.5
for-each: 0.3.3
has-proto: 1.0.1
is-typed-array: 1.1.12
dev: true
/typed-array-byte-offset@1.0.2: /typed-array-byte-offset@1.0.2:
resolution: {integrity: sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==} resolution: {integrity: sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4790,14 +4697,6 @@ packages:
is-typed-array: 1.1.13 is-typed-array: 1.1.13
dev: true dev: true
/typed-array-length@1.0.4:
resolution: {integrity: sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==}
dependencies:
call-bind: 1.0.5
for-each: 0.3.3
is-typed-array: 1.1.12
dev: true
/typed-array-length@1.0.6: /typed-array-length@1.0.6:
resolution: {integrity: sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==} resolution: {integrity: sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
@@ -4951,7 +4850,7 @@ packages:
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}
dependencies: dependencies:
function.prototype.name: 1.1.6 function.prototype.name: 1.1.6
has-tostringtag: 1.0.0 has-tostringtag: 1.0.2
is-async-function: 2.0.0 is-async-function: 2.0.0
is-date-object: 1.0.5 is-date-object: 1.0.5
is-finalizationregistry: 1.0.2 is-finalizationregistry: 1.0.2
@@ -4961,7 +4860,7 @@ packages:
isarray: 2.0.5 isarray: 2.0.5
which-boxed-primitive: 1.0.2 which-boxed-primitive: 1.0.2
which-collection: 1.0.1 which-collection: 1.0.1
which-typed-array: 1.1.13 which-typed-array: 1.1.15
dev: true dev: true
/which-collection@1.0.1: /which-collection@1.0.1:
@@ -4973,17 +4872,6 @@ packages:
is-weakset: 2.0.2 is-weakset: 2.0.2
dev: true dev: true
/which-typed-array@1.1.13:
resolution: {integrity: sha512-P5Nra0qjSncduVPEAr7xhoF5guty49ArDTwzJ/yNuPIbZppyRxFQsRCWrocxIY+CnMVG+qfbU2FmDKyvSGClow==}
engines: {node: '>= 0.4'}
dependencies:
available-typed-arrays: 1.0.5
call-bind: 1.0.5
for-each: 0.3.3
gopd: 1.0.1
has-tostringtag: 1.0.0
dev: true
/which-typed-array@1.1.15: /which-typed-array@1.1.15:
resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==} resolution: {integrity: sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==}
engines: {node: '>= 0.4'} engines: {node: '>= 0.4'}

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Waarskuwings", "alerts": "Waarskuwings",
"bans": "Verbanne" "bans": "Verbanne"
},
"wgeasy": {
"connected": "Gekoppel",
"enabled": "Geaktiveer",
"disabled": "Onaktief",
"total": "Totaal"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "تنبيهات", "alerts": "تنبيهات",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "مفعل",
"disabled": "معطل",
"total": "المجموع"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Предупреждения", "alerts": "Предупреждения",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Активирано",
"disabled": "Деактивирано",
"total": "Общо"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alertes", "alerts": "Alertes",
"bans": "Prohibicions" "bans": "Prohibicions"
},
"wgeasy": {
"connected": "Connectat",
"enabled": "Activat",
"disabled": "Desactivat",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Upozornění", "alerts": "Upozornění",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Povoleno",
"disabled": "Zakázáno",
"total": "Celkem"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Advarsler", "alerts": "Advarsler",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Aktiveret",
"disabled": "Deaktiveret",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Warnungen", "alerts": "Warnungen",
"bans": "Banns" "bans": "Banns"
},
"wgeasy": {
"connected": "Verbunden",
"enabled": "Aktiviert",
"disabled": "Deaktiviert",
"total": "Gesamt"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "Mit Auth",
"outdated": "Veraltet",
"banned": "Gebannt"
} }
} }

View File

@@ -40,14 +40,14 @@
}, },
"resources": { "resources": {
"cpu": "Επεξεργαστής", "cpu": "Επεξεργαστής",
"mem": "MEM", "mem": "Μνήμη",
"total": "Σύνολο", "total": "Σύνολο",
"free": "Δωρεάν", "free": "Δωρεάν",
"used": "χρησιμοποιημένο", "used": "χρησιμοποιημένο",
"load": "Φόρτωση", "load": "Φόρτωση",
"temp": ΕΡΜΟΚΡΑΣΪΑ", "temp": ερμοκρασία",
"max": "Μέγιστο", "max": "Μέγιστο",
"uptime": "ΠΑΝΩ" "uptime": "Χρόνος Λειτουργίας"
}, },
"unifi": { "unifi": {
"users": "Χρήστες", "users": "Χρήστες",
@@ -61,7 +61,7 @@
"wlan_devices": "WLAN Συσκευές", "wlan_devices": "WLAN Συσκευές",
"lan_users": "LAN Χρήστες", "lan_users": "LAN Χρήστες",
"wlan_users": "WLAN Χρήστες", "wlan_users": "WLAN Χρήστες",
"up": "ΠΑΝΩ", "up": "Χρόνος Λειτουργίας",
"down": "ΚΑΤΩ", "down": "ΚΑΤΩ",
"wait": "Παρακαλώ περιμένετε", "wait": "Παρακαλώ περιμένετε",
"empty_data": "Άγνωστη κατάσταση υποσυστήματος" "empty_data": "Άγνωστη κατάσταση υποσυστήματος"
@@ -69,7 +69,7 @@
"docker": { "docker": {
"rx": "RX", "rx": "RX",
"tx": "TX", "tx": "TX",
"mem": "MEM", "mem": "Μνήμη",
"cpu": "Επεξεργαστής", "cpu": "Επεξεργαστής",
"running": "Τρέχων", "running": "Τρέχων",
"offline": "Εκτός σύνδεσης", "offline": "Εκτός σύνδεσης",
@@ -85,16 +85,16 @@
"ping": { "ping": {
"error": "Σφάλμα", "error": "Σφάλμα",
"ping": "Ping", "ping": "Ping",
"down": "Down", "down": "Ping down",
"up": "Up", "up": "Ping up",
"not_available": "Μη διαθέσιμο" "not_available": "Μη διαθέσιμο"
}, },
"siteMonitor": { "siteMonitor": {
"http_status": "Κατάσταση HTTP", "http_status": "Κατάσταση HTTP",
"error": "Σφάλμα", "error": "Σφάλμα",
"response": "Απόκριση", "response": "Απόκριση",
"down": "Down", "down": "Ping down",
"up": "Up", "up": "Ping up",
"not_available": "Μη διαθέσιμο" "not_available": "Μη διαθέσιμο"
}, },
"emby": { "emby": {
@@ -136,16 +136,16 @@
"connectionStatus": "Κατάσταση", "connectionStatus": "Κατάσταση",
"connectionStatusUnconfigured": "Μη Ρυθμισμένο", "connectionStatusUnconfigured": "Μη Ρυθμισμένο",
"connectionStatusConnecting": "Κατάσταση Σύνδεσης", "connectionStatusConnecting": "Κατάσταση Σύνδεσης",
"connectionStatusAuthenticating": "Authenticating", "connectionStatusAuthenticating": "Ταυτοποίηση",
"connectionStatusPendingDisconnect": "Pending Disconnect", "connectionStatusPendingDisconnect": "Εκκρεμεί Αποσύνδεση",
"connectionStatusDisconnecting": "Disconnecting", "connectionStatusDisconnecting": "Αποσύνδεση",
"connectionStatusDisconnected": "Disconnected", "connectionStatusDisconnected": "Αποσυνδέθηκε",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Συνδέθηκε",
"uptime": "Χρόνος Λειτουργίας", "uptime": "Χρόνος Λειτουργίας",
"maxDown": "Max. Down", "maxDown": "Μέγιστο Download",
"maxUp": "Max. Up", "maxUp": "Μέγιστο Upload",
"down": "Down", "down": "Ping down",
"up": "Up", "up": "Ping up",
"received": "Ληφθέντα", "received": "Ληφθέντα",
"sent": "Απεσταλμένα", "sent": "Απεσταλμένα",
"externalIPAddress": "Εξωτερική IP" "externalIPAddress": "Εξωτερική IP"
@@ -217,7 +217,7 @@
"memUsage": "Χρήση μνήμης", "memUsage": "Χρήση μνήμης",
"systemTempC": "Θερμοκρασία συστήματος", "systemTempC": "Θερμοκρασία συστήματος",
"poolUsage": "Χρήση πισίνας", "poolUsage": "Χρήση πισίνας",
"volumeUsage": "Volume Usage", "volumeUsage": "Χρήση Όγκου",
"invalid": "Μη έγκυρο" "invalid": "Μη έγκυρο"
}, },
"deluge": { "deluge": {
@@ -273,15 +273,15 @@
}, },
"overseerr": { "overseerr": {
"pending": "Σε εκκρεμότητα", "pending": "Σε εκκρεμότητα",
"processing": "Processing", "processing": "Σε επεξεργασία",
"approved": "Εγκρίθηκε", "approved": "Εγκρίθηκε",
"available": "Διαθέσιμο" "available": "Διαθέσιμο"
}, },
"netalertx": { "netalertx": {
"total": "Σύνολο", "total": "Σύνολο",
"connected": "Connected", "connected": "Συνδέθηκε",
"new_devices": "New Devices", "new_devices": "Νέες συσκευές",
"down_alerts": "Down Alerts" "down_alerts": "Ειδοποιήσεις offline"
}, },
"pihole": { "pihole": {
"queries": "Queries", "queries": "Queries",
@@ -309,26 +309,26 @@
"address": "Διεύθυνση", "address": "Διεύθυνση",
"expires": "Λήγει", "expires": "Λήγει",
"never": "Ποτέ", "never": "Ποτέ",
"last_seen": "Last Seen", "last_seen": "Τελευταία Σύνδεση",
"now": "Τώρα", "now": "Τώρα",
"years": "{{number}}y", "years": "{{number}}χρόνια",
"weeks": "{{number}}w", "weeks": "{{number}}εβδομάδες",
"days": "{{number}}d", "days": "{{number}}μέρες",
"hours": "{{number}}h", "hours": "{{number}}ώρες",
"minutes": "{{number}}m", "minutes": "{{number}}λεπτά",
"seconds": "{{number}}s", "seconds": "{{number}}δευτερόλεπτα",
"ago": "{{value}} πρίν" "ago": "{{value}} πρίν"
}, },
"tdarr": { "tdarr": {
"queue": "Ουρά", "queue": "Ουρά",
"processed": "Processed", "processed": "Σε επεξεργασία",
"errored": "Errored", "errored": "Σφάλματα",
"saved": "Saved" "saved": "Αποθηκεύτηκε"
}, },
"traefik": { "traefik": {
"routers": "Routers", "routers": "Δρομολογητές",
"services": "Υπηρεσίες", "services": "Υπηρεσίες",
"middleware": "Middleware" "middleware": "Ενδιάμεσο λογισμικό"
}, },
"navidrome": { "navidrome": {
"nothing_streaming": "Δεν υπάρχουν ενεργές ροές", "nothing_streaming": "Δεν υπάρχουν ενεργές ροές",
@@ -360,7 +360,7 @@
}, },
"jackett": { "jackett": {
"configured": "Ρυθμισμένο", "configured": "Ρυθμισμένο",
"errored": "Errored" "errored": "Σφάλματα"
}, },
"strelaysrv": { "strelaysrv": {
"numActiveSessions": "Συνεδρίες", "numActiveSessions": "Συνεδρίες",
@@ -371,7 +371,7 @@
"mastodon": { "mastodon": {
"user_count": "Χρήστες", "user_count": "Χρήστες",
"status_count": "Δημοσιεύσεις", "status_count": "Δημοσιεύσεις",
"domain_count": "Domains" "domain_count": "Τομείς"
}, },
"medusa": { "medusa": {
"wanted": "Επιθυμούντε", "wanted": "Επιθυμούντε",
@@ -386,7 +386,7 @@
"down": "Εκτός σύνδεσης" "down": "Εκτός σύνδεσης"
}, },
"miniflux": { "miniflux": {
"read": "Read", "read": "Διαβάστηκε",
"unread": "Μη Διαβασμένο" "unread": "Μη Διαβασμένο"
}, },
"authentik": { "authentik": {
@@ -395,7 +395,7 @@
"failedLoginsLast24H": "Αποτυχημένες Συνδέσεις (24h)" "failedLoginsLast24H": "Αποτυχημένες Συνδέσεις (24h)"
}, },
"proxmox": { "proxmox": {
"mem": "MEM", "mem": "Μνήμη",
"cpu": "Επεξεργαστής", "cpu": "Επεξεργαστής",
"lxc": "LXC", "lxc": "LXC",
"vms": "VMs" "vms": "VMs"
@@ -404,17 +404,17 @@
"cpu": "Επεξεργαστής", "cpu": "Επεξεργαστής",
"load": "Φόρτωση", "load": "Φόρτωση",
"wait": "Παρακαλώ περιμένετε", "wait": "Παρακαλώ περιμένετε",
"temp": ΕΡΜΟΚΡΑΣΪΑ", "temp": ερμοκρασία",
"_temp": "Temp", "_temp": "Temp",
"warn": "Warn", "warn": "Warn",
"uptime": "ΠΑΝΩ", "uptime": "Χρόνος Λειτουργίας",
"total": "Σύνολο", "total": "Σύνολο",
"free": "Δωρεάν", "free": "Δωρεάν",
"used": "χρησιμοποιημένο", "used": "χρησιμοποιημένο",
"days": "d", "days": "d",
"hours": "h", "hours": "h",
"crit": "Crit", "crit": "Crit",
"read": "Read", "read": "Διαβάστηκε",
"write": "Write", "write": "Write",
"gpu": "GPU", "gpu": "GPU",
"mem": "Μνήμη", "mem": "Μνήμη",
@@ -450,80 +450,80 @@
"55-night": "Heavy Drizzle", "55-night": "Heavy Drizzle",
"56-day": "Light Freezing Drizzle", "56-day": "Light Freezing Drizzle",
"56-night": "Light Freezing Drizzle", "56-night": "Light Freezing Drizzle",
"57-day": "Freezing Drizzle", "57-day": "Παγωμένο ψιχάλισμα",
"57-night": "Freezing Drizzle", "57-night": "Παγωμένο ψιχάλισμα",
"61-day": "Light Rain", "61-day": "Ψιλόβροχο",
"61-night": "Light Rain", "61-night": "Ψιλόβροχο",
"63-day": "Rain", "63-day": "Βροχή",
"63-night": "Rain", "63-night": "Βροχή",
"65-day": "Heavy Rain", "65-day": "Δυνατή βροχή",
"65-night": "Heavy Rain", "65-night": "Δυνατή βροχή",
"66-day": "Freezing Rain", "66-day": "Παγωμένη βροχή",
"66-night": "Freezing Rain", "66-night": "Παγωμένη βροχή",
"67-day": "Freezing Rain", "67-day": "Παγωμένη βροχή",
"67-night": "Freezing Rain", "67-night": "Παγωμένη βροχή",
"71-day": "Light Snow", "71-day": "Ελαφριά Χιονόπτωση",
"71-night": "Light Snow", "71-night": "Ελαφριά Χιονόπτωση",
"73-day": "Snow", "73-day": "Χιόνι",
"73-night": "Snow", "73-night": "Χιόνι",
"75-day": "Heavy Snow", "75-day": "Ισχυρή χιονόπτωση",
"75-night": "Heavy Snow", "75-night": "Ισχυρή χιονόπτωση",
"77-day": "Snow Grains", "77-day": "Κόκκοι Χιονιού",
"77-night": "Snow Grains", "77-night": "Κόκκοι Χιονιού",
"80-day": "Light Showers", "80-day": "Ασθενείς βροχές",
"80-night": "Light Showers", "80-night": "Ασθενείς βροχές",
"81-day": "Showers", "81-day": "Βροχοπτώσεις",
"81-night": "Showers", "81-night": "Βροχοπτώσεις",
"82-day": "Heavy Showers", "82-day": "Ισχυρές βροχοπτώσεις",
"82-night": "Heavy Showers", "82-night": "Ισχυρές βροχοπτώσεις",
"85-day": "Snow Showers", "85-day": "Χιονοπτώσεις",
"85-night": "Snow Showers", "85-night": "Χιονοπτώσεις",
"86-day": "Snow Showers", "86-day": "Χιονοπτώσεις",
"86-night": "Snow Showers", "86-night": "Χιονοπτώσεις",
"95-day": "Thunderstorm", "95-day": "Καταιγίδα",
"95-night": "Thunderstorm", "95-night": "Καταιγίδα",
"96-day": "Thunderstorm With Hail", "96-day": "Καταιγίδα Με Χαλάζι",
"96-night": "Thunderstorm With Hail", "96-night": "Καταιγίδα Με Χαλάζι",
"99-day": "Thunderstorm With Hail", "99-day": "Καταιγίδα Με Χαλάζι",
"99-night": "Thunderstorm With Hail" "99-night": "Καταιγίδα Με Χαλάζι"
}, },
"homebridge": { "homebridge": {
"available_update": "System", "available_update": "Σύστημα",
"updates": "Updates", "updates": "Ενημερώσεις",
"update_available": "Update Available", "update_available": "Διαθέσιμη ενημέρωση",
"up_to_date": "Up to Date", "up_to_date": "Ενημερωμένο",
"child_bridges": "Child Bridges", "child_bridges": "Child Bridges",
"child_bridges_status": "{{ok}}/{{total}}", "child_bridges_status": "{{ok}}/{{total}}",
"up": "Up", "up": "Ping up",
"pending": "Σε εκκρεμότητα", "pending": "Σε εκκρεμότητα",
"down": "Down" "down": "Ping down"
}, },
"healthchecks": { "healthchecks": {
"new": "New", "new": "New",
"up": "Up", "up": "Ping up",
"grace": "In Grace Period", "grace": "In Grace Period",
"down": "Down", "down": "Ping down",
"paused": "Paused", "paused": "Paused",
"status": "Κατάσταση", "status": "Κατάσταση",
"last_ping": "Last Ping", "last_ping": "Τελευταίο Ping",
"never": "No pings yet" "never": "Δεν υπάρχουν ping ακόμα"
}, },
"watchtower": { "watchtower": {
"containers_scanned": "Scanned", "containers_scanned": "Σκαναρισμένο",
"containers_updated": "Updated", "containers_updated": "Ενημερώθηκε",
"containers_failed": "Failed" "containers_failed": "Απέτυχε"
}, },
"autobrr": { "autobrr": {
"approvedPushes": "Εγκρίθηκε", "approvedPushes": "Εγκρίθηκε",
"rejectedPushes": "Rejected", "rejectedPushes": "Απορρίφθηκε",
"filters": "Filters", "filters": "Φίλτρα",
"indexers": "Ευρετήρια" "indexers": "Ευρετήρια"
}, },
"tubearchivist": { "tubearchivist": {
"downloads": "Ουρά", "downloads": "Ουρά",
"videos": "Videos", "videos": "Βίντεο",
"channels": "Κανάλια", "channels": "Κανάλια",
"playlists": "Playlists" "playlists": "Λίστες αναπαραγωγής"
}, },
"truenas": { "truenas": {
"load": "Φόρτος Συστήματος", "load": "Φόρτος Συστήματος",
@@ -544,18 +544,18 @@
"hdhomerun": { "hdhomerun": {
"channels": "Κανάλια", "channels": "Κανάλια",
"hd": "HD", "hd": "HD",
"tunerCount": "Tuners", "tunerCount": "Δέκτες",
"channelNumber": "Κανάλι", "channelNumber": "Κανάλι",
"channelNetwork": "Δίκτυο", "channelNetwork": "Δίκτυο",
"signalStrength": "Strength", "signalStrength": "Ισχύς σήματος",
"signalQuality": "Ποιότητα", "signalQuality": "Ποιότητα",
"symbolQuality": "Ποιότητα", "symbolQuality": "Ποιότητα",
"networkRate": "Ρυθμός bit", "networkRate": "Ρυθμός bit",
"clientIP": "Client" "clientIP": "Πελάτης"
}, },
"scrutiny": { "scrutiny": {
"passed": "Passed", "passed": "Passed",
"failed": "Failed", "failed": "Απέτυχε",
"unknown": "Άγνωστο" "unknown": "Άγνωστο"
}, },
"paperlessngx": { "paperlessngx": {
@@ -617,8 +617,8 @@
"load": "Load Avg", "load": "Load Avg",
"memory": "Mem Usage", "memory": "Mem Usage",
"wanStatus": "WAN Status", "wanStatus": "WAN Status",
"up": "Up", "up": "Ping up",
"down": "Down", "down": "Ping down",
"temp": "Temp", "temp": "Temp",
"disk": "Χρήση δίσκου", "disk": "Χρήση δίσκου",
"wanIP": "WAN IP" "wanIP": "WAN IP"
@@ -632,14 +632,14 @@
"immich": { "immich": {
"users": "Χρήστες", "users": "Χρήστες",
"photos": "Φωτογραφίες", "photos": "Φωτογραφίες",
"videos": "Videos", "videos": "Βίντεο",
"storage": "Storage" "storage": "Αποθηκευτικός χώρος"
}, },
"uptimekuma": { "uptimekuma": {
"up": "Sites Up", "up": "Online τοποθεσίες",
"down": "Sites Down", "down": "Offline τοποθεσίες",
"uptime": "Χρόνος Λειτουργίας", "uptime": "Χρόνος Λειτουργίας",
"incident": "Incident", "incident": "Περιστατικό",
"m": "m" "m": "m"
}, },
"atsumeru": { "atsumeru": {
@@ -666,100 +666,100 @@
"photoprism": { "photoprism": {
"albums": "Άλμπουμ", "albums": "Άλμπουμ",
"photos": "Φωτογραφίες", "photos": "Φωτογραφίες",
"videos": "Videos", "videos": "Βίντεο",
"people": "People" "people": "Άνθρωποι"
}, },
"fileflows": { "fileflows": {
"queue": "Ουρά", "queue": "Ουρά",
"processing": "Processing", "processing": "Σε επεξεργασία",
"processed": "Processed", "processed": "Σε επεξεργασία",
"time": "Time" "time": "Ώρα"
}, },
"grafana": { "grafana": {
"dashboards": "Dashboards", "dashboards": "Πίνακας Ελέγχου",
"datasources": "Data Sources", "datasources": "Πηγές Δεδομένων",
"totalalerts": "Total Alerts", "totalalerts": "Σύνολο Ειδοποιήσεων",
"alertstriggered": "Alerts Triggered" "alertstriggered": "Ενεργοποιημένες Ειδοποιήσεις"
}, },
"nextcloud": { "nextcloud": {
"cpuload": "Cpu Load", "cpuload": "Φόρτος CPU",
"memoryusage": "Memory Usage", "memoryusage": "Χρήση Mνήμης",
"freespace": "Free Space", "freespace": "Ελεύθερος χώρος",
"activeusers": "Active Users", "activeusers": "Ενεργοί χρήστες",
"numfiles": "Files", "numfiles": "Αρχεία",
"numshares": "Shared Items" "numshares": "Κοινόχρηστα στοιχεία"
}, },
"kopia": { "kopia": {
"status": "Κατάσταση", "status": "Κατάσταση",
"size": "Size", "size": "Μέγεθος",
"lastrun": "Last Run", "lastrun": "Τελευταία εκτέλεση",
"nextrun": "Next Run", "nextrun": "Επόμενη εκτέλεση",
"failed": "Failed" "failed": "Απέτυχε"
}, },
"unmanic": { "unmanic": {
"active_workers": "Active Workers", "active_workers": "Ενεργοί χρήστες",
"total_workers": "Total Workers", "total_workers": "Total Workers",
"records_total": "Queue Length" "records_total": "Μήκος Ουράς"
}, },
"pterodactyl": { "pterodactyl": {
"servers": "Διακομιστές", "servers": "Διακομιστές",
"nodes": "Nodes" "nodes": "Κόμβοι [Nodes]"
}, },
"prometheus": { "prometheus": {
"targets_up": "Targets Up", "targets_up": "Στόχοι Πάνω",
"targets_down": "Targets Down", "targets_down": "Στόχοι Κάτω",
"targets_total": "Total Targets" "targets_total": "Συνολικοί Στόχοι"
}, },
"gatus": { "gatus": {
"up": "Sites Up", "up": "Online τοποθεσίες",
"down": "Sites Down", "down": "Offline τοποθεσίες",
"uptime": "Χρόνος Λειτουργίας" "uptime": "Χρόνος Λειτουργίας"
}, },
"ghostfolio": { "ghostfolio": {
"gross_percent_today": "Σήμερα", "gross_percent_today": "Σήμερα",
"gross_percent_1y": "One year", "gross_percent_1y": "Ένας χρόνος",
"gross_percent_max": "All time" "gross_percent_max": "Διαχρονικά"
}, },
"audiobookshelf": { "audiobookshelf": {
"podcasts": "Podcasts", "podcasts": "Podcasts",
"books": "Βιβλία", "books": "Βιβλία",
"podcastsDuration": "Duration", "podcastsDuration": "Διάρκεια",
"booksDuration": "Duration" "booksDuration": "Διάρκεια"
}, },
"homeassistant": { "homeassistant": {
"people_home": "People Home", "people_home": "Σύνολο ανθρώπων στο σπίτι",
"lights_on": "Lights On", "lights_on": "Αναμμένα φώτα",
"switches_on": "Switches On" "switches_on": "Ανοιχτοί διακόπτες"
}, },
"whatsupdocker": { "whatsupdocker": {
"monitoring": "Monitoring", "monitoring": "Παρακολούθηση",
"updates": "Updates" "updates": "Ενημερώσεις"
}, },
"calibreweb": { "calibreweb": {
"books": "Βιβλία", "books": "Βιβλία",
"authors": "Authors", "authors": "Συντάκτες",
"categories": "Κατηγορίες", "categories": "Κατηγορίες",
"series": "Σειρές" "series": "Σειρές"
}, },
"jdownloader": { "jdownloader": {
"downloadCount": "Ουρά", "downloadCount": "Ουρά",
"downloadBytesRemaining": "Υπόλοιπο", "downloadBytesRemaining": "Υπόλοιπο",
"downloadTotalBytes": "Size", "downloadTotalBytes": "Μέγεθος",
"downloadSpeed": "Ταχύτητα" "downloadSpeed": "Ταχύτητα"
}, },
"kavita": { "kavita": {
"seriesCount": "Σειρές", "seriesCount": "Σειρές",
"totalFiles": "Files" "totalFiles": "Αρχεία"
}, },
"azuredevops": { "azuredevops": {
"result": "Result", "result": "Αποτέλεσμα",
"status": "Κατάσταση", "status": "Κατάσταση",
"buildId": "Build ID", "buildId": "Build ID",
"succeeded": "Succeeded", "succeeded": "Πέτυχε",
"notStarted": "Not Started", "notStarted": "Δεν ξεκίνησε",
"failed": "Failed", "failed": "Απέτυχε",
"canceled": "Canceled", "canceled": "Ακυρώθηκε",
"inProgress": "In Progress", "inProgress": "Σε εξέλιξη",
"totalPrs": "Total PRs", "totalPrs": "Total PRs",
"myPrs": "My PRs", "myPrs": "My PRs",
"approved": "Εγκρίθηκε" "approved": "Εγκρίθηκε"
@@ -768,8 +768,8 @@
"status": "Κατάσταση", "status": "Κατάσταση",
"online": "Συνδεδεμένοι", "online": "Συνδεδεμένοι",
"offline": "Εκτός σύνδεσης", "offline": "Εκτός σύνδεσης",
"name": "Name", "name": "Όνομα",
"map": "Map", "map": "Χάρτης",
"currentPlayers": "Current players", "currentPlayers": "Current players",
"players": "Παίκτες", "players": "Παίκτες",
"maxPlayers": "Max players", "maxPlayers": "Max players",
@@ -777,30 +777,30 @@
"ping": "Ping" "ping": "Ping"
}, },
"urbackup": { "urbackup": {
"ok": "Ok", "ok": "Οκ",
"errored": "Errors", "errored": "Σφάλματα",
"noRecent": "Out of Date", "noRecent": "Απαρχαιωμένη έκδοση",
"totalUsed": "Used Storage" "totalUsed": "Χώρος αποθήκευσης σε χρήση"
}, },
"mealie": { "mealie": {
"recipes": "Recipes", "recipes": "Συνταγές",
"users": "Χρήστες", "users": "Χρήστες",
"categories": "Κατηγορίες", "categories": "Κατηγορίες",
"tags": "Tags" "tags": "Ετικέτες"
}, },
"openmediavault": { "openmediavault": {
"downloading": "Downloading", "downloading": "Γίνεται λήψη",
"total": "Σύνολο", "total": "Σύνολο",
"running": "Τρέχων", "running": "Τρέχων",
"stopped": "Σταματημένο", "stopped": "Σταματημένο",
"passed": "Passed", "passed": "Passed",
"failed": "Failed" "failed": "Απέτυχε"
}, },
"openwrt": { "openwrt": {
"uptime": "Χρόνος Λειτουργίας", "uptime": "Χρόνος Λειτουργίας",
"cpuLoad": "CPU Load Avg (5m)", "cpuLoad": "CPU Load Avg (5m)",
"up": "Up", "up": "Ping up",
"down": "Down", "down": "Ping down",
"bytesTx": "Transmitted", "bytesTx": "Transmitted",
"bytesRx": "Ληφθέντα" "bytesRx": "Ληφθέντα"
}, },
@@ -809,13 +809,13 @@
"uptime": "Χρόνος Λειτουργίας", "uptime": "Χρόνος Λειτουργίας",
"lastDown": "Last Downtime", "lastDown": "Last Downtime",
"downDuration": "Downtime Duration", "downDuration": "Downtime Duration",
"sitesUp": "Sites Up", "sitesUp": "Online τοποθεσίες",
"sitesDown": "Sites Down", "sitesDown": "Offline τοποθεσίες",
"paused": "Paused", "paused": "Paused",
"notyetchecked": "Not Yet Checked", "notyetchecked": "Not Yet Checked",
"up": "Up", "up": "Ping up",
"seemsdown": "Seems Down", "seemsdown": "Seems Down",
"down": "Down", "down": "Ping down",
"unknown": "Άγνωστο" "unknown": "Άγνωστο"
}, },
"calendar": { "calendar": {
@@ -857,12 +857,12 @@
"performers": "Performers", "performers": "Performers",
"studios": "Studios", "studios": "Studios",
"movies": "Ταινίες", "movies": "Ταινίες",
"tags": "Tags", "tags": "Ετικέτες",
"oCount": "O Count" "oCount": "O Count"
}, },
"tandoor": { "tandoor": {
"users": "Χρήστες", "users": "Χρήστες",
"recipes": "Recipes", "recipes": "Συνταγές",
"keywords": "Keywords" "keywords": "Keywords"
}, },
"homebox": { "homebox": {
@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Ειδοποιήσεις", "alerts": "Ειδοποιήσεις",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Συνδέθηκε",
"enabled": "Ενεργοποιημένο",
"disabled": "Απενεργοποιημένο",
"total": "Σύνολο"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Totalo"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Desconexión pendiente", "connectionStatusPendingDisconnect": "Desconexión pendiente",
"connectionStatusDisconnecting": "Desconectando", "connectionStatusDisconnecting": "Desconectando",
"connectionStatusDisconnected": "Desconectado", "connectionStatusDisconnected": "Desconectado",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Conectado",
"uptime": "Tiempo activo", "uptime": "Tiempo activo",
"maxDown": "Descarga máxima", "maxDown": "Descarga máxima",
"maxUp": "Subida máxima", "maxUp": "Subida máxima",
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "Total", "total": "Total",
"connected": "Connected", "connected": "Conectado",
"new_devices": "New Devices", "new_devices": "Nuevos dispositivos",
"down_alerts": "Down Alerts" "down_alerts": "Alertas de caída"
}, },
"pihole": { "pihole": {
"queries": "Consultas", "queries": "Consultas",
@@ -544,7 +544,7 @@
"hdhomerun": { "hdhomerun": {
"channels": "Canales", "channels": "Canales",
"hd": "Alta definición", "hd": "Alta definición",
"tunerCount": "Tuners", "tunerCount": "Sintonizadores",
"channelNumber": "Canal", "channelNumber": "Canal",
"channelNetwork": "Red", "channelNetwork": "Red",
"signalStrength": "Intensidad", "signalStrength": "Intensidad",
@@ -827,7 +827,7 @@
}, },
"romm": { "romm": {
"platforms": "Plataformas", "platforms": "Plataformas",
"totalRoms": "Total ROMs" "totalRoms": "ROMs totales"
}, },
"netdata": { "netdata": {
"warnings": "Advertencias", "warnings": "Advertencias",
@@ -835,38 +835,38 @@
}, },
"plantit": { "plantit": {
"events": "Eventos", "events": "Eventos",
"plants": "Plants", "plants": "Plantas",
"photos": "Fotos", "photos": "Fotos",
"species": "Species" "species": "Especies"
}, },
"gitea": { "gitea": {
"notifications": "Notificaciones", "notifications": "Notificaciones",
"issues": "Números", "issues": "Números",
"pulls": "Pull Requests" "pulls": "Solicitudes de cambios"
}, },
"stash": { "stash": {
"scenes": "Scenes", "scenes": "Escenas",
"scenesPlayed": "Scenes Played", "scenesPlayed": "Escenas reproducidas",
"playCount": "Total Plays", "playCount": "Reproducciones totales",
"playDuration": "Time Watched", "playDuration": "Tiempo visto",
"sceneSize": "Scenes Size", "sceneSize": "Tamaño de las escenas",
"sceneDuration": "Scenes Duration", "sceneDuration": "Duración de las escenas",
"images": "Imágenes", "images": "Imágenes",
"imageSize": "Tamaño de imagen", "imageSize": "Tamaño de imagen",
"galleries": "Galerías", "galleries": "Galerías",
"performers": "Performers", "performers": "Intérpretes",
"studios": "Studios", "studios": "Estudios",
"movies": "Películas", "movies": "Películas",
"tags": "Etiquetas", "tags": "Etiquetas",
"oCount": "O Count" "oCount": "O cuenta"
}, },
"tandoor": { "tandoor": {
"users": "Usuarios", "users": "Usuarios",
"recipes": "Recetas", "recipes": "Recetas",
"keywords": "Keywords" "keywords": "Palabras clave"
}, },
"homebox": { "homebox": {
"items": "Items", "items": "Objetos",
"totalWithWarranty": "Con Garantía", "totalWithWarranty": "Con Garantía",
"locations": "Ubicaciones", "locations": "Ubicaciones",
"labels": "Etiquetas", "labels": "Etiquetas",
@@ -875,6 +875,18 @@
}, },
"crowdsec": { "crowdsec": {
"alerts": "Alertas", "alerts": "Alertas",
"bans": "Bans" "bans": "Baneos"
},
"wgeasy": {
"connected": "Conectado",
"enabled": "Activado",
"disabled": "Desactivado",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Guztira"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Käytössä",
"disabled": "Poissa käytöstä",
"total": "Yhteensä"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alertes", "alerts": "Alertes",
"bans": "Exclusions" "bans": "Exclusions"
},
"wgeasy": {
"connected": "Connecté",
"enabled": "Activé",
"disabled": "Désactivé",
"total": "Total"
},
"swagdashboard": {
"proxied": "Par proxy",
"auth": "Avec authentification",
"outdated": "Obsolète",
"banned": "Banni"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "מופעל",
"disabled": "מבוטל",
"total": "סה\"כ"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Odspajanje u tijeku", "connectionStatusPendingDisconnect": "Odspajanje u tijeku",
"connectionStatusDisconnecting": "Odspajanje", "connectionStatusDisconnecting": "Odspajanje",
"connectionStatusDisconnected": "Odspojeno", "connectionStatusDisconnected": "Odspojeno",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Povezano",
"uptime": "Vrijeme rada", "uptime": "Vrijeme rada",
"maxDown": "Maksimum preuzimanja", "maxDown": "Maksimum preuzimanja",
"maxUp": "Maksimum prijenosa", "maxUp": "Maksimum prijenosa",
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "Ukupno", "total": "Ukupno",
"connected": "Connected", "connected": "Povezano",
"new_devices": "New Devices", "new_devices": "Novi uređaji",
"down_alerts": "Down Alerts" "down_alerts": "Obavijesti o nedostupnosti"
}, },
"pihole": { "pihole": {
"queries": "Upiti", "queries": "Upiti",
@@ -834,47 +834,59 @@
"criticals": "Kritično" "criticals": "Kritično"
}, },
"plantit": { "plantit": {
"events": "Events", "events": "Događaji",
"plants": "Plants", "plants": "Biljke",
"photos": "Fotografije", "photos": "Fotografije",
"species": "Species" "species": "Vrste"
}, },
"gitea": { "gitea": {
"notifications": "Notifications", "notifications": "Obavijesti",
"issues": "Problemi", "issues": "Problemi",
"pulls": "Pull Requests" "pulls": "Zahtjevi za povlačenje"
}, },
"stash": { "stash": {
"scenes": "Scenes", "scenes": "Scene",
"scenesPlayed": "Scenes Played", "scenesPlayed": "Reproducirane scene",
"playCount": "Total Plays", "playCount": "Ukupni broj reprodukcija",
"playDuration": "Time Watched", "playDuration": "Vrijeme gledanja",
"sceneSize": "Scenes Size", "sceneSize": "Veličina scene",
"sceneDuration": "Scenes Duration", "sceneDuration": "Trajanje scene",
"images": "Images", "images": "Slike",
"imageSize": "Images Size", "imageSize": "Veličina slike",
"galleries": "Galleries", "galleries": "Galerije",
"performers": "Performers", "performers": "Glumci",
"studios": "Studios", "studios": "Studiji",
"movies": "Filmovi", "movies": "Filmovi",
"tags": "Oznake", "tags": "Oznake",
"oCount": "O Count" "oCount": "O zbroj"
}, },
"tandoor": { "tandoor": {
"users": "Korisnici", "users": "Korisnici",
"recipes": "Recepti", "recipes": "Recepti",
"keywords": "Keywords" "keywords": "Ključne riječi"
}, },
"homebox": { "homebox": {
"items": "Items", "items": "Stavke",
"totalWithWarranty": "With Warranty", "totalWithWarranty": "S garancijom",
"locations": "Locations", "locations": "Lokacije",
"labels": "Labels", "labels": "Oznake",
"users": "Korisnici", "users": "Korisnici",
"totalValue": "Total Value" "totalValue": "Svukupno"
}, },
"crowdsec": { "crowdsec": {
"alerts": "Upozorenja", "alerts": "Upozorenja",
"bans": "Bans" "bans": "Zabrane"
},
"wgeasy": {
"connected": "Povezano",
"enabled": "Aktivirano",
"disabled": "Deaktivirano",
"total": "Ukupno"
},
"swagdashboard": {
"proxied": "Posredovano",
"auth": "S autentifikacijom",
"outdated": "Zastarjelo",
"banned": "Zabranjen pristup"
} }
} }

View File

@@ -45,7 +45,7 @@
"free": "Szabad", "free": "Szabad",
"used": "Használt", "used": "Használt",
"load": "Terhelés", "load": "Terhelés",
"temp": "HŐ", "temp": "HŐM",
"max": "Max", "max": "Max",
"uptime": "FUT" "uptime": "FUT"
}, },
@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Szétkapcsolás függőben", "connectionStatusPendingDisconnect": "Szétkapcsolás függőben",
"connectionStatusDisconnecting": "Kapcsolat bontása", "connectionStatusDisconnecting": "Kapcsolat bontása",
"connectionStatusDisconnected": "Kapcsolat bontva", "connectionStatusDisconnected": "Kapcsolat bontva",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Csatlakozva",
"uptime": "Üzemidő", "uptime": "Üzemidő",
"maxDown": "Max let.", "maxDown": "Max let.",
"maxUp": "Max felt.", "maxUp": "Max felt.",
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "Összes", "total": "Összes",
"connected": "Connected", "connected": "Csatlakozva",
"new_devices": "New Devices", "new_devices": "Új eszközök",
"down_alerts": "Down Alerts" "down_alerts": "Leállási riasztások"
}, },
"pihole": { "pihole": {
"queries": "Lekérdezések", "queries": "Lekérdezések",
@@ -404,7 +404,7 @@
"cpu": "Processzor", "cpu": "Processzor",
"load": "Terhelés", "load": "Terhelés",
"wait": "Kérjük várjon", "wait": "Kérjük várjon",
"temp": "HŐ", "temp": "HŐM",
"_temp": "Hőmérséklet", "_temp": "Hőmérséklet",
"warn": "Figyelmeztet", "warn": "Figyelmeztet",
"uptime": "FUT", "uptime": "FUT",
@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Riasztások", "alerts": "Riasztások",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Csatlakozva",
"enabled": "Bekapcsolva",
"disabled": "Kikapcsolva",
"total": "Összes"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Peringatan", "alerts": "Peringatan",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Aktif",
"disabled": "Nonaktif",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Allarmi", "alerts": "Allarmi",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connesso",
"enabled": "Abilitato",
"disabled": "Disabilitati",
"total": "Totale"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -47,7 +47,7 @@
"load": "ロード", "load": "ロード",
"temp": "温度", "temp": "温度",
"max": "最大", "max": "最大",
"uptime": "上へ" "uptime": "UP"
}, },
"unifi": { "unifi": {
"users": "ユーザ", "users": "ユーザ",
@@ -61,7 +61,7 @@
"wlan_devices": "WLAN デバイス", "wlan_devices": "WLAN デバイス",
"lan_users": "LAN ユーザ", "lan_users": "LAN ユーザ",
"wlan_users": "WLAN ユーザ", "wlan_users": "WLAN ユーザ",
"up": "上へ", "up": "UP",
"down": "下へ", "down": "下へ",
"wait": "お待ちください", "wait": "お待ちください",
"empty_data": "サブシステムの状態は不明" "empty_data": "サブシステムの状態は不明"
@@ -86,7 +86,7 @@
"error": "エラー", "error": "エラー",
"ping": "Ping", "ping": "Ping",
"down": "下へ", "down": "下へ",
"up": "上へ", "up": "稼働",
"not_available": "利用できません。" "not_available": "利用できません。"
}, },
"siteMonitor": { "siteMonitor": {
@@ -94,7 +94,7 @@
"error": "エラー", "error": "エラー",
"response": "応答", "response": "応答",
"down": "下へ", "down": "下へ",
"up": "上へ", "up": "稼働",
"not_available": "利用できません。" "not_available": "利用できません。"
}, },
"emby": { "emby": {
@@ -140,12 +140,12 @@
"connectionStatusPendingDisconnect": "接続を切断する", "connectionStatusPendingDisconnect": "接続を切断する",
"connectionStatusDisconnecting": "接続を切断中", "connectionStatusDisconnecting": "接続を切断中",
"connectionStatusDisconnected": "切断されました", "connectionStatusDisconnected": "切断されました",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "接続済",
"uptime": "稼働時間", "uptime": "稼働時間",
"maxDown": "最大ダウン", "maxDown": "最大ダウン",
"maxUp": "最大アップ", "maxUp": "最大アップ",
"down": "下へ", "down": "下へ",
"up": "上へ", "up": "稼働",
"received": "受信済み", "received": "受信済み",
"sent": "送信済み", "sent": "送信済み",
"externalIPAddress": "退出ID" "externalIPAddress": "退出ID"
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "合計", "total": "合計",
"connected": "Connected", "connected": "接続済",
"new_devices": "New Devices", "new_devices": "新規デバイス",
"down_alerts": "Down Alerts" "down_alerts": "ダウンアラート"
}, },
"pihole": { "pihole": {
"queries": "クエリ", "queries": "クエリ",
@@ -407,7 +407,7 @@
"temp": "温度", "temp": "温度",
"_temp": "温度", "_temp": "温度",
"warn": "警告", "warn": "警告",
"uptime": "上へ", "uptime": "UP",
"total": "合計", "total": "合計",
"free": "空き", "free": "空き",
"used": "使用", "used": "使用",
@@ -494,13 +494,13 @@
"up_to_date": "最新", "up_to_date": "最新",
"child_bridges": "子ブリッジ", "child_bridges": "子ブリッジ",
"child_bridges_status": "{{ok}}/{{total}}", "child_bridges_status": "{{ok}}/{{total}}",
"up": "上へ", "up": "稼働",
"pending": "保留中", "pending": "保留中",
"down": "下へ" "down": "下へ"
}, },
"healthchecks": { "healthchecks": {
"new": "新着", "new": "新着",
"up": "上へ", "up": "稼働",
"grace": "猶予期間中", "grace": "猶予期間中",
"down": "下へ", "down": "下へ",
"paused": "一時停止中", "paused": "一時停止中",
@@ -617,7 +617,7 @@
"load": "読み込み平均", "load": "読み込み平均",
"memory": "メモリ使用量", "memory": "メモリ使用量",
"wanStatus": "WANステータス", "wanStatus": "WANステータス",
"up": "上へ", "up": "稼働",
"down": "下へ", "down": "下へ",
"temp": "温度", "temp": "温度",
"disk": "ディスク使用量", "disk": "ディスク使用量",
@@ -799,7 +799,7 @@
"openwrt": { "openwrt": {
"uptime": "稼働時間", "uptime": "稼働時間",
"cpuLoad": "CPU 平均負荷5 分)", "cpuLoad": "CPU 平均負荷5 分)",
"up": "上へ", "up": "稼働",
"down": "下へ", "down": "下へ",
"bytesTx": "送信済み", "bytesTx": "送信済み",
"bytesRx": "受信済み" "bytesRx": "受信済み"
@@ -813,7 +813,7 @@
"sitesDown": "サイトDown", "sitesDown": "サイトDown",
"paused": "一時停止中", "paused": "一時停止中",
"notyetchecked": "チェックされていません", "notyetchecked": "チェックされていません",
"up": "上へ", "up": "稼働",
"seemsdown": "ダウンしているようです", "seemsdown": "ダウンしているようです",
"down": "下へ", "down": "下へ",
"unknown": "不明" "unknown": "不明"
@@ -875,6 +875,18 @@
}, },
"crowdsec": { "crowdsec": {
"alerts": "アラート", "alerts": "アラート",
"bans": "Bans" "bans": "禁止"
},
"wgeasy": {
"connected": "接続済",
"enabled": "有効",
"disabled": "無効",
"total": "合計"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "경고", "alerts": "경고",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "활성",
"disabled": "비활성",
"total": "총합"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Paziņojumi", "alerts": "Paziņojumi",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Kopā"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Didayakan",
"disabled": "Dinyahdayakan",
"total": "Jumlah"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Meldingen", "alerts": "Meldingen",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Verbonden",
"enabled": "Ingeschakeld",
"disabled": "Uitgeschakeld",
"total": "Totaal"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Varsler", "alerts": "Varsler",
"bans": "Utestengelse" "bans": "Utestengelse"
},
"wgeasy": {
"connected": "Tilkoblet",
"enabled": "Aktivert",
"disabled": "Deaktivert",
"total": "Totalt"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alarmy", "alerts": "Alarmy",
"bans": "Bany" "bans": "Bany"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Włączone",
"disabled": "Wyłączone",
"total": "Całkowite"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alertas", "alerts": "Alertas",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Ativo",
"disabled": "Desabilitado",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -143,7 +143,7 @@
"connectionStatusConnected": "Conectado", "connectionStatusConnected": "Conectado",
"uptime": "Ligado", "uptime": "Ligado",
"maxDown": "Max. Down", "maxDown": "Max. Down",
"maxUp": "Max. Up", "maxUp": "Máx. Acima",
"down": "Inativo", "down": "Inativo",
"up": "Ativo", "up": "Ativo",
"received": "Recebido", "received": "Recebido",
@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alertas", "alerts": "Alertas",
"bans": "Banimentos" "bans": "Banimentos"
},
"wgeasy": {
"connected": "Conectado",
"enabled": "Ativo",
"disabled": "Desabilitado",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Activat",
"disabled": "Dezactivat",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Ожидает отключения", "connectionStatusPendingDisconnect": "Ожидает отключения",
"connectionStatusDisconnecting": "Отключение", "connectionStatusDisconnecting": "Отключение",
"connectionStatusDisconnected": "Отключено", "connectionStatusDisconnected": "Отключено",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Подключено",
"uptime": "Время работы", "uptime": "Время работы",
"maxDown": "Макс. Загрузка", "maxDown": "Макс. Загрузка",
"maxUp": "Макс. Отдача", "maxUp": "Макс. Отдача",
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "Всего", "total": "Всего",
"connected": "Connected", "connected": "Подключено",
"new_devices": "New Devices", "new_devices": "Новое устройство",
"down_alerts": "Down Alerts" "down_alerts": "Оповещение о недоступности"
}, },
"pihole": { "pihole": {
"queries": "Запросы", "queries": "Запросы",
@@ -875,6 +875,18 @@
}, },
"crowdsec": { "crowdsec": {
"alerts": "Предупреждения", "alerts": "Предупреждения",
"bans": "Bans" "bans": "Запреты"
},
"wgeasy": {
"connected": "Подключено",
"enabled": "Включено",
"disabled": "Выключено",
"total": "Всего"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Čakám na odpojenie", "connectionStatusPendingDisconnect": "Čakám na odpojenie",
"connectionStatusDisconnecting": "Odpájanie", "connectionStatusDisconnecting": "Odpájanie",
"connectionStatusDisconnected": "Odpojené", "connectionStatusDisconnected": "Odpojené",
"connectionStatusConnected": "Connected", "connectionStatusConnected": "Pripojené",
"uptime": "Prevádzka", "uptime": "Prevádzka",
"maxDown": "Max. sťahovanie", "maxDown": "Max. sťahovanie",
"maxUp": "Max. nahrávanie", "maxUp": "Max. nahrávanie",
@@ -279,9 +279,9 @@
}, },
"netalertx": { "netalertx": {
"total": "Celkovo", "total": "Celkovo",
"connected": "Connected", "connected": "Pripojené",
"new_devices": "New Devices", "new_devices": "Nové zariadenia",
"down_alerts": "Down Alerts" "down_alerts": "Upozornenia o výpadkoch"
}, },
"pihole": { "pihole": {
"queries": "Dopyty", "queries": "Dopyty",
@@ -846,7 +846,7 @@
}, },
"stash": { "stash": {
"scenes": "Scény", "scenes": "Scény",
"scenesPlayed": "Scenes Played", "scenesPlayed": "Prehrané scény",
"playCount": "Celkovo prehraní", "playCount": "Celkovo prehraní",
"playDuration": "Pozeraný čas", "playDuration": "Pozeraný čas",
"sceneSize": "Veľkosť obrazovky", "sceneSize": "Veľkosť obrazovky",
@@ -869,12 +869,24 @@
"items": "Položky", "items": "Položky",
"totalWithWarranty": "So zárukou", "totalWithWarranty": "So zárukou",
"locations": "Umiestnenia", "locations": "Umiestnenia",
"labels": "Labels", "labels": "Štítky",
"users": "Používatelia", "users": "Používatelia",
"totalValue": "Total Value" "totalValue": "Celková hodnota"
}, },
"crowdsec": { "crowdsec": {
"alerts": "Upozornenia", "alerts": "Upozornenia",
"bans": "Bans" "bans": "Bany"
},
"wgeasy": {
"connected": "Pripojené",
"enabled": "Povolené",
"disabled": "Zakázané",
"total": "Celkovo"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Zastarané",
"banned": "Zabanovaný"
} }
} }

View File

@@ -153,7 +153,7 @@
"caddy": { "caddy": {
"upstreams": "Pretok gor", "upstreams": "Pretok gor",
"requests": "Trenutnih zahtev", "requests": "Trenutnih zahtev",
"requests_failed": "Neuspešnih zahtev" "requests_failed": "Neuspeš. zahtev"
}, },
"changedetectionio": { "changedetectionio": {
"totalObserved": "Skupaj opazovano", "totalObserved": "Skupaj opazovano",
@@ -355,8 +355,8 @@
"enableIndexers": "Indekserji", "enableIndexers": "Indekserji",
"numberOfGrabs": "Zajemi", "numberOfGrabs": "Zajemi",
"numberOfQueries": "Poizvedbe", "numberOfQueries": "Poizvedbe",
"numberOfFailGrabs": "Neuspešni zajemi", "numberOfFailGrabs": "Neuspeš. zajem",
"numberOfFailQueries": "Neuspešne poizvedbe" "numberOfFailQueries": "Neuspeš. poizvedb"
}, },
"jackett": { "jackett": {
"configured": "Nastavljeno", "configured": "Nastavljeno",
@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Opozorila", "alerts": "Opozorila",
"bans": "Prepovedi" "bans": "Prepovedi"
},
"wgeasy": {
"connected": "Povezan",
"enabled": "Omogočen",
"disabled": "Onemogočen",
"total": "Skupaj"
},
"swagdashboard": {
"proxied": "Čez proxi",
"auth": "Z Auth",
"outdated": "Zastarelo",
"banned": "Prepovedan"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Aktiverad",
"disabled": "Inaktiverad",
"total": "Total"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "ప్రారంభించబడింది",
"disabled": "డిసేబ్లెడ్",
"total": "మొత్తం"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "ทั้งหมด"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alarmlar", "alerts": "Alarmlar",
"bans": "Yasaklar" "bans": "Yasaklar"
},
"wgeasy": {
"connected": "Bağlandı",
"enabled": "Etkin",
"disabled": "Devre Dışı",
"total": "Toplam"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Оповіщення", "alerts": "Оповіщення",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Увімкнено",
"disabled": "Вимкнено",
"total": "Усього"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "Alerts", "alerts": "Alerts",
"bans": "Bans" "bans": "Bans"
},
"wgeasy": {
"connected": "Connected",
"enabled": "Enabled",
"disabled": "Disabled",
"total": "Tổng"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "警示", "alerts": "警示",
"bans": "禁止" "bans": "禁止"
},
"wgeasy": {
"connected": "Connected",
"enabled": "啟用",
"disabled": "停用咗",
"total": "全部"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -54,10 +54,10 @@
"uptime": "运行时间", "uptime": "运行时间",
"days": "天", "days": "天",
"wan": "WAN", "wan": "WAN",
"lan": "局域网", "lan": "LAN",
"wlan": "无线局域网", "wlan": "WLAN",
"devices": "设备", "devices": "设备",
"lan_devices": "有线设备", "lan_devices": "LAN 设备",
"wlan_devices": "WLAN 设备", "wlan_devices": "WLAN 设备",
"lan_users": "有线用户", "lan_users": "有线用户",
"wlan_users": "无线用户", "wlan_users": "无线用户",
@@ -146,7 +146,7 @@
"maxUp": "", "maxUp": "",
"down": "离线", "down": "离线",
"up": "在线", "up": "在线",
"received": "最大上传数", "received": "已接收",
"sent": "已发送", "sent": "已发送",
"externalIPAddress": "外部IP" "externalIPAddress": "外部IP"
}, },
@@ -234,7 +234,7 @@
}, },
"sonarr": { "sonarr": {
"wanted": "关注中", "wanted": "关注中",
"queued": "已加入队列", "queued": "队列",
"series": "剧集", "series": "剧集",
"queue": "队列", "queue": "队列",
"unknown": "未知" "unknown": "未知"
@@ -242,19 +242,19 @@
"radarr": { "radarr": {
"wanted": "关注中", "wanted": "关注中",
"missing": "缺失", "missing": "缺失",
"queued": "已加入队列", "queued": "队列",
"movies": "电影", "movies": "电影",
"queue": "队列", "queue": "队列",
"unknown": "未知" "unknown": "未知"
}, },
"lidarr": { "lidarr": {
"wanted": "关注中", "wanted": "关注中",
"queued": "已加入队列", "queued": "队列",
"artists": "艺术家" "artists": "艺术家"
}, },
"readarr": { "readarr": {
"wanted": "关注中", "wanted": "关注中",
"queued": "已加入队列", "queued": "队列",
"books": "书籍" "books": "书籍"
}, },
"bazarr": { "bazarr": {
@@ -375,7 +375,7 @@
}, },
"medusa": { "medusa": {
"wanted": "关注中", "wanted": "关注中",
"queued": "已加入队列", "queued": "队列",
"series": "剧集" "series": "剧集"
}, },
"minecraft": { "minecraft": {
@@ -802,7 +802,7 @@
"up": "在线", "up": "在线",
"down": "离线", "down": "离线",
"bytesTx": "已传输", "bytesTx": "已传输",
"bytesRx": "最大上传数" "bytesRx": "已接收"
}, },
"uptimerobot": { "uptimerobot": {
"status": "状态", "status": "状态",
@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "警告", "alerts": "警告",
"bans": "禁用" "bans": "禁用"
},
"wgeasy": {
"connected": "已连接",
"enabled": "启用",
"disabled": "禁用",
"total": "总计"
},
"swagdashboard": {
"proxied": "已代理",
"auth": "使用认证",
"outdated": "已过期",
"banned": "已禁止"
} }
} }

View File

@@ -876,5 +876,17 @@
"crowdsec": { "crowdsec": {
"alerts": "警示", "alerts": "警示",
"bans": "禁止" "bans": "禁止"
},
"wgeasy": {
"connected": "Connected",
"enabled": "已啟用",
"disabled": "已停用",
"total": "全部"
},
"swagdashboard": {
"proxied": "Proxied",
"auth": "With Auth",
"outdated": "Outdated",
"banned": "Banned"
} }
} }

View File

@@ -1,26 +1,18 @@
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useEffect, useState, useRef, useCallback, useContext } from "react"; import { useEffect, useState, useRef, useCallback, useContext } from "react";
import classNames from "classnames"; import classNames from "classnames";
import useSWR from "swr";
import ResolvedIcon from "./resolvedicon"; import ResolvedIcon from "./resolvedicon";
import { getStoredProvider, searchProviders } from "./widgets/search/search";
import { SettingsContext } from "utils/contexts/settings"; import { SettingsContext } from "utils/contexts/settings";
export default function QuickLaunch({ export default function QuickLaunch({ servicesAndBookmarks, searchString, setSearchString, isOpen, close }) {
servicesAndBookmarks,
searchString,
setSearchString,
isOpen,
close,
searchProvider,
}) {
const { t } = useTranslation(); const { t } = useTranslation();
const { settings } = useContext(SettingsContext); const { settings } = useContext(SettingsContext);
const { searchDescriptions = false, hideVisitURL = false } = settings?.quicklaunch ?? {}; const { searchDescriptions = false, hideVisitURL = false } = settings?.quicklaunch ?? {};
const showSearchSuggestions = !!(
settings?.quicklaunch?.showSearchSuggestions ?? searchProvider?.showSearchSuggestions
);
const searchField = useRef(); const searchField = useRef();
@@ -29,9 +21,42 @@ export default function QuickLaunch({
const [url, setUrl] = useState(null); const [url, setUrl] = useState(null);
const [searchSuggestions, setSearchSuggestions] = useState([]); const [searchSuggestions, setSearchSuggestions] = useState([]);
const { data: widgets } = useSWR("/api/widgets");
const searchWidget = Object.values(widgets).find((w) => w.type === "search");
let searchProvider;
if (settings?.quicklaunch?.provider === "custom" && settings?.quicklaunch?.url?.length > 0) {
searchProvider = settings.quicklaunch;
} else if (settings?.quicklaunch?.provider && settings?.quicklaunch?.provider !== "custom") {
searchProvider = searchProviders[settings.quicklaunch.provider];
} else if (searchWidget) {
// If there is no search provider in quick launch settings, try to get it from the search widget
if (Array.isArray(searchWidget.options?.provider)) {
// If search provider is a list, try to retrieve from localstorage, fall back to the first
searchProvider = getStoredProvider() ?? searchProviders[searchWidget.options.provider[0]];
} else if (searchWidget.options?.provider === "custom") {
searchProvider = searchWidget.options;
} else {
searchProvider = searchProviders[searchWidget.options?.provider];
}
}
if (searchProvider) {
searchProvider.showSearchSuggestions = !!(
settings?.quicklaunch?.showSearchSuggestions ??
searchWidget?.options?.showSearchSuggestions ??
false
);
}
function openCurrentItem(newWindow) { function openCurrentItem(newWindow) {
const result = results[currentItemIndex]; const result = results[currentItemIndex];
window.open(result.href, newWindow ? "_blank" : result.target ?? settings.target ?? "_blank", "noreferrer"); window.open(
result.href,
newWindow ? "_blank" : result.target ?? searchProvider?.target ?? settings.target ?? "_blank",
"noreferrer",
);
} }
const closeAndReset = useCallback(() => { const closeAndReset = useCallback(() => {
@@ -119,7 +144,7 @@ export default function QuickLaunch({
type: "search", type: "search",
}); });
if (showSearchSuggestions && searchProvider.suggestionUrl) { if (searchProvider.showSearchSuggestions && searchProvider.suggestionUrl) {
if (searchString.trim() !== searchSuggestions[0]?.trim()) { if (searchString.trim() !== searchSuggestions[0]?.trim()) {
fetch( fetch(
`/api/search/searchSuggestion?query=${encodeURIComponent(searchString)}&providerName=${ `/api/search/searchSuggestion?query=${encodeURIComponent(searchString)}&providerName=${
@@ -172,17 +197,7 @@ export default function QuickLaunch({
return () => { return () => {
abortController.abort(); abortController.abort();
}; };
}, [ }, [searchString, servicesAndBookmarks, searchDescriptions, hideVisitURL, searchSuggestions, searchProvider, url, t]);
searchString,
servicesAndBookmarks,
searchDescriptions,
hideVisitURL,
showSearchSuggestions,
searchSuggestions,
searchProvider,
url,
t,
]);
const [hidden, setHidden] = useState(true); const [hidden, setHidden] = useState(true);
useEffect(() => { useEffect(() => {

View File

@@ -1,5 +1,5 @@
import { useTranslation } from "next-i18next"; import { useTranslation } from "next-i18next";
import { FaThermometerHalf } from "react-icons/fa"; import { FiHardDrive } from "react-icons/fi";
import Resource from "../widget/resource"; import Resource from "../widget/resource";
import WidgetLabel from "../widget/widget_label"; import WidgetLabel from "../widget/widget_label";
@@ -10,7 +10,7 @@ export default function Node({ data, expanded, labels }) {
return ( return (
<Resource <Resource
additionalClassNames="information-widget-longhorn-node" additionalClassNames="information-widget-longhorn-node"
icon={FaThermometerHalf} icon={FiHardDrive}
value={t("common.bytes", { value: data.node.available })} value={t("common.bytes", { value: data.node.available })}
label={t("resources.free")} label={t("resources.free")}
expandedValue={t("common.bytes", { value: data.node.maximum })} expandedValue={t("common.bytes", { value: data.node.maximum })}

View File

@@ -14,8 +14,8 @@ import useWidgetAPI from "utils/proxy/use-widget-api";
export default function Widget({ options }) { export default function Widget({ options }) {
const { t } = useTranslation(); const { t } = useTranslation();
// eslint-disable-next-line no-param-reassign // eslint-disable-next-line no-param-reassign, no-multi-assign
options.type = "unifi_console"; options.service_group = options.service_name = "unifi_console";
const { data: statsData, error: statsError } = useWidgetAPI(options, "stat/sites", { index: options.index }); const { data: statsData, error: statsError } = useWidgetAPI(options, "stat/sites", { index: options.index });
if (statsError) { if (statsError) {

View File

@@ -43,8 +43,9 @@ export default async function handler(req, res) {
if (pods.length === 0) { if (pods.length === 0) {
res.status(404).send({ res.status(404).send({
error: `no pods found with namespace=${namespace} and labelSelector=${labelSelector}`, status: "not found",
}); });
logger.error(`no pods found with namespace=${namespace} and labelSelector=${labelSelector}`);
return; return;
} }
const someReady = pods.find((pod) => pod.status.phase === "Running"); const someReady = pods.find((pod) => pod.status.phase === "Running");

View File

@@ -2,12 +2,21 @@ import { formatApiCall } from "utils/proxy/api-helpers";
import createLogger from "utils/logger"; import createLogger from "utils/logger";
import genericProxyHandler from "utils/proxy/handlers/generic"; import genericProxyHandler from "utils/proxy/handlers/generic";
import widgets from "widgets/widgets"; import widgets from "widgets/widgets";
import calendarProxyHandler from "widgets/calendar/proxy";
import getServiceWidget from "utils/config/service-helpers";
const logger = createLogger("servicesProxy"); const logger = createLogger("servicesProxy");
export default async function handler(req, res) { export default async function handler(req, res) {
try { try {
const { type } = req.query; const { service, group } = req.query;
const serviceWidget = await getServiceWidget(group, service);
let type = serviceWidget?.type;
// exceptions
if (type === "calendar") type = "ical";
else if (service === "unifi_console" && group === "unifi_console") type = "unifi_console";
const widget = widgets[type]; const widget = widgets[type];
if (!widget) { if (!widget) {
@@ -18,6 +27,11 @@ export default async function handler(req, res) {
const serviceProxyHandler = widget.proxyHandler || genericProxyHandler; const serviceProxyHandler = widget.proxyHandler || genericProxyHandler;
if (serviceProxyHandler instanceof Function) { if (serviceProxyHandler instanceof Function) {
// quick return for no endpoint services, calendar is an exception
if (!req.query.endpoint || serviceProxyHandler === calendarProxyHandler) {
return serviceProxyHandler(req, res);
}
// map opaque endpoints to their actual endpoint // map opaque endpoints to their actual endpoint
if (widget?.mappings) { if (widget?.mappings) {
const mapping = widget?.mappings?.[req.query.endpoint]; const mapping = widget?.mappings?.[req.query.endpoint];
@@ -27,6 +41,11 @@ export default async function handler(req, res) {
const endpoint = mapping?.endpoint; const endpoint = mapping?.endpoint;
const endpointProxy = mapping?.proxyHandler || serviceProxyHandler; const endpointProxy = mapping?.proxyHandler || serviceProxyHandler;
if (mapping.method && mapping.method !== req.method) {
logger.debug("Unsupported method: %s", req.method);
return res.status(403).json({ error: "Unsupported method" });
}
if (!endpoint) { if (!endpoint) {
logger.debug("Unsupported service endpoint: %s", type); logger.debug("Unsupported service endpoint: %s", type);
return res.status(403).json({ error: "Unsupported service endpoint" }); return res.status(403).json({ error: "Unsupported service endpoint" });
@@ -38,6 +57,17 @@ export default async function handler(req, res) {
if (req.query.segments) { if (req.query.segments) {
const segments = JSON.parse(req.query.segments); const segments = JSON.parse(req.query.segments);
let validSegments = true;
Object.keys(segments).forEach((key) => {
if (!mapping.segments.includes(key)) {
logger.debug("Unsupported segment: %s", key);
validSegments = false;
} else if (segments[key].includes("/") || segments[key].includes("\\") || segments[key].includes("..")) {
logger.debug("Unsupported segment value: %s", segments[key]);
validSegments = false;
}
});
if (!validSegments) return res.status(403).json({ error: "Unsupported segment" });
req.query.endpoint = formatApiCall(endpoint, segments); req.query.endpoint = formatApiCall(endpoint, segments);
} }
@@ -66,7 +96,14 @@ export default async function handler(req, res) {
return serviceProxyHandler(req, res, map); return serviceProxyHandler(req, res, map);
} }
return serviceProxyHandler(req, res); if (widget.allowedEndpoints instanceof RegExp) {
if (widget.allowedEndpoints.test(req.query.endpoint)) {
return serviceProxyHandler(req, res);
}
}
logger.debug("Unmapped proxy request.");
return res.status(403).json({ error: "Unmapped proxy request." });
} }
logger.debug("Unknown proxy service type: %s", type); logger.debug("Unknown proxy service type: %s", type);

View File

@@ -1,9 +1,11 @@
import cachedFetch from "utils/proxy/cached-fetch"; import cachedFetch from "utils/proxy/cached-fetch";
import { getSettings } from "utils/config/config"; import { getSettings } from "utils/config/config";
import { getPrivateWidgetOptions } from "utils/config/widget-helpers";
export default async function handler(req, res) { export default async function handler(req, res) {
const { latitude, longitude, units, provider, cache, lang } = req.query; const { latitude, longitude, units, provider, cache, lang, index } = req.query;
let { apiKey } = req.query; const privateWidgetOptions = await getPrivateWidgetOptions("openweathermap", index);
let { apiKey } = privateWidgetOptions;
if (!apiKey && !provider) { if (!apiKey && !provider) {
return res.status(400).json({ error: "Missing API key or provider" }); return res.status(400).json({ error: "Missing API key or provider" });

View File

@@ -1,9 +1,11 @@
import cachedFetch from "utils/proxy/cached-fetch"; import cachedFetch from "utils/proxy/cached-fetch";
import { getSettings } from "utils/config/config"; import { getSettings } from "utils/config/config";
import { getPrivateWidgetOptions } from "utils/config/widget-helpers";
export default async function handler(req, res) { export default async function handler(req, res) {
const { latitude, longitude, provider, cache, lang } = req.query; const { latitude, longitude, provider, cache, lang, index } = req.query;
let { apiKey } = req.query; const privateWidgetOptions = await getPrivateWidgetOptions("weatherapi", index);
let { apiKey } = privateWidgetOptions;
if (!apiKey && !provider) { if (!apiKey && !provider) {
return res.status(400).json({ error: "Missing API key or provider" }); return res.status(400).json({ error: "Missing API key or provider" });

View File

@@ -26,7 +26,6 @@ import { bookmarksResponse, servicesResponse, widgetsResponse } from "utils/conf
import ErrorBoundary from "components/errorboundry"; import ErrorBoundary from "components/errorboundry";
import themes from "utils/styles/themes"; import themes from "utils/styles/themes";
import QuickLaunch from "components/quicklaunch"; import QuickLaunch from "components/quicklaunch";
import { getStoredProvider, searchProviders } from "components/widgets/search/search";
const ThemeToggle = dynamic(() => import("components/toggles/theme"), { const ThemeToggle = dynamic(() => import("components/toggles/theme"), {
ssr: false, ssr: false,
@@ -204,20 +203,6 @@ function Home({ initialSettings }) {
const [searching, setSearching] = useState(false); const [searching, setSearching] = useState(false);
const [searchString, setSearchString] = useState(""); const [searchString, setSearchString] = useState("");
let searchProvider = null;
const searchWidget = Object.values(widgets).find((w) => w.type === "search");
if (searchWidget) {
if (Array.isArray(searchWidget.options?.provider)) {
// if search provider is a list, try to retrieve from localstorage, fall back to the first
searchProvider = getStoredProvider() ?? searchProviders[searchWidget.options.provider[0]];
} else if (searchWidget.options?.provider === "custom") {
searchProvider = searchWidget.options;
} else {
searchProvider = searchProviders[searchWidget.options?.provider];
}
// to pass to quicklaunch
searchProvider.showSearchSuggestions = searchWidget.options?.showSearchSuggestions;
}
const headerStyle = settings?.headerStyle || "underlined"; const headerStyle = settings?.headerStyle || "underlined";
useEffect(() => { useEffect(() => {
@@ -227,7 +212,8 @@ function Home({ initialSettings }) {
(e.key.length === 1 && (e.key.length === 1 &&
e.key.match(/(\w|\s|[à-ü]|[À-Ü]|[\w\u0430-\u044f])/gi) && e.key.match(/(\w|\s|[à-ü]|[À-Ü]|[\w\u0430-\u044f])/gi) &&
!(e.altKey || e.ctrlKey || e.metaKey || e.shiftKey)) || !(e.altKey || e.ctrlKey || e.metaKey || e.shiftKey)) ||
e.key.match(/([à-ü]|[À-Ü])/g) || // accented characters may require modifier keys // accented characters and the bang may require modifier keys
e.key.match(/([à-ü]|[À-Ü]|!)/g) ||
(e.key === "v" && (e.ctrlKey || e.metaKey)) (e.key === "v" && (e.ctrlKey || e.metaKey))
) { ) {
setSearching(true); setSearching(true);
@@ -403,7 +389,6 @@ function Home({ initialSettings }) {
setSearchString={setSearchString} setSearchString={setSearchString}
isOpen={searching} isOpen={searching}
close={setSearching} close={setSearching}
searchProvider={settings.quicklaunch?.hideInternetSearch ? null : searchProvider}
/> />
<div <div
id="information-widgets" id="information-widgets"

View File

@@ -254,7 +254,8 @@ export async function servicesFromKubernetes() {
ingress.metadata.annotations && ingress.metadata.annotations &&
ingress.metadata.annotations[`${ANNOTATION_BASE}/enabled`] === "true" && ingress.metadata.annotations[`${ANNOTATION_BASE}/enabled`] === "true" &&
(!ingress.metadata.annotations[`${ANNOTATION_BASE}/instance`] || (!ingress.metadata.annotations[`${ANNOTATION_BASE}/instance`] ||
ingress.metadata.annotations[`${ANNOTATION_BASE}/instance`] === instanceName), ingress.metadata.annotations[`${ANNOTATION_BASE}/instance`] === instanceName ||
`${ANNOTATION_BASE}/instance.${instanceName}` in ingress.metadata.annotations),
) )
.map((ingress) => { .map((ingress) => {
let constructedService = { let constructedService = {
@@ -462,6 +463,9 @@ export function cleanServiceGroups(groups) {
// unifi // unifi
site, site,
// wgeasy
threshold,
} = cleanedService.widget; } = cleanedService.widget;
let fieldsList = fields; let fieldsList = fields;
@@ -596,6 +600,9 @@ export function cleanServiceGroups(groups) {
cleanedService.widget.bitratePrecision = parseInt(bitratePrecision, 10); cleanedService.widget.bitratePrecision = parseInt(bitratePrecision, 10);
} }
} }
if (type === "wgeasy") {
if (threshold !== undefined) cleanedService.widget.threshold = parseInt(threshold, 10);
}
} }
return cleanedService; return cleanedService;

View File

@@ -32,7 +32,7 @@ export async function cleanWidgetGroups(widgets) {
const optionKeys = Object.keys(sanitizedOptions); const optionKeys = Object.keys(sanitizedOptions);
// delete private options from the sanitized options // delete private options from the sanitized options
["username", "password", "key"].forEach((pO) => { ["username", "password", "key", "apiKey"].forEach((pO) => {
if (optionKeys.includes(pO)) { if (optionKeys.includes(pO)) {
delete sanitizedOptions[pO]; delete sanitizedOptions[pO];
} }
@@ -57,7 +57,7 @@ export async function getPrivateWidgetOptions(type, widgetIndex) {
const widgets = await widgetsFromConfig(); const widgets = await widgetsFromConfig();
const privateOptions = widgets.map((widget) => { const privateOptions = widgets.map((widget) => {
const { index, url, username, password, key } = widget.options; const { index, url, username, password, key, apiKey } = widget.options;
return { return {
type: widget.type, type: widget.type,
@@ -67,6 +67,7 @@ export async function getPrivateWidgetOptions(type, widgetIndex) {
username, username,
password, password,
key, key,
apiKey,
}, },
}; };
}); });

View File

@@ -8,22 +8,15 @@ export function formatApiCall(url, args) {
return url.replace(/\/+$/, "").replace(find, replace).replace(find, replace); return url.replace(/\/+$/, "").replace(find, replace).replace(find, replace);
} }
function getURLSearchParams(widget, endpoint) { export function getURLSearchParams(widget, endpoint) {
const params = new URLSearchParams({ const params = new URLSearchParams({
type: widget.type,
group: widget.service_group, group: widget.service_group,
service: widget.service_name, service: widget.service_name,
endpoint,
}); });
return params; if (endpoint) {
} params.append("endpoint", endpoint);
export function formatProxyUrlWithSegments(widget, endpoint, segments) {
const params = getURLSearchParams(widget, endpoint);
if (segments) {
params.append("segments", JSON.stringify(segments));
} }
return `/api/services/proxy?${params.toString()}`; return params;
} }
export function formatProxyUrl(widget, endpoint, queryParams) { export function formatProxyUrl(widget, endpoint, queryParams) {
@@ -59,6 +52,7 @@ export function sanitizeErrorURL(errorURL) {
const url = new URL(errorURL); const url = new URL(errorURL);
["apikey", "api_key", "token", "t", "access_token", "auth"].forEach((key) => { ["apikey", "api_key", "token", "t", "access_token", "auth"].forEach((key) => {
if (url.searchParams.has(key)) url.searchParams.set(key, "***"); if (url.searchParams.has(key)) url.searchParams.set(key, "***");
if (url.hash.includes(key)) url.hash = url.hash.replace(new RegExp(`${key}=[^&]+`), `${key}=***`);
}); });
return url.toString(); return url.toString();
} }

View File

@@ -31,6 +31,10 @@ export async function sendJsonRpcRequest(url, method, params, username, password
if (status === 200) { if (status === 200) {
const json = JSON.parse(data.toString()); const json = JSON.parse(data.toString());
if (json.id === null) {
json.id = 1;
}
// in order to get access to the underlying error object in the JSON response // in order to get access to the underlying error object in the JSON response
// you must set `result` equal to undefined // you must set `result` equal to undefined
if (json.error && json.result === null) { if (json.error && json.result === null) {

View File

@@ -8,7 +8,7 @@ export default function Component({ service }) {
const { t } = useTranslation(); const { t } = useTranslation();
const { widget } = service; const { widget } = service;
const { data: resultData, error: resultError } = useWidgetAPI(widget, "result"); const { data: resultData, error: resultError } = useWidgetAPI(widget, "upstreams");
if (resultError) { if (resultError) {
return <Container service={service} error={resultError} />; return <Container service={service} error={resultError} />;
@@ -17,7 +17,6 @@ export default function Component({ service }) {
if (!resultData) { if (!resultData) {
return ( return (
<Container service={service}> <Container service={service}>
,
<Block label="caddy.upstreams" /> <Block label="caddy.upstreams" />
<Block label="caddy.requests" /> <Block label="caddy.requests" />
<Block label="caddy.requests_failed" /> <Block label="caddy.requests_failed" />

View File

@@ -1,8 +1,14 @@
import genericProxyHandler from "utils/proxy/handlers/generic"; import genericProxyHandler from "utils/proxy/handlers/generic";
const widget = { const widget = {
api: "{url}/reverse_proxy/upstreams", api: "{url}/{endpoint}",
proxyHandler: genericProxyHandler, proxyHandler: genericProxyHandler,
mappings: {
upstreams: {
endpoint: "reverse_proxy/upstreams",
},
},
}; };
export default widget; export default widget;

View File

@@ -65,7 +65,7 @@ export default function Component({ service }) {
return ( return (
<Container service={service}> <Container service={service}>
<div className={classNames(service.description ? "-top-10" : "-top-8", "absolute right-1")}> <div className={classNames(service.description ? "-top-10" : "-top-8", "absolute right-1 z-20")}>
<Dropdown options={dateRangeOptions} value={dateRange} setValue={setDateRange} /> <Dropdown options={dateRangeOptions} value={dateRange} setValue={setDateRange} />
</div> </div>

View File

@@ -103,6 +103,7 @@ const components = {
speedtest: dynamic(() => import("./speedtest/component")), speedtest: dynamic(() => import("./speedtest/component")),
stash: dynamic(() => import("./stash/component")), stash: dynamic(() => import("./stash/component")),
strelaysrv: dynamic(() => import("./strelaysrv/component")), strelaysrv: dynamic(() => import("./strelaysrv/component")),
swagdashboard: dynamic(() => import("./swagdashboard/component")),
tailscale: dynamic(() => import("./tailscale/component")), tailscale: dynamic(() => import("./tailscale/component")),
tandoor: dynamic(() => import("./tandoor/component")), tandoor: dynamic(() => import("./tandoor/component")),
tautulli: dynamic(() => import("./tautulli/component")), tautulli: dynamic(() => import("./tautulli/component")),
@@ -117,6 +118,7 @@ const components = {
uptimerobot: dynamic(() => import("./uptimerobot/component")), uptimerobot: dynamic(() => import("./uptimerobot/component")),
urbackup: dynamic(() => import("./urbackup/component")), urbackup: dynamic(() => import("./urbackup/component")),
watchtower: dynamic(() => import("./watchtower/component")), watchtower: dynamic(() => import("./watchtower/component")),
wgeasy: dynamic(() => import("./wgeasy/component")),
whatsupdocker: dynamic(() => import("./whatsupdocker/component")), whatsupdocker: dynamic(() => import("./whatsupdocker/component")),
xteve: dynamic(() => import("./xteve/component")), xteve: dynamic(() => import("./xteve/component")),
}; };

View File

@@ -4,7 +4,7 @@ import { MdOutlineSmartDisplay } from "react-icons/md";
import Block from "components/services/widget/block"; import Block from "components/services/widget/block";
import Container from "components/services/widget/container"; import Container from "components/services/widget/container";
import { formatProxyUrlWithSegments } from "utils/proxy/api-helpers"; import { getURLSearchParams } from "utils/proxy/api-helpers";
import useWidgetAPI from "utils/proxy/use-widget-api"; import useWidgetAPI from "utils/proxy/use-widget-api";
function ticksToTime(ticks) { function ticksToTime(ticks) {
@@ -217,11 +217,17 @@ export default function Component({ service }) {
}); });
async function handlePlayCommand(session, command) { async function handlePlayCommand(session, command) {
const url = formatProxyUrlWithSegments(widget, "PlayControl", { const params = getURLSearchParams(widget, command);
sessionId: session.Id, params.append(
command, "segments",
}); JSON.stringify({
await fetch(url).then(() => { sessionId: session.Id,
}),
);
const url = `/api/services/proxy?${params.toString()}`;
await fetch(url, {
method: "POST",
}).then(() => {
sessionMutate(); sessionMutate();
}); });
} }

View File

@@ -10,12 +10,16 @@ const widget = {
}, },
Count: { Count: {
endpoint: "Items/Counts", endpoint: "Items/Counts",
segments: ["MovieCount", "SeriesCount", "EpisodeCount", "SongCount"],
}, },
PlayControl: { Unpause: {
method: "POST", method: "POST",
endpoint: "Sessions/{sessionId}/Playing/{command}", endpoint: "Sessions/{sessionId}/Playing/Unpause",
segments: ["sessionId", "command"], segments: ["sessionId"],
},
Pause: {
method: "POST",
endpoint: "Sessions/{sessionId}/Playing/Pause",
segments: ["sessionId"],
}, },
}, },
}; };

View File

@@ -17,7 +17,6 @@ export default function Component({ service }) {
if (!stateData) { if (!stateData) {
return ( return (
<Container service={service}> <Container service={service}>
,
<Block label="evcc.pv_power" /> <Block label="evcc.pv_power" />
<Block label="evcc.grid_power" /> <Block label="evcc.grid_power" />
<Block label="evcc.home_power" /> <Block label="evcc.home_power" />

View File

@@ -2,6 +2,12 @@ import floodProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: floodProxyHandler, proxyHandler: floodProxyHandler,
mappings: {
torrents: {
endpoint: "torrents",
},
},
}; };
export default widget; export default widget;

View File

@@ -2,6 +2,7 @@ import fritzboxProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: fritzboxProxyHandler, proxyHandler: fritzboxProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -2,6 +2,7 @@ import gamedigProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: gamedigProxyHandler, proxyHandler: gamedigProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -5,8 +5,12 @@ import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api"; import useWidgetAPI from "utils/proxy/use-widget-api";
function getPerformancePercent(t, performanceRange) { function getPerformancePercent(t, performanceRange) {
return `${performanceRange.performance.currentGrossPerformancePercent > 0 ? "+" : ""}${t("common.percent", { // ghostfolio v2.79.0 changed to grossPerformancePercentage
value: performanceRange.performance.currentGrossPerformancePercent * 100, const percent =
performanceRange.performance.grossPerformancePercentage ??
performanceRange.performance.currentGrossPerformancePercent;
return `${percent > 0 ? "+" : ""}${t("common.percent", {
value: percent * 100,
maximumFractionDigits: 2, maximumFractionDigits: 2,
})}`; })}`;
} }
@@ -24,6 +28,10 @@ export default function Component({ service }) {
return <Container service={service} error={finalError} />; return <Container service={service} error={finalError} />;
} }
if (performanceToday?.statusCode === 401) {
return <Container service={service} error={performanceToday} />;
}
if (!performanceToday || !performanceYear || !performanceMax) { if (!performanceToday || !performanceYear || !performanceMax) {
return ( return (
<Container service={service}> <Container service={service}>

View File

@@ -3,6 +3,7 @@ import credentialedProxyHandler from "utils/proxy/handlers/credentialed";
const widget = { const widget = {
api: "{url}/api/{endpoint}", api: "{url}/api/{endpoint}",
proxyHandler: credentialedProxyHandler, proxyHandler: credentialedProxyHandler,
allowedEndpoints: /\d\/quicklook|diskio|cpu|fs|gpu|system|mem|network|processlist|sensors/,
}; };
export default widget; export default widget;

View File

@@ -18,10 +18,13 @@ export default function Component({ service }) {
); );
if (statsError || statusError) { if (statsError || statusError) {
return <Container service={service} error={statsError ?? statusError} />; return <Container service={service} error={statsError ?? statusError ?? statusData} />;
} }
if (statusData && !(statusData.status.includes("running") || statusData.status.includes("partial"))) { if (
statusData &&
(!statusData.status || !(statusData.status.includes("running") || statusData.status.includes("partial")))
) {
return ( return (
<Container> <Container>
<Block label={t("widget.status")} value={t("docker.offline")} /> <Block label={t("widget.status")} value={t("docker.offline")} />

View File

@@ -2,6 +2,7 @@ import minecraftProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: minecraftProxyHandler, proxyHandler: minecraftProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -5,7 +5,7 @@ import useWidgetAPI from "utils/proxy/use-widget-api";
export default function Component({ service }) { export default function Component({ service }) {
const { widget } = service; const { widget } = service;
const { data: infoData, error: infoError } = useWidgetAPI(widget, "nginx/proxy-hosts"); const { data: infoData, error: infoError } = useWidgetAPI(widget, "hosts");
if (infoError) { if (infoError) {
return <Container service={service} error={infoError} />; return <Container service={service} error={infoError} />;

View File

@@ -3,6 +3,12 @@ import npmProxyHandler from "./proxy";
const widget = { const widget = {
api: "{url}/api/{endpoint}", api: "{url}/api/{endpoint}",
proxyHandler: npmProxyHandler, proxyHandler: npmProxyHandler,
mappings: {
hosts: {
endpoint: "nginx/proxy-hosts",
},
},
}; };
export default widget; export default widget;

View File

@@ -3,6 +3,7 @@ import jsonrpcProxyHandler from "utils/proxy/handlers/jsonrpc";
const widget = { const widget = {
api: "{url}/jsonrpc", api: "{url}/jsonrpc",
proxyHandler: jsonrpcProxyHandler, proxyHandler: jsonrpcProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -9,7 +9,7 @@ export default function Component({ service }) {
const { widget } = service; const { widget } = service;
const { data: omadaData, error: omadaAPIError } = useWidgetAPI(widget, { const { data: omadaData, error: omadaAPIError } = useWidgetAPI(widget, "info", {
refreshInterval: 5000, refreshInterval: 5000,
}); });

View File

@@ -2,6 +2,12 @@ import omadaProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: omadaProxyHandler, proxyHandler: omadaProxyHandler,
mappings: {
info: {
endpoint: "api/info",
},
},
}; };
export default widget; export default widget;

View File

@@ -77,7 +77,7 @@ async function fetchSystem(url) {
const systemResponse = JSON.parse(data.toString())[1]; const systemResponse = JSON.parse(data.toString())[1];
const response = { const response = {
uptime: systemResponse.uptime, uptime: systemResponse.uptime,
cpuLoad: systemResponse.load[1], cpuLoad: (systemResponse.load[1] / 65536.0).toFixed(2),
}; };
return [200, contentType, response]; return [200, contentType, response];
} }

View File

@@ -9,7 +9,7 @@ export default function Component({ service }) {
const { widget } = service; const { widget } = service;
const { data: torrentData, error: torrentError } = useWidgetAPI(widget, "torrents/info"); const { data: torrentData, error: torrentError } = useWidgetAPI(widget, "torrents");
if (torrentError) { if (torrentError) {
return <Container service={service} error={torrentError} />; return <Container service={service} error={torrentError} />;

View File

@@ -2,6 +2,12 @@ import qbittorrentProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: qbittorrentProxyHandler, proxyHandler: qbittorrentProxyHandler,
mappings: {
torrents: {
endpoint: "torrents/info",
},
},
}; };
export default widget; export default widget;

View File

@@ -3,6 +3,7 @@ import qnapProxyHandler from "./proxy";
const widget = { const widget = {
api: "{url}", api: "{url}",
proxyHandler: qnapProxyHandler, proxyHandler: qnapProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -1,18 +1,26 @@
import { useTranslation } from "next-i18next"; import { useTranslation } from "next-i18next";
import { useEffect, useState } from "react";
import Container from "components/services/widget/container"; import Container from "components/services/widget/container";
import Block from "components/services/widget/block"; import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api"; import { formatProxyUrl } from "utils/proxy/api-helpers";
export default function Component({ service }) { export default function Component({ service }) {
const { t } = useTranslation(); const { t } = useTranslation();
const { widget } = service; const { widget } = service;
const { data: stats, error: stashError } = useWidgetAPI(widget, "stats"); const [stats, setStats] = useState(null);
if (stashError) { useEffect(() => {
return <Container service={service} error={stashError} />; async function fetchStats() {
} const url = formatProxyUrl(widget, "stats");
const res = await fetch(url, { method: "POST" });
setStats(await res.json());
}
if (!stats) {
fetchStats();
}
}, [widget, stats]);
if (!stats) { if (!stats) {
return ( return (

View File

@@ -0,0 +1,33 @@
import Container from "components/services/widget/container";
import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api";
export default function Component({ service }) {
const { widget } = service;
const { data: swagData, error: swagError } = useWidgetAPI(widget, "overview");
if (swagError) {
return <Container service={service} error={swagError} />;
}
if (!swagData) {
return (
<Container service={service}>
<Block label="swagdashboard.proxied" />
<Block label="swagdashboard.auth" />
<Block label="swagdashboard.outdated" />
<Block label="swagdashboard.banned" />
</Container>
);
}
return (
<Container service={service}>
<Block label="swagdashboard.proxied" value={swagData.proxied} />
<Block label="swagdashboard.auth" value={swagData.auth} />
<Block label="swagdashboard.outdated" value={swagData.outdated} />
<Block label="swagdashboard.banned" value={swagData.banned} />
</Container>
);
}

View File

@@ -0,0 +1,9 @@
import genericProxyHandler from "utils/proxy/handlers/generic";
const widget = {
api: "{url}/?stats=true",
proxyHandler: genericProxyHandler,
allowedEndpoints: /overview/,
};
export default widget;

View File

@@ -8,7 +8,7 @@ const proxyName = "tdarrProxyHandler";
const logger = createLogger(proxyName); const logger = createLogger(proxyName);
export default async function tdarrProxyHandler(req, res) { export default async function tdarrProxyHandler(req, res) {
const { group, service, endpoint } = req.query; const { group, service } = req.query;
if (!group || !service) { if (!group || !service) {
logger.debug("Invalid or missing service '%s' or group '%s'", service, group); logger.debug("Invalid or missing service '%s' or group '%s'", service, group);
@@ -22,7 +22,7 @@ export default async function tdarrProxyHandler(req, res) {
return res.status(400).json({ error: "Invalid proxy service type" }); return res.status(400).json({ error: "Invalid proxy service type" });
} }
const url = new URL(formatApiCall(widgets[widget.type].api, { endpoint, ...widget })); const url = new URL(formatApiCall(widgets[widget.type].api, { endpoint: undefined, ...widget }));
const [status, contentType, data] = await httpProxy(url, { const [status, contentType, data] = await httpProxy(url, {
method: "POST", method: "POST",

View File

@@ -11,7 +11,7 @@ const headerCacheKey = `${proxyName}__headers`;
const logger = createLogger(proxyName); const logger = createLogger(proxyName);
export default async function transmissionProxyHandler(req, res) { export default async function transmissionProxyHandler(req, res) {
const { group, service, endpoint } = req.query; const { group, service } = req.query;
if (!group || !service) { if (!group || !service) {
logger.debug("Invalid or missing service '%s' or group '%s'", service, group); logger.debug("Invalid or missing service '%s' or group '%s'", service, group);
@@ -35,7 +35,7 @@ export default async function transmissionProxyHandler(req, res) {
const api = `${widget.url}${widget.rpcUrl || widgets[widget.type].rpcUrl}rpc`; const api = `${widget.url}${widget.rpcUrl || widgets[widget.type].rpcUrl}rpc`;
const url = new URL(formatApiCall(api, { endpoint, ...widget })); const url = new URL(formatApiCall(api, { endpoint: undefined, ...widget }));
const csrfHeaderName = "x-transmission-session-id"; const csrfHeaderName = "x-transmission-session-id";
const method = "POST"; const method = "POST";

View File

@@ -14,13 +14,13 @@ const prefixCacheKey = `${proxyName}__prefix`;
const logger = createLogger(proxyName); const logger = createLogger(proxyName);
async function getWidget(req) { async function getWidget(req) {
const { group, service, type } = req.query; const { group, service } = req.query;
let widget = null; let widget = null;
if (type === "unifi_console") { if (group === "unifi_console" && service === "unifi_console") {
// info widget // info widget
const index = req.query?.query ? JSON.parse(req.query.query).index : undefined; const index = req.query?.query ? JSON.parse(req.query.query).index : undefined;
widget = await getPrivateWidgetOptions(type, index); widget = await getPrivateWidgetOptions("unifi_console", index);
if (!widget) { if (!widget) {
logger.debug("Error retrieving settings for this Unifi widget"); logger.debug("Error retrieving settings for this Unifi widget");
return null; return null;

View File

@@ -1,16 +1,30 @@
import { useEffect, useState } from "react";
import Container from "components/services/widget/container"; import Container from "components/services/widget/container";
import Block from "components/services/widget/block"; import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api"; import useWidgetAPI from "utils/proxy/use-widget-api";
import { formatProxyUrl } from "utils/proxy/api-helpers";
export default function Component({ service }) { export default function Component({ service }) {
const { widget } = service; const { widget } = service;
const { data: workersData, error: workersError } = useWidgetAPI(widget, "workers"); const { data: workersData, error: workersError } = useWidgetAPI(widget, "workers");
const { data: pendingData, error: pendingError } = useWidgetAPI(widget, "pending");
if (workersError || pendingError) { const [pendingData, setPendingData] = useState(null);
const finalError = workersError ?? pendingError;
return <Container service={service} error={finalError} />; useEffect(() => {
async function fetchPending() {
const url = formatProxyUrl(widget, "pending");
const res = await fetch(url, { method: "POST" });
setPendingData(await res.json());
}
if (!pendingData) {
fetchPending();
}
}, [widget, pendingData]);
if (workersError) {
return <Container service={service} error={workersError} />;
} }
if (!workersData || !pendingData) { if (!workersData || !pendingData) {

View File

@@ -1,18 +1,26 @@
import { useTranslation } from "next-i18next"; import { useTranslation } from "next-i18next";
import { useEffect, useState } from "react";
import Container from "components/services/widget/container"; import Container from "components/services/widget/container";
import Block from "components/services/widget/block"; import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api"; import { formatProxyUrl } from "utils/proxy/api-helpers";
export default function Component({ service }) { export default function Component({ service }) {
const { widget } = service; const { widget } = service;
const { t } = useTranslation(); const { t } = useTranslation();
const { data: uptimerobotData, error: uptimerobotError } = useWidgetAPI(widget, "getmonitors"); const [uptimerobotData, setUptimerobotData] = useState(null);
if (uptimerobotError) { useEffect(() => {
return <Container service={service} error={uptimerobotError} />; async function fetchData() {
} const url = formatProxyUrl(widget, "getmonitors");
const res = await fetch(url, { method: "POST" });
setUptimerobotData(await res.json());
}
if (!uptimerobotData) {
fetchData();
}
}, [widget, uptimerobotData]);
if (!uptimerobotData) { if (!uptimerobotData) {
return ( return (

View File

@@ -2,6 +2,7 @@ import urbackupProxyHandler from "./proxy";
const widget = { const widget = {
proxyHandler: urbackupProxyHandler, proxyHandler: urbackupProxyHandler,
allowedEndpoints: /status/,
}; };
export default widget; export default widget;

View File

@@ -0,0 +1,45 @@
import Container from "components/services/widget/container";
import Block from "components/services/widget/block";
import useWidgetAPI from "utils/proxy/use-widget-api";
export default function Component({ service }) {
const { widget } = service;
const { data: infoData, error: infoError } = useWidgetAPI(widget);
if (!widget.fields) {
widget.fields = ["connected", "enabled", "total"];
}
if (infoError) {
return <Container service={service} error={infoError} />;
}
if (!infoData) {
return (
<Container service={service}>
<Block label="wgeasy.connected" />
<Block label="wgeasy.enabled" />
<Block label="wgeasy.disabled" />
<Block label="wgeasy.total" />
</Container>
);
}
const enabled = infoData.filter((item) => item.enabled).length;
const disabled = infoData.length - enabled;
const connectionThreshold = (widget.threshold ?? 2) * 60 * 1000;
const currentTime = new Date();
const connected = infoData.filter(
(item) => currentTime - new Date(item.latestHandshakeAt) < connectionThreshold,
).length;
return (
<Container service={service}>
<Block label="wgeasy.connected" value={connected} />
<Block label="wgeasy.enabled" value={enabled} />
<Block label="wgeasy.diabled" value={disabled} />
<Block label="wgeasy.total" value={infoData.length} />
</Container>
);
}

Some files were not shown because too many files have changed in this diff Show More