Compare commits

..

45 Commits

Author SHA1 Message Date
github-actions[bot]
986a18170c New Crowdin translations by GitHub Action (#3321)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-05-05 20:51:37 -07:00
Ben Phelps
198835a697 allow seperate href for widget container links
addresses #3140
2024-05-04 19:34:38 +03:00
dependabot[bot]
5efed2e740 Chore(deps-dev): Bump eslint-config-next from 12.3.4 to 14.2.3 (#3399)
Bumps [eslint-config-next](https://github.com/vercel/next.js/tree/HEAD/packages/eslint-config-next) from 12.3.4 to 14.2.3.
- [Release notes](https://github.com/vercel/next.js/releases)
- [Changelog](https://github.com/vercel/next.js/blob/canary/release.js)
- [Commits](https://github.com/vercel/next.js/commits/v14.2.3/packages/eslint-config-next)

---
updated-dependencies:
- dependency-name: eslint-config-next
  dependency-type: direct:development
  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-01 12:08:26 -07:00
dependabot[bot]
43e258a58d Chore(deps): Bump react from 18.2.0 to 18.3.1 (#3400)
Bumps [react](https://github.com/facebook/react/tree/HEAD/packages/react) from 18.2.0 to 18.3.1.
- [Release notes](https://github.com/facebook/react/releases)
- [Changelog](https://github.com/facebook/react/blob/main/CHANGELOG.md)
- [Commits](https://github.com/facebook/react/commits/v18.3.1/packages/react)

---
updated-dependencies:
- dependency-name: react
  dependency-type: direct:production
  update-type: version-update:semver-minor
...

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-05-01 09:44:31 -07:00
dependabot[bot]
8ca7f422da Chore(deps): Bump systeminformation from 5.22.0 to 5.22.7 (#3398)
Bumps [systeminformation](https://github.com/sebhildebrandt/systeminformation) from 5.22.0 to 5.22.7.
- [Changelog](https://github.com/sebhildebrandt/systeminformation/blob/master/CHANGELOG.md)
- [Commits](https://github.com/sebhildebrandt/systeminformation/compare/v5.22.0...v5.22.7)

---
updated-dependencies:
- dependency-name: systeminformation
  dependency-type: direct:production
  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-05-01 09:44:07 -07:00
dependabot[bot]
4fc70eb1ff Chore(deps-dev): Bump typescript from 4.9.5 to 5.4.5 (#3396)
Bumps [typescript](https://github.com/Microsoft/TypeScript) from 4.9.5 to 5.4.5.
- [Release notes](https://github.com/Microsoft/TypeScript/releases)
- [Changelog](https://github.com/microsoft/TypeScript/blob/main/azure-pipelines.release.yml)
- [Commits](https://github.com/Microsoft/TypeScript/compare/v4.9.5...v5.4.5)

---
updated-dependencies:
- dependency-name: typescript
  dependency-type: direct:development
  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-01 09:43:52 -07:00
dependabot[bot]
d90bf8079a Chore(deps): Bump recharts from 2.12.3 to 2.12.6 (#3397)
Bumps [recharts](https://github.com/recharts/recharts) from 2.12.3 to 2.12.6.
- [Release notes](https://github.com/recharts/recharts/releases)
- [Changelog](https://github.com/recharts/recharts/blob/3.x/CHANGELOG.md)
- [Commits](https://github.com/recharts/recharts/compare/v2.12.3...v2.12.6)

---
updated-dependencies:
- dependency-name: recharts
  dependency-type: direct:production
  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-05-01 09:24:21 -07:00
shamoon
ea63716b61 Fix: some error URLs aren't sanitized (#3385) 2024-04-29 17:18:55 -07:00
Edward Salter
daa51f9740 Fix: format Romm statistics (#3358) 2024-04-23 14:13:53 -07:00
Ameer Abdallah
340424391f Enhancement: add bitrate precision config option for speedtest-tracker (#3354)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-22 21:20:08 -07:00
Ameer Abdallah
312e97d18b Enhancement: additional tautulli jellyfin emby configuration options (#3350) 2024-04-22 17:49:19 -07:00
Liam Dyer
f4fc30cd9f Documentation: update Authentik suggested permissions (#3349) 2024-04-22 13:59:23 -07:00
Nuno Alexandre
595049f7fc Documentation: clarify uptime kuma slug (#3345)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-22 12:33:35 -07:00
shamoon
79e3eb9c90 Documentation: fix docker stats link 2024-04-21 07:12:49 -07:00
David Hirsch
c95837f54e Enhancement: configurable CPU temp scale (#3332)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-19 16:32:14 -07:00
lavavex
068e664f16 Documentation: correct Medusa link (#3320) 2024-04-17 17:00:37 -07:00
github-actions[bot]
12ec1cfdcb New Crowdin translations by GitHub Action (#3270)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-04-17 01:42:55 -07:00
shamoon
303a62369f Fix: pihole ads_percentage_today sometimes returned as string (#3313) 2024-04-16 15:50:32 -07:00
shamoon
034f6d29d6 Docs: show carbon ads on more pages too 2024-04-15 12:53:15 -07:00
Ben Phelps
60098d3909 Docs: move Carbon ads to sidebar (#3302) 2024-04-15 12:20:39 -07:00
Ben Phelps
2c68f1e7ee place carbon ads in docs (#3296)
* place carbon ads in docs

* fix lint

* keep 4 space tabs
2024-04-15 15:59:30 +03:00
brikim
bfd392026d Enhancement: option to show user for Tautulli and Emby/Jellyfin widgets (#3287)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-12 20:33:40 -07:00
shamoon
c18fd02c8e Fix typo in crowdsec docs 2024-04-08 15:19:35 -07:00
github-actions[bot]
ffad2b890e New Crowdin translations by GitHub Action (#3199)
Co-authored-by: Crowdin Bot <support+bot@crowdin.com>
2024-04-08 10:20:52 -07:00
shamoon
4239e8fe97 Update contributing / development guidelines 2024-04-08 08:11:55 -07:00
shamoon
f82a122e26 Fix site monitor with error 2024-04-06 20:00:16 -07:00
shamoon
d49a06efd9 Fix: rename pialert to netalertx (#3253) 2024-04-06 17:35:12 -07:00
mrmorganmurphy
9904c2db2f Documentation: update diskstation instructions (#3230) 2024-04-03 10:03:12 -07:00
XavierDupuis
60db01cc57 Documentation: fix allowed fields uniformity (#3224) 2024-04-02 15:42:34 -07:00
rgon10
4e69ea6088 Fix: TrueNAS Core support for pool stats (#3206)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-01 14:32:39 -07:00
dependabot[bot]
268d8efa0e Chore(deps-dev): Bump tailwindcss from 3.4.1 to 3.4.3 (#3214)
Bumps [tailwindcss](https://github.com/tailwindlabs/tailwindcss) from 3.4.1 to 3.4.3.
- [Release notes](https://github.com/tailwindlabs/tailwindcss/releases)
- [Changelog](https://github.com/tailwindlabs/tailwindcss/blob/v3.4.3/CHANGELOG.md)
- [Commits](https://github.com/tailwindlabs/tailwindcss/compare/v3.4.1...v3.4.3)

---
updated-dependencies:
- dependency-name: tailwindcss
  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-04-01 14:18:39 -07:00
dependabot[bot]
43bbb69d53 Chore(deps-dev): Bump eslint-plugin-react from 7.33.2 to 7.34.1 (#3213)
Bumps [eslint-plugin-react](https://github.com/jsx-eslint/eslint-plugin-react) from 7.33.2 to 7.34.1.
- [Release notes](https://github.com/jsx-eslint/eslint-plugin-react/releases)
- [Changelog](https://github.com/jsx-eslint/eslint-plugin-react/blob/v7.34.1/CHANGELOG.md)
- [Commits](https://github.com/jsx-eslint/eslint-plugin-react/compare/v7.33.2...v7.34.1)

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

Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2024-04-01 14:18:29 -07:00
dependabot[bot]
cdfb5a11f7 Chore(deps): Bump recharts from 2.12.2 to 2.12.3 (#3215)
Bumps [recharts](https://github.com/recharts/recharts) from 2.12.2 to 2.12.3.
- [Release notes](https://github.com/recharts/recharts/releases)
- [Changelog](https://github.com/recharts/recharts/blob/3.x/CHANGELOG.md)
- [Commits](https://github.com/recharts/recharts/compare/v2.12.2...v2.12.3)

---
updated-dependencies:
- dependency-name: recharts
  dependency-type: direct:production
  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-04-01 14:18:07 -07:00
Luca Herrero
2ebcb311e8 Fix: homebridge widget with numeric username or password (#3220) 2024-04-01 14:17:56 -07:00
dependabot[bot]
0d7b77260f Chore(deps-dev): Bump postcss from 8.4.35 to 8.4.38 (#3212)
Bumps [postcss](https://github.com/postcss/postcss) from 8.4.35 to 8.4.38.
- [Release notes](https://github.com/postcss/postcss/releases)
- [Changelog](https://github.com/postcss/postcss/blob/main/CHANGELOG.md)
- [Commits](https://github.com/postcss/postcss/compare/8.4.35...8.4.38)

---
updated-dependencies:
- dependency-name: postcss
  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-04-01 10:18:28 -07:00
dependabot[bot]
6f07acab15 Chore(deps): Bump follow-redirects from 1.15.5 to 1.15.6 (#3211)
Bumps [follow-redirects](https://github.com/follow-redirects/follow-redirects) from 1.15.5 to 1.15.6.
- [Release notes](https://github.com/follow-redirects/follow-redirects/releases)
- [Commits](https://github.com/follow-redirects/follow-redirects/compare/v1.15.5...v1.15.6)

---
updated-dependencies:
- dependency-name: follow-redirects
  dependency-type: direct:production
  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-04-01 10:17:34 -07:00
shamoon
29447c55ff Fix: remove invalid form attribute (#3210) 2024-04-01 07:49:58 -07:00
Jalin Wang
dd819ad677 Documentation: correct link for Docker automatic service discovery (#3208) 2024-04-01 07:25:14 -07:00
Jalin Wang
212e517ebb Chore: fix ypos in the PR template (#3209)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-04-01 07:23:52 -07:00
shamoon
29ac7bfea7 Feature: Support pi-hole v6 (#3207) 2024-03-31 20:34:46 -07:00
shamoon
b0d57866a0 Fix glances service widgets handling of default version 2024-03-29 06:32:25 -07:00
shamoon
97d193faf1 Fix crowdsec widget with no bans / alerts 2024-03-28 21:50:10 -07:00
shamoon
d4c0e482d3 Feature: crowdsec widget (#3197) 2024-03-28 13:39:40 -07:00
shamoon
def9b27006 Enhancement: support for glances v4 (#3196) 2024-03-28 11:16:00 -05:00
ThorTheStorm
4fe4ae9622 Documentation: Update authentik api key info (#3195)
---------

Co-authored-by: shamoon <4887959+shamoon@users.noreply.github.com>
2024-03-28 09:51:07 -05:00
113 changed files with 3337 additions and 1975 deletions

View File

@@ -1,12 +1,12 @@
## Proposed change
<!--
Please include a summary of the change. Screenshots and / or videos can also be helpful if appropriate.
Please include a summary of the change. Screenshots and/or videos can also be helpful if appropriate.
*** Please see the development guidelines for new widgets: https://gethomepage.dev/latest/more/development/#service-widget-guidelines
*** If you do not follow these guidelines your PR will likely be closed without review.
New service widgets should include example(s) of relevant relevant API output as well updates to the docs for the new widget.
New service widgets should include example(s) of relevant API output as well as updates to the docs for the new widget.
-->
Closes # (issue)

View File

@@ -48,11 +48,15 @@ Please see information in the docs regarding [code formatting with pre-commit ho
By contributing, you agree that your contributions will be licensed under its GNU General Public License.
## Use of AI for pull requests
In general, homepage does not accept "AI-generated" PRs. If you choose to use something like that to aid the development process to generate a significant proportion of the pull request, please make sure this is explicitly stated in the PR itself.
## References
This document was adapted from the open-source contribution guidelines for [Facebook's Draft](https://github.com/facebook/draft-js/blob/main/CONTRIBUTING.md)
# Automatic Respository Maintenance
## Automatic Respository Maintenance
The homepage team appreciates all effort and interest from the community in filing bug reports, creating feature requests, sharing ideas and helping other community members. That said, in an effort to keep the repository organized and managebale the project uses automatic handling of certain areas:

View File

@@ -41,7 +41,7 @@ With features like quick search, bookmarks, weather support, a wide range of int
## Docker Integration
Homepage has built-in support for Docker, and can automatically discover and add services to the homepage based on labels. See the [Docker](https://gethomepage.dev/latest/installation/docker/) page for more information.
Homepage has built-in support for Docker, and can automatically discover and add services to the homepage based on labels. See the [Docker Service Discovery](https://gethomepage.dev/latest/configs/docker/#automatic-service-discovery) page for more information.
## Service Widgets

View File

@@ -235,4 +235,4 @@ You can show the docker stats by clicking the status indicator but this can also
showStats: true
```
Also see the settings for [show docker stats](docker.md#show-docker-stats).
Also see the settings for [show docker stats](settings.md#show-docker-stats).

View File

@@ -39,6 +39,11 @@ Once installed, hooks will run when you commit. If the formatting isn't quite ri
See the [pre-commit documentation](https://pre-commit.com/#install) to get started.
## Preferring self-hosted open-source software
In general, homepage is meant to be a dashboard for 'self-hosted' services and we believe it is a small way we can help showcase this kind of software. While exceptions are made, mostly when there is no viable
self-hosted / open-source alternative, we ask that any widgets, etc. are developed primarily for a self-hosted tool.
## New Feature Guidelines
- New features should be linked to an existing feature request with at least 10 'up-votes'. The purpose of this requirement is to avoid the addition (and maintenance) of features that might only benefit a small number of users.

47
docs/overrides/main.html Normal file
View File

@@ -0,0 +1,47 @@
{% extends "base.html" %}
{% block site_nav %}
<!-- Navigation -->
{% if nav %}
{% if page.meta and page.meta.hide %}
{% set hidden = "hidden" if "navigation" in page.meta.hide %}
{% endif %}
<div
class="md-sidebar md-sidebar--primary"
data-md-component="sidebar"
data-md-type="navigation"
{{ hidden }}
>
<div class="md-sidebar__scrollwrap">
<div class="md-sidebar__inner">
{% include "partials/nav.html" %}
{% if 'widgets/' not in page.url and 'more/' not in page.url %}
<script async type="text/javascript" src="//cdn.carbonads.com/carbon.js?serve=CWYIL2JU&placement=gethomepagedev&format=cover" id="_carbonads_js"></script>
{% endif %}
</div>
</div>
</div>
{% endif %}
<!-- Table of contents -->
{% if "toc.integrate" not in features %}
{% if page.meta and page.meta.hide %}
{% set hidden = "hidden" if "toc" in page.meta.hide %}
{% endif %}
<div
class="md-sidebar md-sidebar--secondary"
data-md-component="sidebar"
data-md-type="toc"
{{ hidden }}
>
<div class="md-sidebar__scrollwrap" style="height: 200px;">
<div class="md-sidebar__inner">
{% include "partials/toc.html" %}
{% if 'widgets/' in page.url or 'more/' in page.url %}
<script async type="text/javascript" src="//cdn.carbonads.com/carbon.js?serve=CWYIL2JU&placement=gethomepagedev&format=cover" id="_carbonads_js"></script>
{% endif %}
</div>
</div>
</div>
{% endif %}
{% endblock %}

View File

@@ -22,3 +22,15 @@
#glimeRoot * {
font-family: var(--md-text-font) !important;
}
#carbonads {
margin-top: 10px;
}
#carbon-responsive {
--carbon-padding: 1em;
--carbon-max-char: 20ch;
--carbon-bg-primary: var(--md-default-bg-color) !important;
--carbon-bg-secondary: var(--md-default-fg-color--lightest) !important;
--carbon-text-color: var(--md-typeset-color) !important;
}

View File

@@ -12,6 +12,7 @@ The Glances widget allows you to monitor the resources (CPU, memory, storage, te
url: http://host.or.ip:port
username: user # optional if auth enabled in Glances
password: pass # optional if auth enabled in Glances
version: 4 # required only if running glances v4 or higher, defaults to 3
cpu: true # optional, enabled by default, disable by setting to false
mem: true # optional, enabled by default, disable by setting to false
cputemp: true # disabled by default

View File

@@ -19,6 +19,8 @@ _Note: unfortunately, the package used for getting CPU temp ([systeminformation]
memory: true
disk: /disk/mount/path
cputemp: true
tempmin: 0 # optional, minimum cpu temp
tempmax: 100 # optional, maximum cpu temp
uptime: true
units: imperial # only used by cpu temp
refresh: 3000 # optional, in ms

View File

@@ -7,14 +7,13 @@ Learn more about [Authentik](https://github.com/goauthentik/authentik).
This widget reads the number of active users in the system, as well as logins for the last 24 hours.
You will need to generate an API token for an existing user. To do so follow these steps:
You will need to generate an API token for an existing user under `Admin Portal` > `Directory` > `Tokens & App passwords`.
Make sure to set Intent to "API Token".
1. Navigate to the Authentik Admin Portal
2. Expand Directory, the click Tokens & App passwords
3. Click the Create button
4. Fill out the dialog making sure to set Intent to API Token
5. Click the Create button on the dialog
6. Click the copy button on the far right of the newly created API Token
The account you made the API token for also needs the following **Assigned global permissions** in Authentik:
- authentik Core -> Can view User (Model: User)
- authentik Events -> Can view Event (Model: Event)
Allowed fields: `["users", "loginsLast24H", "failedLoginsLast24H"]`.

View File

@@ -7,10 +7,10 @@ Learn more about [Azure DevOps](https://azure.microsoft.com/en-us/products/devop
This widget has 2 functions:
1. Pipelines: checks if the relevant pipeline is running or not, and if not, reports the last status.\
1. Pipelines: checks if the relevant pipeline is running or not, and if not, reports the last status.<br>
Allowed fields: `["result", "status"]`.
2. Pull Requests: returns the amount of open PRs, the amount of the PRs you have open, and how many PRs that you open are marked as 'Approved' by at least 1 person and not yet completed.\
2. Pull Requests: returns the amount of open PRs, the amount of the PRs you have open, and how many PRs that you open are marked as 'Approved' by at least 1 person and not yet completed.<br>
Allowed fields: `["totalPrs", "myPrs", "approved"]`.
You will need to generate a personal access token for an existing user, see the [azure documentation](https://learn.microsoft.com/en-us/azure/devops/organizations/accounts/use-personal-access-tokens-to-authenticate?view=azure-devops&tabs=Windows#create-a-pat)

View File

@@ -0,0 +1,19 @@
---
title: Crowdsec
description: Crowdsec Widget Configuration
---
Learn more about [Crowdsec](https://crowdsec.net).
See the [crowdsec docs](https://docs.crowdsec.net/docs/local_api/intro/#machines) for information about registering a machine,
in most instances you can use the default credentials (`/etc/crowdsec/local_api_credentials.yaml`).
Allowed fields: `["alerts", "bans"]`.
```yaml
widget:
type: crowdsec
url: http://crowdsechostorip:port
username: localhost # machine_id in crowdsec
password: password
```

View File

@@ -11,22 +11,27 @@ An optional 'volume' parameter can be supplied to specify which volume's free sp
Allowed fields: `["uptime", "volumeAvailable", "resources.cpu", "resources.mem"]`.
To access these system metrics you need to connect to the DiskStation with an account that is a member of the default `Administrators` group. That is because these metrics are requested from the API's `SYNO.Core.System` part that is only available to admin users. In order to keep the security impact as small as possible we can set the account in DSM up to limit the user's permissions inside the Synology system. In DSM 7.x, for instance, follow these steps:
To access these system metrics you need to connect to the DiskStation (`DSM`) with an account that is a member of the default `Administrators` group. That is because these metrics are requested from the API's `SYNO.Core.System` part that is only available to admin users. In order to keep the security impact as small as possible we can set the account in DSM up to limit the user's permissions inside the Synology system. In DSM 7.x, for instance, follow these steps:
1. Create a new user, i.e. `remote_stats`.
2. Set up a strong password for the new user
3. Under the `User Groups` tab of the user config dialogue check the box for `Administrators`.
4. On the `Permissions` tab check the top box for `No Access`, effectively prohibiting the user from accessing anything in the shared folders.
5. Under `Applications` check the box next to `Deny` in the header to explicitly prohibit login to all applications.
6. Now _only_ allow login to the `Download Station` application, either by
6. Now _only_ allow login to the `DSM` application, either by
- unchecking `Deny` in the respective row, or (if inheriting permission doesn't work because of other group settings)
- checking `Allow` for this app, or
- checking `By IP` for this app to limit the source of login attempts to one or more IP addresses/subnets.
7. When the `Preview` column shows `Allow` in the `Download Station` row, click `Save`.
7. When the `Preview` column shows `Allow` in the `DSM` row, click `Save`.
Now configure the widget with the correct login information and test it.
If you encounter issues during testing, make sure to uncheck the option for automatic blocking due to invalid logins under `Control Panel > Security > Protection`. If desired, this setting can be reactivated once the login is established working.
If you encounter issues during testing:
1. Make sure to uncheck the option for automatic blocking due to invalid logins under `Control Panel > Security > Protection`.
- If desired, this setting can be reactivated once the login is established working.
2. Login to your Synology DSM with the newly created account and accept terms and conditions.
3. Reattempt
```yaml
widget:

View File

@@ -16,4 +16,7 @@ widget:
key: apikeyapikeyapikeyapikeyapikey
enableBlocks: true # optional, defaults to false
enableNowPlaying: true # optional, defaults to true
enableUser: true # optional, defaults to false
showEpisodeNumber: true # optional, defaults to false
expandOneStreamToTwoRows: false # optional, defaults to true
```

View File

@@ -7,7 +7,7 @@ Learn more about [Gitea](https://gitea.com).
API token requires `notifications`, `repository` and `issue` permissions. See the [gitea documentation](https://docs.gitea.com/development/api-usage#generating-and-listing-api-tokens) for details on generating tokens.
Allowed fields: ["notifications", "issues", "pulls"]
Allowed fields: `["notifications", "issues", "pulls"]`.
```yaml
widget:

View File

@@ -17,6 +17,7 @@ widget:
url: http://glances.host.or.ip:port
username: user # optional if auth enabled in Glances
password: pass # optional if auth enabled in Glances
version: 4 # required only if running glances v4 or higher, defaults to 3
metric: cpu
diskUnits: bytes # optional, bytes (default) or bbytes. Only applies to disk
refreshInterval: 5000 # optional - in milliseconds, defaults to 1000 or more, depending on the metric

View File

@@ -16,4 +16,7 @@ widget:
key: apikeyapikeyapikeyapikeyapikey
enableBlocks: true # optional, defaults to false
enableNowPlaying: true # optional, defaults to true
enableUser: true # optional, defaults to false
showEpisodeNumber: true # optional, defaults to false
expandOneStreamToTwoRows: false # optional, defaults to true
```

View File

@@ -3,7 +3,7 @@ title: Medusa
description: Medusa Widget Configuration
---
Learn more about [Medusa](https://github.com/medusajs/medusa).
Learn more about [Medusa](https://github.com/pymedusa/Medusa).
Allowed fields: `["wanted", "queued", "series"]`.

View File

@@ -0,0 +1,16 @@
---
title: NetAlertX
description: NetAlertX (formerly PiAlert) Widget Configuration
---
Learn more about [NetAlertX](https://github.com/jokob-sk/NetAlertX).
_Note that the project was renamed from PiAlert to NetAlertX._
Allowed fields: `["total", "connected", "new_devices", "down_alerts"]`.
```yaml
widget:
type: netalertx
url: http://ip:port
```

View File

@@ -9,7 +9,7 @@ This widget adds support for [Network UPS Tools](https://networkupstools.org/) v
The default ups name is `ups`. To configure more than one ups, you must create multiple peanut services.
Allowed fields: `["battery_charge", "ups_load", "ups_status"]`
Allowed fields: `["battery_charge", "ups_load", "ups_status"]`.
!!! note

View File

@@ -1,16 +0,0 @@
---
title: PiAlert
description: PiAlert Widget Configuration
---
Learn more about [PiAlert](https://github.com/jokob-sk/Pi.Alert).
Note that [pucherot/PiAlert](https://github.com/pucherot/Pi.Alert) has been abandoned and might not work properly.
Allowed fields: `["total", "connected", "new_devices", "down_alerts"]`.
```yaml
widget:
type: pialert
url: http://ip:port
```

View File

@@ -15,6 +15,7 @@ Note: by default the "blocked" and "blocked_percent" fields are merged e.g. "1,2
widget:
type: pihole
url: http://pi.hole.or.ip
version: 6 # required if running v6 or higher, defaults to 5
key: yourpiholeapikey # optional
```

View File

@@ -14,4 +14,7 @@ widget:
type: tautulli
url: http://tautulli.host.or.ip
key: apikeyapikeyapikeyapikeyapikey
enableUser: true # optional, defaults to false
showEpisodeNumber: true # optional, defaults to false
expandOneStreamToTwoRows: false # optional, defaults to true
```

View File

@@ -5,7 +5,7 @@ description: Prometheus Widget Configuration
Learn more about [Prometheus](https://github.com/prometheus/prometheus).
Allowed fields: `["targets_up", "targets_down", "targets_total"]`
Allowed fields: `["targets_up", "targets_down", "targets_total"]`.
```yaml
widget:

View File

@@ -5,7 +5,7 @@ description: Pterodactyl Widget Configuration
Learn more about [Pterodactyl](https://github.com/pterodactyl).
Allowed fields: `["nodes", "servers"]`
Allowed fields: `["nodes", "servers"]`.
```yaml
widget:

View File

@@ -16,4 +16,5 @@ Allowed fields: `["download", "upload", "ping"]`.
widget:
type: speedtest
url: http://speedtest.host.or.ip
bitratePrecision: 3 # optional, default is 0
```

View File

@@ -11,6 +11,8 @@ To create an API Key, follow [the official TrueNAS documentation](https://www.tr
A detailed pool listing is disabled by default, but can be enabled with the `enablePools` option.
To use the `enablePools` option with TrueNAS Core, the `nasType` parameter is required.
```yaml
widget:
type: truenas
@@ -19,4 +21,5 @@ widget:
password: pass # not required if using api key
key: yourtruenasapikey # not required if using username / password
enablePools: true # optional, defaults to false
nasType: scale # defaults to scale, must be set to 'core' if using enablePools with TrueNAS Core
```

View File

@@ -5,7 +5,7 @@ description: Uptime Kuma Widget Configuration
Learn more about [Uptime Kuma](https://github.com/louislam/uptime-kuma).
As Uptime Kuma does not yet have a full API the widget uses data from a single "status page". As such you will need a status page setup with a group of monitored sites, which is where you get the slug (without the `/status/` portion).
As Uptime Kuma does not yet have a full API the widget uses data from a single "status page". As such you will need a status page setup with a group of monitored sites, which is where you get the slug (the url without the `/status/` portion). E.g. if your status page is URL http://uptimekuma.host/status/statuspageslug, insert `slug: statuspageslug`.
Allowed fields: `["up", "down", "uptime", "incident"]`.

View File

@@ -44,6 +44,7 @@ nav:
- widgets/services/channelsdvrserver.md
- widgets/services/cloudflared.md
- widgets/services/coin-market-cap.md
- widgets/services/crowdsec.md
- widgets/services/customapi.md
- widgets/services/deluge.md
- widgets/services/diskstation.md
@@ -89,6 +90,7 @@ nav:
- widgets/services/mylar.md
- widgets/services/navidrome.md
- widgets/services/netdata.md
- widgets/services/netalertx.md
- widgets/services/nextcloud.md
- widgets/services/nextdns.md
- widgets/services/nginx-proxy-manager.md
@@ -99,12 +101,12 @@ nav:
- widgets/services/opendtu.md
- widgets/services/openmediavault.md
- widgets/services/opnsense.md
- widgets/services/openwrt.md
- widgets/services/overseerr.md
- widgets/services/paperlessngx.md
- widgets/services/peanut.md
- widgets/services/pfsense.md
- widgets/services/photoprism.md
- widgets/services/pialert.md
- widgets/services/pihole.md
- widgets/services/plantit.md
- widgets/services/plex-tautulli.md
@@ -129,6 +131,7 @@ nav:
- widgets/services/stash.md
- widgets/services/syncthing-relay-server.md
- widgets/services/tailscale.md
- widgets/services/tandoor.md
- widgets/services/tdarr.md
- widgets/services/traefik.md
- widgets/services/transmission.md
@@ -165,6 +168,7 @@ nav:
theme:
name: material
custom_dir: docs/overrides
language: en
palette:
- media: "(prefers-color-scheme)"

854
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -16,7 +16,7 @@
"classnames": "^2.5.1",
"compare-versions": "^6.1.0",
"dockerode": "^4.0.2",
"follow-redirects": "^1.15.5",
"follow-redirects": "^1.15.6",
"gamedig": "^4.3.1",
"i18next": "^21.10.0",
"js-yaml": "^4.1.0",
@@ -29,14 +29,14 @@
"ping": "^0.4.4",
"pretty-bytes": "^6.1.1",
"raw-body": "^2.5.2",
"react": "^18.2.0",
"react": "^18.3.1",
"react-dom": "^18.2.0",
"react-i18next": "^11.18.6",
"react-icons": "^4.12.0",
"recharts": "^2.12.2",
"recharts": "^2.12.6",
"rrule": "^2.8.1",
"swr": "^1.3.0",
"systeminformation": "^5.22.0",
"systeminformation": "^5.22.7",
"tough-cookie": "^4.1.3",
"urbackup-server-api": "^0.8.9",
"winston": "^3.11.0",
@@ -47,18 +47,18 @@
"autoprefixer": "^10.4.17",
"eslint": "^8.57.0",
"eslint-config-airbnb": "^19.0.4",
"eslint-config-next": "^12.3.4",
"eslint-config-next": "^14.2.3",
"eslint-config-prettier": "^9.1.0",
"eslint-plugin-import": "^2.29.1",
"eslint-plugin-jsx-a11y": "^6.8.0",
"eslint-plugin-prettier": "^4.2.1",
"eslint-plugin-react": "^7.33.2",
"eslint-plugin-react": "^7.34.1",
"eslint-plugin-react-hooks": "^4.6.0",
"postcss": "^8.4.35",
"postcss": "^8.4.38",
"prettier": "^3.2.5",
"tailwind-scrollbar": "^3.0.5",
"tailwindcss": "^3.4.1",
"typescript": "^4.9.5"
"tailwindcss": "^3.4.3",
"typescript": "^5.4.5"
},
"optionalDependencies": {
"osx-temperature-sensor": "^1.0.8"

786
pnpm-lock.yaml generated

File diff suppressed because it is too large Load Diff

View File

@@ -277,7 +277,7 @@
"approved": "Goedgekeur",
"available": "Beskikbaar"
},
"pialert": {
"netalertx": {
"total": "Totaal",
"connected": "Gekoppel",
"new_devices": "Nuwe Toestelle",
@@ -872,5 +872,9 @@
"labels": "Etikette",
"users": "Gebruikers",
"totalValue": "Totale Waarde"
},
"crowdsec": {
"alerts": "Waarskuwings",
"bans": "Verbanne"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "في انتظار قطع الاتصال",
"connectionStatusDisconnecting": "جار قطع الاتصال",
"connectionStatusDisconnected": "غير متصل",
"connectionStatusConnected": "متصل",
"connectionStatusConnected": "Connected",
"uptime": "مدة التشغيل",
"maxDown": "أقصى حد للتنزيل",
"maxUp": "أقصى حد للتحميل",
@@ -277,11 +277,11 @@
"approved": "مصدق",
"available": "متاح"
},
"pialert": {
"netalertx": {
"total": "المجموع",
"connected": "متصل",
"new_devices": "أجهزة جديدة",
"down_alerts": "تنبيهات تعطل الخوادم"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "الاستعلامات",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "المستخدمون",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "تنبيهات",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusDisconnecting": "Disconnecting",
"connectionStatusDisconnected": "Disconnected",
"connectionStatusConnected": "Свързано",
"connectionStatusConnected": "Connected",
"uptime": "Uptime",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
@@ -277,10 +277,10 @@
"approved": "Одобрен",
"available": "Наличен"
},
"pialert": {
"netalertx": {
"total": "Общо",
"connected": "Свързано",
"new_devices": "Нови устройства",
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Потребители",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Предупреждения",
"bans": "Bans"
}
}

View File

@@ -14,7 +14,7 @@
"date": "{{value, date}}",
"relativeDate": "{{value, relativeDate}}",
"uptime": "{{value, uptime}}",
"months": "mo",
"months": "mes",
"days": "d",
"hours": "h",
"minutes": "m",
@@ -46,8 +46,8 @@
"used": "Utilitzat",
"load": "Càrrega",
"temp": "TEMP",
"max": "Max",
"uptime": "UP"
"max": "Màx.",
"uptime": "ACTIU"
},
"unifi": {
"users": "Usuaris",
@@ -61,65 +61,65 @@
"wlan_devices": "Dispositius WLAN",
"lan_users": "Usuaris LAN",
"wlan_users": "Usuaris WLAN",
"up": "UP",
"up": "ACTIU",
"down": "INACTIU",
"wait": "Si us plau, espereu",
"empty_data": "Subsystem status unknown"
"empty_data": "Estat del subsistema desconegut"
},
"docker": {
"rx": "Rebut",
"tx": "Transmès",
"mem": "MEM",
"cpu": "CPU",
"running": "Running",
"running": "En execució",
"offline": "Fora de línia",
"error": "Error",
"unknown": "Desconegut",
"healthy": "Healthy",
"starting": "Starting",
"unhealthy": "Unhealthy",
"not_found": "Not Found",
"exited": "Exited",
"partial": "Partial"
"healthy": "Saludable",
"starting": "Iniciant",
"unhealthy": "No saludable",
"not_found": "No trobat",
"exited": "Tancat",
"partial": "Parcial"
},
"ping": {
"error": "Error",
"ping": "Ping",
"down": "Down",
"up": "Up",
"not_available": "Not Available"
"ping": "Latència",
"down": "Inactiu",
"up": "Actiu",
"not_available": "No Disponible"
},
"siteMonitor": {
"http_status": "HTTP status",
"http_status": "Estat HTTP",
"error": "Error",
"response": "Response",
"down": "Down",
"up": "Up",
"not_available": "Not Available"
"response": "Resposta",
"down": "Inactiu",
"up": "Actiu",
"not_available": "No Disponible"
},
"emby": {
"playing": "Reproduint",
"transcoding": "Transcodificant",
"bitrate": "Taxa de bits",
"no_active": "Sense reproduccions actives",
"movies": "Movies",
"series": "Series",
"episodes": "Episodes",
"songs": "Songs"
"movies": "Pel·lícules",
"series": "Sèries",
"episodes": "Episodis",
"songs": "Cançons"
},
"esphome": {
"offline": "Fora de línia",
"offline_alt": "Fora de línia",
"online": "Online",
"online": "En línia",
"total": "Total",
"unknown": "Desconegut"
},
"evcc": {
"pv_power": "Production",
"battery_soc": "Battery",
"grid_power": "Grid",
"home_power": "Consumption",
"charge_power": "Charger",
"pv_power": "Producció",
"battery_soc": "Bateria",
"grid_power": "Xarxa",
"home_power": "Consum",
"charge_power": "Carregador",
"watt_hour": "Wh"
},
"flood": {
@@ -129,55 +129,55 @@
"seed": "Llavor"
},
"freshrss": {
"subscriptions": "Subscriptions",
"unread": "Unread"
"subscriptions": "Subcripcions",
"unread": "Sense llegir"
},
"fritzbox": {
"connectionStatus": "Estat",
"connectionStatusUnconfigured": "Unconfigured",
"connectionStatusConnecting": "Connecting",
"connectionStatusAuthenticating": "Authenticating",
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusDisconnecting": "Disconnecting",
"connectionStatusDisconnected": "Disconnected",
"connectionStatusConnected": "Connected",
"connectionStatusUnconfigured": "Sense configurar",
"connectionStatusConnecting": "Connectant",
"connectionStatusAuthenticating": "Autenticant",
"connectionStatusPendingDisconnect": "Desconnexió pendent",
"connectionStatusDisconnecting": "Desconnectant",
"connectionStatusDisconnected": "Desconnectat",
"connectionStatusConnected": "Connectat",
"uptime": "Temps actiu",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
"down": "Down",
"up": "Up",
"received": "Received",
"sent": "Sent",
"externalIPAddress": "Ext. IP"
"maxDown": "Màx. Descàrrega",
"maxUp": "Màx. Càrrega",
"down": "Inactiu",
"up": "Actiu",
"received": "Rebuts",
"sent": "Enviats",
"externalIPAddress": "IP ext."
},
"caddy": {
"upstreams": "Upstreams",
"requests": "Current requests",
"requests_failed": "Failed requests"
"requests": "Peticions actuals",
"requests_failed": "Peticions fallides"
},
"changedetectionio": {
"totalObserved": "Total d'observats",
"diffsDetected": "Diferències detectades"
},
"channelsdvrserver": {
"shows": "Shows",
"recordings": "Recordings",
"scheduled": "Scheduled",
"passes": "Passes"
"shows": "Sèries",
"recordings": "Gravacions",
"scheduled": "Programat",
"passes": "Aprovat"
},
"tautulli": {
"playing": "Reproduint",
"transcoding": "Transcodificant",
"bitrate": "Taxa de bits",
"no_active": "Sense reproduccions actives",
"plex_connection_error": "Check Plex Connection"
"plex_connection_error": "Comprova la connexió de Plex"
},
"omada": {
"connectedAp": "Connected APs",
"activeUser": "Active devices",
"alerts": "Alerts",
"connectedGateway": "Connected gateways",
"connectedSwitches": "Connected switches"
"connectedAp": "AP connectats",
"activeUser": "Dispositius actius",
"alerts": "Alertes",
"connectedGateway": "Pasarel·les connectades",
"connectedSwitches": "Conmutadors connectats"
},
"nzbget": {
"rate": "Taxa",
@@ -187,7 +187,7 @@
"plex": {
"streams": "Transmissions actives",
"albums": "Àlbums",
"movies": "Movies",
"movies": "Pel·lícules",
"tv": "Sèries"
},
"sabnzbd": {
@@ -213,12 +213,12 @@
"seed": "Llavor"
},
"qnap": {
"cpuUsage": "CPU Usage",
"memUsage": "MEM Usage",
"systemTempC": "System Temp",
"poolUsage": "Pool Usage",
"volumeUsage": "Volume Usage",
"invalid": "Invalid"
"cpuUsage": "Ús de CPU",
"memUsage": "Ús de Memòria",
"systemTempC": "Temp. Sistema",
"poolUsage": "Ús de les Reserves",
"volumeUsage": "Ús dels Volums",
"invalid": "No vàlid"
},
"deluge": {
"download": "Descarregar",
@@ -235,7 +235,7 @@
"sonarr": {
"wanted": "Volgut",
"queued": "En cua",
"series": "Series",
"series": "Sèries",
"queue": "Cua",
"unknown": "Desconegut"
},
@@ -243,14 +243,14 @@
"wanted": "Volgut",
"missing": "Faltant",
"queued": "En cua",
"movies": "Movies",
"movies": "Pel·lícules",
"queue": "Cua",
"unknown": "Desconegut"
},
"lidarr": {
"wanted": "Volgut",
"queued": "En cua",
"artists": "Artists"
"artists": "Artistes"
},
"readarr": {
"wanted": "Volgut",
@@ -277,17 +277,17 @@
"approved": "Aprovat",
"available": "Disponible"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
"connected": "Connectat",
"new_devices": "Nous dispositius",
"down_alerts": "Alertes de caigudes"
},
"pihole": {
"queries": "Consultes",
"blocked": "Bloquejat",
"blocked_percent": "Blocked %",
"gravity": "Gravity"
"blocked_percent": "Bloquejat %",
"gravity": "Gravetat"
},
"adguard": {
"queries": "Consultes",
@@ -298,37 +298,37 @@
"speedtest": {
"upload": "Pujada",
"download": "Descarregar",
"ping": "Ping"
"ping": "Latència"
},
"portainer": {
"running": "Running",
"running": "En execució",
"stopped": "Aturat",
"total": "Total"
},
"tailscale": {
"address": "Address",
"expires": "Expires",
"never": "Never",
"last_seen": "Last Seen",
"now": "Now",
"years": "{{number}}y",
"weeks": "{{number}}w",
"address": "Adreça",
"expires": "Caduca",
"never": "Mai",
"last_seen": "Vist per darrer cop",
"now": "Ara",
"years": "{{number}}a",
"weeks": "{{number}}set",
"days": "{{number}}d",
"hours": "{{number}}h",
"minutes": "{{number}}m",
"seconds": "{{number}}s",
"ago": "{{value}} Ago"
"ago": "Fa {{value}}"
},
"tdarr": {
"queue": "Cua",
"processed": "Processed",
"errored": "Errored",
"saved": "Saved"
"processed": "Processat",
"errored": "Error",
"saved": "Desat"
},
"traefik": {
"routers": "Encaminadors",
"services": "Serveis",
"middleware": "Middleware"
"middleware": "Intermediari"
},
"navidrome": {
"nothing_streaming": "Sense reproduccions actives",
@@ -360,7 +360,7 @@
},
"jackett": {
"configured": "Configurat",
"errored": "Errored"
"errored": "Error"
},
"strelaysrv": {
"numActiveSessions": "Sessions",
@@ -376,18 +376,18 @@
"medusa": {
"wanted": "Volgut",
"queued": "En cua",
"series": "Series"
"series": "Sèries"
},
"minecraft": {
"players": "Players",
"version": "Version",
"players": "Jugadors",
"version": "Versió",
"status": "Estat",
"up": "Online",
"up": "En línia",
"down": "Fora de línia"
},
"miniflux": {
"read": "Read",
"unread": "Unread"
"read": "Llegir",
"unread": "Sense llegir"
},
"authentik": {
"users": "Usuaris",
@@ -406,28 +406,28 @@
"wait": "Si us plau, espereu",
"temp": "TEMP",
"_temp": "Temp",
"warn": "Warn",
"uptime": "UP",
"warn": "Avís",
"uptime": "ACTIU",
"total": "Total",
"free": "Lliure",
"used": "Utilitzat",
"days": "d",
"hours": "h",
"crit": "Crit",
"read": "Read",
"write": "Write",
"crit": "Crític",
"read": "Llegir",
"write": "Escriure",
"gpu": "GPU",
"mem": "Mem",
"swap": "Swap"
"swap": "Intercanvi"
},
"quicklaunch": {
"bookmark": "Marcador",
"service": "Servei",
"search": "Search",
"custom": "Custom",
"visit": "Visit",
"search": "Cercar",
"custom": "Personalitzat",
"visit": "Visitar",
"url": "URL",
"searchsuggestion": "Suggestion"
"searchsuggestion": "Suggeriment"
},
"wmo": {
"0-day": "Assolellat",
@@ -492,21 +492,21 @@
"updates": "Actualitzacions",
"update_available": "Actualització disponible",
"up_to_date": "Actualitzat",
"child_bridges": "Child Bridges",
"child_bridges": "Ponts fills",
"child_bridges_status": "{{ok}}/{{total}}",
"up": "Up",
"up": "Actiu",
"pending": "Pendent",
"down": "Down"
"down": "Inactiu"
},
"healthchecks": {
"new": "New",
"up": "Up",
"grace": "In Grace Period",
"down": "Down",
"paused": "Paused",
"new": "Nou",
"up": "Actiu",
"grace": "En Període de gràcia",
"down": "Inactiu",
"paused": "En pausa",
"status": "Estat",
"last_ping": "Last Ping",
"never": "No pings yet"
"last_ping": "Últim ping",
"never": "Sense pings"
},
"watchtower": {
"containers_scanned": "Escanejat",
@@ -528,7 +528,7 @@
"truenas": {
"load": "Càrrega del sistema",
"uptime": "Temps actiu",
"alerts": "Alerts"
"alerts": "Alertes"
},
"pyload": {
"speed": "Velocitat",
@@ -544,12 +544,12 @@
"hdhomerun": {
"channels": "Canals",
"hd": "HD",
"tunerCount": "Tuners",
"channelNumber": "Channel",
"channelNetwork": "Network",
"signalStrength": "Strength",
"signalQuality": "Quality",
"symbolQuality": "Quality",
"tunerCount": "Sintonitzadors",
"channelNumber": "Canal",
"channelNetwork": "Xarxa",
"signalStrength": "Intensitat",
"signalQuality": "Qualitat",
"symbolQuality": "Qualitat",
"networkRate": "Taxa de bits",
"clientIP": "Client"
},
@@ -563,94 +563,94 @@
"total": "Total"
},
"peanut": {
"battery_charge": "Battery Charge",
"ups_load": "UPS Load",
"ups_status": "UPS Status",
"online": "Online",
"on_battery": "On Battery",
"low_battery": "Low Battery"
"battery_charge": "Càrrega de la bateria",
"ups_load": "Càrrega del SAI",
"ups_status": "Estat del SAI",
"online": "En línia",
"on_battery": "En Bateria",
"low_battery": "Bateria Baixa"
},
"nextdns": {
"wait": "Espereu si us plau",
"no_devices": "No Device Data Received"
"no_devices": "No s'han rebut dades del Dispositiu"
},
"mikrotik": {
"cpuLoad": "CPU Load",
"memoryUsed": "Memory Used",
"cpuLoad": "Càrrega de CPU",
"memoryUsed": "Memoria en ús",
"uptime": "Temps actiu",
"numberOfLeases": "Leases"
"numberOfLeases": "IPs assignades"
},
"xteve": {
"streams_all": "All Streams",
"streams_all": "Tots els fluxos",
"streams_active": "Transmissions actives",
"streams_xepg": "XEPG Channels"
"streams_xepg": "Canals XEPG"
},
"opendtu": {
"yieldDay": "Today",
"absolutePower": "Power",
"relativePower": "Power %",
"limit": "Limit"
"yieldDay": "Avui",
"absolutePower": "Potència",
"relativePower": "Potència %",
"limit": "Límit"
},
"opnsense": {
"cpu": "CPU Load",
"memory": "Active Memory",
"wanUpload": "WAN Upload",
"wanDownload": "WAN Download"
"cpu": "Càrrega de CPU",
"memory": "Memòria activa",
"wanUpload": "Pujada WAN",
"wanDownload": "Baixada WAN"
},
"moonraker": {
"printer_state": "Printer State",
"print_status": "Print Status",
"printer_state": "Estat de l'impressora",
"print_status": "Estat de l'impressió",
"print_progress": "Progress",
"layers": "Layers"
"layers": "Capes"
},
"octoprint": {
"printer_state": "Estat",
"temp_tool": "Tool temp",
"temp_bed": "Bed temp",
"job_completion": "Completion"
"temp_tool": "Temperatura capçal",
"temp_bed": "Temperatura llit",
"job_completion": "Finalització"
},
"cloudflared": {
"origin_ip": "Origin IP",
"origin_ip": "IP Origen",
"status": "Estat"
},
"pfsense": {
"load": "Load Avg",
"memory": "Mem Usage",
"wanStatus": "WAN Status",
"up": "Up",
"down": "Down",
"load": "Promig Càrrega",
"memory": "Ús Memòria",
"wanStatus": "Estat WAN",
"up": "Actiu",
"down": "Inactiu",
"temp": "Temp",
"disk": "Disk Usage",
"disk": "Ús Disc",
"wanIP": "WAN IP"
},
"proxmoxbackupserver": {
"datastore_usage": "Datastore",
"failed_tasks_24h": "Failed Tasks 24h",
"failed_tasks_24h": "Tasques fallides (24h)",
"cpu_usage": "CPU",
"memory_usage": "Memory"
"memory_usage": "Memòria"
},
"immich": {
"users": "Usuaris",
"photos": "Photos",
"photos": "Fotos",
"videos": "Vídeos",
"storage": "Storage"
"storage": "Emmagatzematge"
},
"uptimekuma": {
"up": "Sites Up",
"down": "Sites Down",
"up": "Actius",
"down": "Caiguts",
"uptime": "Temps actiu",
"incident": "Incident",
"incident": "Incidència",
"m": "m"
},
"atsumeru": {
"series": "Series",
"archives": "Archives",
"chapters": "Chapters",
"series": "Sèries",
"archives": "Arxius",
"chapters": "Capítols",
"categories": "Categories"
},
"komga": {
"libraries": "Libraries",
"series": "Series",
"libraries": "Biblioteques",
"series": "Sèries",
"books": "Llibres"
},
"diskstation": {
@@ -659,77 +659,77 @@
"volumeAvailable": "Disponible"
},
"mylar": {
"series": "Series",
"issues": "Issues",
"series": "Sèries",
"issues": "Problemes",
"wanted": "Volgut"
},
"photoprism": {
"albums": "Àlbums",
"photos": "Photos",
"photos": "Fotos",
"videos": "Vídeos",
"people": "People"
"people": "Gent"
},
"fileflows": {
"queue": "Cua",
"processing": "Processant",
"processed": "Processed",
"time": "Time"
"processed": "Processat",
"time": "Temps"
},
"grafana": {
"dashboards": "Dashboards",
"datasources": "Data Sources",
"totalalerts": "Total Alerts",
"alertstriggered": "Alerts Triggered"
"dashboards": "Taulells",
"datasources": "Origen de dades",
"totalalerts": "Alertes Totals",
"alertstriggered": "Alertes disparades"
},
"nextcloud": {
"cpuload": "Cpu Load",
"memoryusage": "Memory Usage",
"freespace": "Free Space",
"activeusers": "Active Users",
"numfiles": "Files",
"numshares": "Shared Items"
"cpuload": "Càrrega de CPU",
"memoryusage": "Ús Memòria",
"freespace": "Espai lliure",
"activeusers": "Usuaris actius",
"numfiles": "Fitxers",
"numshares": "Elements compartits"
},
"kopia": {
"status": "Estat",
"size": "Size",
"lastrun": "Last Run",
"nextrun": "Next Run",
"size": "Mida",
"lastrun": "Darrera execució",
"nextrun": "Següent execució",
"failed": "Error"
},
"unmanic": {
"active_workers": "Active Workers",
"total_workers": "Total Workers",
"records_total": "Queue Length"
"active_workers": "Treballadors actius",
"total_workers": "Treballadors Totals",
"records_total": "Llargada de la Cua"
},
"pterodactyl": {
"servers": "Servers",
"servers": "Servidors",
"nodes": "Nodes"
},
"prometheus": {
"targets_up": "Targets Up",
"targets_down": "Targets Down",
"targets_total": "Total Targets"
"targets_up": "Objectius actius",
"targets_down": "Objectius caiguts",
"targets_total": "Objectius Totals"
},
"gatus": {
"up": "Sites Up",
"down": "Sites Down",
"up": "Actius",
"down": "Caiguts",
"uptime": "Temps actiu"
},
"ghostfolio": {
"gross_percent_today": "Today",
"gross_percent_1y": "One year",
"gross_percent_max": "All time"
"gross_percent_today": "Avui",
"gross_percent_1y": "Un any",
"gross_percent_max": "Tot"
},
"audiobookshelf": {
"podcasts": "Podcasts",
"books": "Llibres",
"podcastsDuration": "Duration",
"booksDuration": "Duration"
"podcastsDuration": "Durada",
"booksDuration": "Durada"
},
"homeassistant": {
"people_home": "People Home",
"lights_on": "Lights On",
"switches_on": "Switches On"
"people_home": "Gent a casa",
"lights_on": "Llums enceses",
"switches_on": "Endolls activats"
},
"whatsupdocker": {
"monitoring": "Supervisió",
@@ -737,140 +737,144 @@
},
"calibreweb": {
"books": "Llibres",
"authors": "Authors",
"authors": "Autors",
"categories": "Categories",
"series": "Series"
"series": "Sèries"
},
"jdownloader": {
"downloadCount": "Cua",
"downloadBytesRemaining": "Restant",
"downloadTotalBytes": "Size",
"downloadTotalBytes": "Mida",
"downloadSpeed": "Velocitat"
},
"kavita": {
"seriesCount": "Series",
"totalFiles": "Files"
"seriesCount": "Sèries",
"totalFiles": "Fitxers"
},
"azuredevops": {
"result": "Result",
"result": "Resultat",
"status": "Estat",
"buildId": "Build ID",
"succeeded": "Succeeded",
"notStarted": "Not Started",
"buildId": "Id de compilació",
"succeeded": "Amb èxit",
"notStarted": "No Iniciat",
"failed": "Error",
"canceled": "Canceled",
"inProgress": "In Progress",
"totalPrs": "Total PRs",
"myPrs": "My PRs",
"canceled": "Cancel·lat",
"inProgress": "En curs",
"totalPrs": "RP Totals",
"myPrs": "Els meus RP",
"approved": "Aprovat"
},
"gamedig": {
"status": "Estat",
"online": "Online",
"online": "En línia",
"offline": "Fora de línia",
"name": "Name",
"map": "Map",
"currentPlayers": "Current players",
"players": "Players",
"maxPlayers": "Max players",
"name": "Nom",
"map": "Mapa",
"currentPlayers": "Jugadors actuals",
"players": "Jugadors",
"maxPlayers": "Màxim de jugadors",
"bots": "Bots",
"ping": "Ping"
"ping": "Latència"
},
"urbackup": {
"ok": "Ok",
"errored": "Errors",
"noRecent": "Out of Date",
"totalUsed": "Used Storage"
"noRecent": "Obsolet",
"totalUsed": "Emmagatzematge utilitzat"
},
"mealie": {
"recipes": "Recipes",
"recipes": "Receptes",
"users": "Usuaris",
"categories": "Categories",
"tags": "Tags"
"tags": "Etiquetes"
},
"openmediavault": {
"downloading": "Downloading",
"downloading": "Descarregant",
"total": "Total",
"running": "Running",
"running": "En execució",
"stopped": "Aturat",
"passed": "Aprobat",
"failed": "Error"
},
"openwrt": {
"uptime": "Temps actiu",
"cpuLoad": "CPU Load Avg (5m)",
"up": "Up",
"down": "Down",
"bytesTx": "Transmitted",
"bytesRx": "Received"
"cpuLoad": "Càrrega promig de CPU (5m)",
"up": "Actiu",
"down": "Inactiu",
"bytesTx": "Enviat",
"bytesRx": "Rebuts"
},
"uptimerobot": {
"status": "Estat",
"uptime": "Temps actiu",
"lastDown": "Last Downtime",
"downDuration": "Downtime Duration",
"sitesUp": "Sites Up",
"sitesDown": "Sites Down",
"paused": "Paused",
"notyetchecked": "Not Yet Checked",
"up": "Up",
"seemsdown": "Seems Down",
"down": "Down",
"lastDown": "Darrera Inactivitat",
"downDuration": "Duració d'Inactivitat",
"sitesUp": "Actius",
"sitesDown": "Caiguts",
"paused": "En pausa",
"notyetchecked": "Sense verificar",
"up": "Actiu",
"seemsdown": "Sembla caigut",
"down": "Inactiu",
"unknown": "Desconegut"
},
"calendar": {
"inCinemas": "In cinemas",
"physicalRelease": "Physical release",
"digitalRelease": "Digital release",
"noEventsToday": "No events for today!",
"noEventsFound": "No events found"
"inCinemas": "En cines",
"physicalRelease": "Estrena física",
"digitalRelease": "Estrena digital",
"noEventsToday": "Cap esdeveniment per avui!",
"noEventsFound": "No s'han trobat esdeveniments"
},
"romm": {
"platforms": "Platforms",
"totalRoms": "Total ROMs"
"platforms": "Plataformes",
"totalRoms": "ROMs totals"
},
"netdata": {
"warnings": "Warnings",
"criticals": "Criticals"
"warnings": "Avisos",
"criticals": "Crítics"
},
"plantit": {
"events": "Events",
"plants": "Plants",
"photos": "Photos",
"species": "Species"
"events": "Esdeveniments",
"plants": "Plantes",
"photos": "Fotos",
"species": "Espècies"
},
"gitea": {
"notifications": "Notifications",
"issues": "Issues",
"pulls": "Pull Requests"
"notifications": "Notificacions",
"issues": "Problemes",
"pulls": "Sol·licitud de Canvis"
},
"stash": {
"scenes": "Scenes",
"scenesPlayed": "Scenes Played",
"playCount": "Total Plays",
"playDuration": "Time Watched",
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"images": "Images",
"imageSize": "Images Size",
"galleries": "Galleries",
"performers": "Performers",
"studios": "Studios",
"movies": "Movies",
"tags": "Tags",
"scenes": "Escenes",
"scenesPlayed": "Escenes reproduïdes",
"playCount": "Total reproduccions",
"playDuration": "Temps visionat",
"sceneSize": "Tamany Escena",
"sceneDuration": "Duració Escenes",
"images": "Imatges",
"imageSize": "Mida Imatges",
"galleries": "Biblioteques",
"performers": "Intèrprets",
"studios": "Estudis",
"movies": "Pel·lícules",
"tags": "Etiquetes",
"oCount": "O Count"
},
"tandoor": {
"users": "Usuaris",
"recipes": "Recipes",
"keywords": "Keywords"
"recipes": "Receptes",
"keywords": "Paraules claus"
},
"homebox": {
"items": "Items",
"totalWithWarranty": "With Warranty",
"locations": "Locations",
"labels": "Labels",
"items": "Elements",
"totalWithWarranty": "Amb Garantia",
"locations": "Ubicacions",
"labels": "Etiquetes",
"users": "Usuaris",
"totalValue": "Total Value"
"totalValue": "Valor total"
},
"crowdsec": {
"alerts": "Alertes",
"bans": "Prohibicions"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Čeká na odpojení",
"connectionStatusDisconnecting": "Odpojování",
"connectionStatusDisconnected": "Odpojeno",
"connectionStatusConnected": "Připojeno",
"connectionStatusConnected": "Connected",
"uptime": "Doba spuštění",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
@@ -277,10 +277,10 @@
"approved": "Schváleno",
"available": "Dostupné"
},
"pialert": {
"netalertx": {
"total": "Celkem",
"connected": "Připojeno",
"new_devices": "Nová zařízení",
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Uživatelé",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Upozornění",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusDisconnecting": "Disconnecting",
"connectionStatusDisconnected": "Disconnected",
"connectionStatusConnected": "Forbundet",
"connectionStatusConnected": "Connected",
"uptime": "Oppetid",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
@@ -277,11 +277,11 @@
"approved": "Godkendt",
"available": "Tilgængelig"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Forbundet",
"new_devices": "Nye Enheder",
"down_alerts": "Nedadvarsler"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Forespørgsler",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Brugere",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Advarsler",
"bans": "Bans"
}
}

View File

@@ -142,8 +142,8 @@
"connectionStatusDisconnected": "Getrennt",
"connectionStatusConnected": "Verbunden",
"uptime": "Betriebszeit",
"maxDown": "Max. Empfang",
"maxUp": "Max. Senden",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
"down": "Empfangen",
"up": "Senden",
"received": "Empfangen",
@@ -277,7 +277,7 @@
"approved": "Genehmigt",
"available": "Verfügbar"
},
"pialert": {
"netalertx": {
"total": "Gesamt",
"connected": "Verbunden",
"new_devices": "Neue Geräte",
@@ -392,7 +392,7 @@
"authentik": {
"users": "Benutzer",
"loginsLast24H": "Anmeldungen (24 h)",
"failedLoginsLast24H": "Fehlgeschlagene Anmeldungen (24 h)"
"failedLoginsLast24H": "Fehlversuche (24 h)"
},
"proxmox": {
"mem": "RAM",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Benutzer",
"totalValue": "Gesamtwert"
},
"crowdsec": {
"alerts": "Warnungen",
"bans": "Banns"
}
}

View File

@@ -146,9 +146,9 @@
"maxUp": "Max. Up",
"down": "Down",
"up": "Up",
"received": "Received",
"sent": "Sent",
"externalIPAddress": "Ext. IP"
"received": "Ληφθέντα",
"sent": "Απεσταλμένα",
"externalIPAddress": "Εξωτερική IP"
},
"caddy": {
"upstreams": "Upstreams",
@@ -277,7 +277,7 @@
"approved": "Εγκρίθηκε",
"available": "Διαθέσιμο"
},
"pialert": {
"netalertx": {
"total": "Σύνολο",
"connected": "Connected",
"new_devices": "New Devices",
@@ -327,7 +327,7 @@
},
"traefik": {
"routers": "Routers",
"services": "Services",
"services": "Υπηρεσίες",
"middleware": "Middleware"
},
"navidrome": {
@@ -522,7 +522,7 @@
"tubearchivist": {
"downloads": "Ουρά",
"videos": "Videos",
"channels": "Channels",
"channels": "Κανάλια",
"playlists": "Playlists"
},
"truenas": {
@@ -542,14 +542,14 @@
"country": "Χώρα"
},
"hdhomerun": {
"channels": "Channels",
"channels": "Κανάλια",
"hd": "HD",
"tunerCount": "Tuners",
"channelNumber": "Channel",
"channelNetwork": "Network",
"channelNumber": "Κανάλι",
"channelNetwork": "Δίκτυο",
"signalStrength": "Strength",
"signalQuality": "Quality",
"symbolQuality": "Quality",
"signalQuality": "Ποιότητα",
"symbolQuality": "Ποιότητα",
"networkRate": "Ρυθμός bit",
"clientIP": "Client"
},
@@ -559,7 +559,7 @@
"unknown": "Άγνωστο"
},
"paperlessngx": {
"inbox": "Inbox",
"inbox": "Εισερχόμενα",
"total": "Σύνολο"
},
"peanut": {
@@ -567,8 +567,8 @@
"ups_load": "UPS Load",
"ups_status": "UPS Status",
"online": "Συνδεδεμένοι",
"on_battery": "On Battery",
"low_battery": "Low Battery"
"on_battery": "Σε μπαταρία",
"low_battery": "Χαμηλή μπαταρία"
},
"nextdns": {
"wait": "Παρακαλώ περιμένετε",
@@ -620,7 +620,7 @@
"up": "Up",
"down": "Down",
"temp": "Temp",
"disk": "Disk Usage",
"disk": "Χρήση δίσκου",
"wanIP": "WAN IP"
},
"proxmoxbackupserver": {
@@ -631,7 +631,7 @@
},
"immich": {
"users": "Χρήστες",
"photos": "Photos",
"photos": "Φωτογραφίες",
"videos": "Videos",
"storage": "Storage"
},
@@ -646,7 +646,7 @@
"series": "Σειρές",
"archives": "Archives",
"chapters": "Chapters",
"categories": "Categories"
"categories": "Κατηγορίες"
},
"komga": {
"libraries": "Libraries",
@@ -665,7 +665,7 @@
},
"photoprism": {
"albums": "Άλμπουμ",
"photos": "Photos",
"photos": "Φωτογραφίες",
"videos": "Videos",
"people": "People"
},
@@ -738,7 +738,7 @@
"calibreweb": {
"books": "Βιβλία",
"authors": "Authors",
"categories": "Categories",
"categories": "Κατηγορίες",
"series": "Σειρές"
},
"jdownloader": {
@@ -785,7 +785,7 @@
"mealie": {
"recipes": "Recipes",
"users": "Χρήστες",
"categories": "Categories",
"categories": "Κατηγορίες",
"tags": "Tags"
},
"openmediavault": {
@@ -802,7 +802,7 @@
"up": "Up",
"down": "Down",
"bytesTx": "Transmitted",
"bytesRx": "Received"
"bytesRx": "Ληφθέντα"
},
"uptimerobot": {
"status": "Κατάσταση",
@@ -836,7 +836,7 @@
"plantit": {
"events": "Events",
"plants": "Plants",
"photos": "Photos",
"photos": "Φωτογραφίες",
"species": "Species"
},
"gitea": {
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Χρήστες",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Ειδοποιήσεις",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Aprobita",
"available": "Havebla"
},
"pialert": {
"netalertx": {
"total": "Totalo",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Uzantoj",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -12,7 +12,7 @@
"number": "{{value, number}}",
"ms": "{{value, number}}",
"date": "{{value, date}}",
"relativeDate": "{{value, relativeDate}}",
"relativeDate": "{{valor, relativaFecha}}",
"uptime": "{{value, uptime}}",
"months": "me",
"days": "d",
@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Desconexión pendiente",
"connectionStatusDisconnecting": "Desconectando",
"connectionStatusDisconnected": "Desconectado",
"connectionStatusConnected": "Conectado",
"connectionStatusConnected": "Connected",
"uptime": "Tiempo activo",
"maxDown": "Descarga máxima",
"maxUp": "Subida máxima",
@@ -277,11 +277,11 @@
"approved": "Aprobado",
"available": "Disponible"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Conectado",
"new_devices": "Nuevos dispositivos",
"down_alerts": "Alertas de caídas"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Consultas",
@@ -427,7 +427,7 @@
"custom": "Personalizado",
"visit": "Visitar",
"url": "Enlace",
"searchsuggestion": "Suggestion"
"searchsuggestion": "Sugerencia"
},
"wmo": {
"0-day": "Soleado",
@@ -546,12 +546,12 @@
"hd": "Alta definición",
"tunerCount": "Tuners",
"channelNumber": "Canal",
"channelNetwork": "Network",
"signalStrength": "Strength",
"signalQuality": "Quality",
"symbolQuality": "Quality",
"channelNetwork": "Red",
"signalStrength": "Intensidad",
"signalQuality": "Calidad",
"symbolQuality": "Calidad",
"networkRate": "Tasa de bits",
"clientIP": "Client"
"clientIP": "Cliente"
},
"scrutiny": {
"passed": "Aprobado",
@@ -798,10 +798,10 @@
},
"openwrt": {
"uptime": "Tiempo activo",
"cpuLoad": "CPU Load Avg (5m)",
"cpuLoad": "Carga promedio del CPU (5m)",
"up": "Activo",
"down": "Inactivo",
"bytesTx": "Transmitted",
"bytesTx": "Transmitido",
"bytesRx": "Recibido"
},
"uptimerobot": {
@@ -826,21 +826,21 @@
"noEventsFound": "No se encontraron eventos"
},
"romm": {
"platforms": "Platforms",
"platforms": "Plataformas",
"totalRoms": "Total ROMs"
},
"netdata": {
"warnings": "Warnings",
"criticals": "Criticals"
"warnings": "Advertencias",
"criticals": "Críticos"
},
"plantit": {
"events": "Events",
"events": "Eventos",
"plants": "Plants",
"photos": "Fotos",
"species": "Species"
},
"gitea": {
"notifications": "Notifications",
"notifications": "Notificaciones",
"issues": "Números",
"pulls": "Pull Requests"
},
@@ -852,7 +852,7 @@
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"images": "Imágenes",
"imageSize": "Images Size",
"imageSize": "Tamaño de imagen",
"galleries": "Galerías",
"performers": "Performers",
"studios": "Studios",
@@ -869,8 +869,12 @@
"items": "Items",
"totalWithWarranty": "Con Garantía",
"locations": "Ubicaciones",
"labels": "Labels",
"labels": "Etiquetas",
"users": "Usuarios",
"totalValue": "Total Value"
"totalValue": "Valor total"
},
"crowdsec": {
"alerts": "Alertas",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Guztira",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Hyväksytty",
"available": "Saatavilla"
},
"pialert": {
"netalertx": {
"total": "Yhteensä",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -15,13 +15,13 @@
"relativeDate": "{{value, relativeDate}}",
"uptime": "{{value, uptime}}",
"months": "mo",
"days": "d",
"days": "j",
"hours": "h",
"minutes": "m",
"seconds": "s"
},
"widget": {
"missing_type": "Widget manquant: {{type}}",
"missing_type": "Type de widget manquant: {{type}}",
"api_error": "Erreur API",
"information": "Informations",
"status": "Statut",
@@ -40,7 +40,7 @@
},
"resources": {
"cpu": "CPU",
"mem": "Mém",
"mem": "MÉM",
"total": "Total",
"free": "Libre",
"used": "Utilisé",
@@ -69,7 +69,7 @@
"docker": {
"rx": "Rx",
"tx": "Tx",
"mem": "Mém",
"mem": "MÉM",
"cpu": "CPU",
"running": "Démarré",
"offline": "Hors ligne",
@@ -277,11 +277,11 @@
"approved": "Validé",
"available": "Disponible"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connecté",
"new_devices": "Nouvel Appareil",
"down_alerts": "Alertes"
"new_devices": "Nouveaux appareils",
"down_alerts": "Alertes d'arrêt"
},
"pihole": {
"queries": "Requêtes",
@@ -395,7 +395,7 @@
"failedLoginsLast24H": "Cnx. échouées (24h)"
},
"proxmox": {
"mem": "Mém",
"mem": "MÉM",
"cpu": "CPU",
"lxc": "LxC",
"vms": "VMs"
@@ -411,7 +411,7 @@
"total": "Total",
"free": "Libre",
"used": "Utilisé",
"days": "d",
"days": "j",
"hours": "h",
"crit": "Crit.",
"read": "Lu",
@@ -847,18 +847,18 @@
"stash": {
"scenes": "Scènes",
"scenesPlayed": "Scènes jouées",
"playCount": "Total Plays",
"playDuration": "Time Watched",
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"playCount": "Lectures Totales",
"playDuration": "Temps regardé",
"sceneSize": "Taille des scènes",
"sceneDuration": "Durée des scènes",
"images": "Images",
"imageSize": "Taille des images",
"galleries": "Galeries",
"performers": "Performers",
"performers": "Acteurs",
"studios": "Studios",
"movies": "Films",
"tags": "Étiquettes",
"oCount": "O Count"
"oCount": "0 Compte"
},
"tandoor": {
"users": "Utilisateurs",
@@ -871,6 +871,10 @@
"locations": "Emplacements",
"labels": "Étiquettes",
"users": "Utilisateurs",
"totalValue": "Total Value"
"totalValue": "Valeur Totale"
},
"crowdsec": {
"alerts": "Alertes",
"bans": "Exclusions"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "מאושר",
"available": "זמין"
},
"pialert": {
"netalertx": {
"total": "סה\"כ",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Odspajanje u tijeku",
"connectionStatusDisconnecting": "Odspajanje",
"connectionStatusDisconnected": "Odspojeno",
"connectionStatusConnected": "Povezano",
"connectionStatusConnected": "Connected",
"uptime": "Vrijeme rada",
"maxDown": "Maksimum preuzimanja",
"maxUp": "Maksimum prijenosa",
@@ -277,11 +277,11 @@
"approved": "Odobreno",
"available": "Dostupno"
},
"pialert": {
"netalertx": {
"total": "Ukupno",
"connected": "Povezano",
"new_devices": "Novi uređaji",
"down_alerts": "Obavijesti o nedostupnosti"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Upiti",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Korisnici",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Upozorenja",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Szétkapcsolás függőben",
"connectionStatusDisconnecting": "Kapcsolat bontása",
"connectionStatusDisconnected": "Kapcsolat bontva",
"connectionStatusConnected": "Csatlakoztatott",
"connectionStatusConnected": "Connected",
"uptime": "Üzemidő",
"maxDown": "Max let.",
"maxUp": "Max felt.",
@@ -277,11 +277,11 @@
"approved": "Engedélyezett",
"available": "Elérhető"
},
"pialert": {
"netalertx": {
"total": "Összes",
"connected": "Csatlakoztatott",
"new_devices": "Új Eszközök",
"down_alerts": "Leállási Figyelmeztetések"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Lekérdezések",
@@ -872,5 +872,9 @@
"labels": "Címkék",
"users": "Felhasználók",
"totalValue": "Teljes érték"
},
"crowdsec": {
"alerts": "Riasztások",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Menunggu Terputus",
"connectionStatusDisconnecting": "Sedan Memutus",
"connectionStatusDisconnected": "Terputus",
"connectionStatusConnected": "Tersambung",
"connectionStatusConnected": "Connected",
"uptime": "Waktu Aktif",
"maxDown": "Maks Unduh",
"maxUp": "Maks Unggah",
@@ -277,11 +277,11 @@
"approved": "Tersetujui",
"available": "Tersedia"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Tersambung",
"new_devices": "Perangkat Baru",
"down_alerts": "Alert Mati"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Kueri",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Pengguna",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Peringatan",
"bans": "Bans"
}
}

View File

@@ -15,8 +15,8 @@
"relativeDate": "{{value, relativeDate}}",
"uptime": "{{value, uptime}}",
"months": "mo",
"days": "d",
"hours": "h",
"days": "g",
"hours": "o",
"minutes": "m",
"seconds": "s"
},
@@ -277,7 +277,7 @@
"approved": "Approvati",
"available": "Disponibili"
},
"pialert": {
"netalertx": {
"total": "Totale",
"connected": "Connesso",
"new_devices": "Nuovi Dispositivi",
@@ -411,8 +411,8 @@
"total": "Totale",
"free": "Libero",
"used": "In utilizzo",
"days": "d",
"hours": "h",
"days": "g",
"hours": "o",
"crit": "Critico",
"read": "Letti",
"write": "Scrittura",
@@ -427,7 +427,7 @@
"custom": "Personalizzato",
"visit": "Visita",
"url": "URL",
"searchsuggestion": "Suggestion"
"searchsuggestion": "Suggerimenti"
},
"wmo": {
"0-day": "Soleggiato",
@@ -546,8 +546,8 @@
"hd": "HD",
"tunerCount": "Tuners",
"channelNumber": "Channel",
"channelNetwork": "Network",
"signalStrength": "Strength",
"channelNetwork": "Rete",
"signalStrength": "Intensità",
"signalQuality": "Quality",
"symbolQuality": "Quality",
"networkRate": "Bitrate",
@@ -830,32 +830,32 @@
"totalRoms": "Total ROMs"
},
"netdata": {
"warnings": "Warnings",
"warnings": "Avvisi",
"criticals": "Criticals"
},
"plantit": {
"events": "Events",
"plants": "Plants",
"photos": "Foto",
"species": "Species"
"species": "Specie"
},
"gitea": {
"notifications": "Notifications",
"notifications": "Notifiche",
"issues": "Problemi",
"pulls": "Pull Requests"
"pulls": "Richieste di Pull"
},
"stash": {
"scenes": "Scenes",
"scenesPlayed": "Scenes Played",
"playCount": "Total Plays",
"playDuration": "Time Watched",
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"images": "Images",
"scenes": "Scene",
"scenesPlayed": "Scene Riprodotte",
"playCount": "Totale Riproduzioni",
"playDuration": "Tempo Guardato",
"sceneSize": "Dimensione Delle Scene",
"sceneDuration": "Durata Delle Scene",
"images": "Immagini",
"imageSize": "Images Size",
"galleries": "Galleries",
"performers": "Performers",
"studios": "Studios",
"performers": "Esecutori",
"studios": "Studi",
"movies": "Film",
"tags": "Tag",
"oCount": "O Count"
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Utenti",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Allarmi",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "接続を切断する",
"connectionStatusDisconnecting": "接続を切断中",
"connectionStatusDisconnected": "切断されました",
"connectionStatusConnected": "接続済み",
"connectionStatusConnected": "Connected",
"uptime": "稼働時間",
"maxDown": "最大ダウン",
"maxUp": "最大アップ",
@@ -277,11 +277,11 @@
"approved": "承認済",
"available": "利用可"
},
"pialert": {
"netalertx": {
"total": "合計",
"connected": "接続済み",
"new_devices": "新しいデバイス",
"down_alerts": "ダウンアラート"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "クエリ",
@@ -872,5 +872,9 @@
"labels": "ラベル",
"users": "ユーザ",
"totalValue": "合計値"
},
"crowdsec": {
"alerts": "アラート",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "승인됨",
"available": "이용 가능"
},
"pialert": {
"netalertx": {
"total": "총합",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "사용자",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "경고",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Kopā",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Lietotāji",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Paziņojumi",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Lulus",
"available": "Sudah Ada"
},
"pialert": {
"netalertx": {
"total": "Jumlah",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Pengguna",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Goedgekeurd",
"available": "Beschikbaar"
},
"pialert": {
"netalertx": {
"total": "Totaal",
"connected": "Verbonden",
"new_devices": "Nieuwe Apparaten",
@@ -826,8 +826,8 @@
"noEventsFound": "Geen gebeurtenissen gevonden"
},
"romm": {
"platforms": "Platforms",
"totalRoms": "Total ROMs"
"platforms": "Platformen",
"totalRoms": "Totale ROM's"
},
"netdata": {
"warnings": "Waarschuwingen",
@@ -840,7 +840,7 @@
"species": "Soorten"
},
"gitea": {
"notifications": "Notifications",
"notifications": "Notificaties",
"issues": "Problemen",
"pulls": "Pull Requests"
},
@@ -863,7 +863,7 @@
"tandoor": {
"users": "Gebruikers",
"recipes": "Recepten",
"keywords": "Keywords"
"keywords": "Trefwoorden"
},
"homebox": {
"items": "Items",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Gebruikers",
"totalValue": "Totale waarde"
},
"crowdsec": {
"alerts": "Meldingen",
"bans": "Bans"
}
}

File diff suppressed because it is too large Load Diff

View File

@@ -40,7 +40,7 @@
},
"resources": {
"cpu": "Procesor",
"mem": "PAM",
"mem": "RAM",
"total": "Całkowite",
"free": "Wolne",
"used": "Użyte",
@@ -53,9 +53,9 @@
"users": "Użytkownicy",
"uptime": "Czas działania",
"days": "Dni",
"wan": "Sieć WAN",
"lan": "Sieć LAN",
"wlan": "Sieć WLAN",
"wan": "WAN",
"lan": "LAN",
"wlan": "WLAN",
"devices": "Urządzenia",
"lan_devices": "Urządzenia LAN",
"wlan_devices": "Urządzenia WLAN",
@@ -64,23 +64,23 @@
"up": "CZAS",
"down": "Pobieranie",
"wait": "Proszę czekać",
"empty_data": "Nieznany stan"
"empty_data": "Status podsystemu nieznany"
},
"docker": {
"rx": "Rx",
"tx": "Tx",
"mem": "PAM",
"mem": "RAM",
"cpu": "Procesor",
"running": "Działa",
"offline": "Nieosiągalny",
"error": "Błąd",
"unknown": "Nieznany",
"healthy": "Zdrowy",
"starting": "Rozpoczynanie",
"unhealthy": "Niezdrowe",
"starting": "Uruchamianie",
"unhealthy": "Niezdrowy",
"not_found": "Nie znaleziono",
"exited": "Zakończone",
"partial": "Częściowe"
"exited": "Zakończony",
"partial": "Częściowy"
},
"ping": {
"error": "Błąd",
@@ -137,18 +137,18 @@
"connectionStatusUnconfigured": "Nieskonfigurowane",
"connectionStatusConnecting": "Łączenie",
"connectionStatusAuthenticating": "Uwierzytelnianie",
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusPendingDisconnect": "Oczekujące rozłączenie",
"connectionStatusDisconnecting": "Rozłączanie",
"connectionStatusDisconnected": "Rozłączono",
"connectionStatusConnected": "Połączony",
"connectionStatusConnected": "Connected",
"uptime": "Czas działania",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
"maxDown": "Maks. Pobieranie",
"maxUp": "Maks. Wysyłanie",
"down": "Niedostępny",
"up": "Dostępny",
"received": "Odebrane",
"sent": "Wysłane",
"externalIPAddress": "Ext. IP"
"externalIPAddress": "Pub. IP"
},
"caddy": {
"upstreams": "Upstreams",
@@ -156,12 +156,12 @@
"requests_failed": "Nieudane zapytania"
},
"changedetectionio": {
"totalObserved": "Obserwowanych ogółem",
"diffsDetected": "Wykryto różnic"
"totalObserved": "Łącznie obserwowanych",
"diffsDetected": "Wykrytych różnic"
},
"channelsdvrserver": {
"shows": "Seriale",
"recordings": "Nagrywanie",
"recordings": "Nagrania",
"scheduled": "W kolejce",
"passes": "Przebiegi"
},
@@ -277,11 +277,11 @@
"approved": "Zaakceptowane",
"available": "Dostępne"
},
"pialert": {
"netalertx": {
"total": "Całkowite",
"connected": "Połączony",
"new_devices": "Nowe urządzenia",
"down_alerts": "Powiadomienia o niedostępności"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Zapytania",
@@ -395,7 +395,7 @@
"failedLoginsLast24H": "Nieudane logowania (24h)"
},
"proxmox": {
"mem": "PAM",
"mem": "RAM",
"cpu": "Procesor",
"lxc": "Kontenery LXC",
"vms": "Maszyn wirtualnych"
@@ -413,7 +413,7 @@
"used": "Użyte",
"days": "d",
"hours": "g",
"crit": "Crit",
"crit": "Krytyczyny",
"read": "Przeczytane",
"write": "Zapis",
"gpu": "Karta graficzna",
@@ -492,7 +492,7 @@
"updates": "Aktualizacje",
"update_available": "Dostępna aktualizacja",
"up_to_date": "Aktualny",
"child_bridges": "Child Bridges",
"child_bridges": "Mostki podrzędne",
"child_bridges_status": "{{ok}}/{{total}}",
"up": "Dostępny",
"pending": "Oczekiwane",
@@ -501,12 +501,12 @@
"healthchecks": {
"new": "Nowy",
"up": "Dostępny",
"grace": "In Grace Period",
"grace": "W okresie karencji",
"down": "Niedostępny",
"paused": "Zatrzymane",
"status": "Stan",
"last_ping": "Ostatni ping",
"never": "No pings yet"
"never": "Brak pingów"
},
"watchtower": {
"containers_scanned": "Zeskanowane",
@@ -544,7 +544,7 @@
"hdhomerun": {
"channels": "Kanały",
"hd": "HD",
"tunerCount": "Tuners",
"tunerCount": "Tunery",
"channelNumber": "Kanał",
"channelNetwork": "Sieć",
"signalStrength": "Siła",
@@ -563,7 +563,7 @@
"total": "Całkowite"
},
"peanut": {
"battery_charge": "Battery Charge",
"battery_charge": "Stan baterii",
"ups_load": "Obciążenie UPS",
"ups_status": "Status UPS",
"online": "Dostępny",
@@ -576,7 +576,7 @@
},
"mikrotik": {
"cpuLoad": "Obciążenie procesora",
"memoryUsed": "Zuyżyta pamięć",
"memoryUsed": "Zużyta pamięć",
"uptime": "Czas działania",
"numberOfLeases": "Dzierżawy"
},
@@ -587,8 +587,8 @@
},
"opendtu": {
"yieldDay": "Dzisiaj",
"absolutePower": "Power",
"relativePower": "Power %",
"absolutePower": "Zasilanie",
"relativePower": "Moc %",
"limit": "Limit"
},
"opnsense": {
@@ -605,16 +605,16 @@
},
"octoprint": {
"printer_state": "Stan",
"temp_tool": "Tool temp",
"temp_bed": "Bed temp",
"temp_tool": "Temperatura narzędzia",
"temp_bed": "Temp. łóżka",
"job_completion": "Ukończono"
},
"cloudflared": {
"origin_ip": "Origin IP",
"origin_ip": "IP Źródła",
"status": "Stan"
},
"pfsense": {
"load": "Load Avg",
"load": "Śr. Obciążenie",
"memory": "Użycie pamięci",
"wanStatus": "Status WAN",
"up": "Dostępny",
@@ -624,8 +624,8 @@
"wanIP": "WAN IP"
},
"proxmoxbackupserver": {
"datastore_usage": "Datastore",
"failed_tasks_24h": "Failed Tasks 24h",
"datastore_usage": "Magazyn danych",
"failed_tasks_24h": "Nieudane zadania 24h",
"cpu_usage": "Procesor",
"memory_usage": "Pamięć"
},
@@ -679,7 +679,7 @@
"dashboards": "Panel główny",
"datasources": "Źródła danych",
"totalalerts": "Wszystkie alerty",
"alertstriggered": "Alerts Triggered"
"alertstriggered": "Wywołane alerty"
},
"nextcloud": {
"cpuload": "Obciążenie CPU",
@@ -687,7 +687,7 @@
"freespace": "Wolna przestrzeń",
"activeusers": "Aktywni użytkownicy",
"numfiles": "Pliki",
"numshares": "Shared Items"
"numshares": "Udostępnione elementy"
},
"kopia": {
"status": "Stan",
@@ -698,7 +698,7 @@
},
"unmanic": {
"active_workers": "Aktywni pracownicy",
"total_workers": "Total Workers",
"total_workers": "Wszyscy pracownicy",
"records_total": "Długość kolejki"
},
"pterodactyl": {
@@ -706,9 +706,9 @@
"nodes": "Węzły"
},
"prometheus": {
"targets_up": "Targets Up",
"targets_down": "Targets Down",
"targets_total": "Total Targets"
"targets_up": "Cele włączone",
"targets_down": "Cele wyłączone",
"targets_total": "Wszystkich Celi"
},
"gatus": {
"up": "Działające",
@@ -727,9 +727,9 @@
"booksDuration": "Czas trwania"
},
"homeassistant": {
"people_home": "People Home",
"lights_on": "Lights On",
"switches_on": "Switches On"
"people_home": "Dom ludzi",
"lights_on": "Światła włączone",
"switches_on": "Przełączniki włączone"
},
"whatsupdocker": {
"monitoring": "Monitoring",
@@ -756,12 +756,12 @@
"status": "Stan",
"buildId": "ID kompilacji",
"succeeded": "Ukończono",
"notStarted": "Not Started",
"notStarted": "Nierozpoczęte",
"failed": "Niepowodzenie",
"canceled": "Anulowano",
"inProgress": "W trakcie",
"totalPrs": "Total PRs",
"myPrs": "My PRs",
"totalPrs": "Łącznie PRs",
"myPrs": "Moje PRs",
"approved": "Zaakceptowane"
},
"gamedig": {
@@ -770,7 +770,7 @@
"offline": "Nieosiągalny",
"name": "Nazwa",
"map": "Mapa",
"currentPlayers": "Current players",
"currentPlayers": "Gracze online",
"players": "Gracze",
"maxPlayers": "Maksymalna ilość graczy",
"bots": "Boty",
@@ -783,7 +783,7 @@
"totalUsed": "Użyta pamięć"
},
"mealie": {
"recipes": "Recipes",
"recipes": "Przepisy",
"users": "Użytkownicy",
"categories": "Kategorie",
"tags": "Tagi"
@@ -798,79 +798,83 @@
},
"openwrt": {
"uptime": "Czas działania",
"cpuLoad": "CPU Load Avg (5m)",
"cpuLoad": "Śr. obciążenie CPU (5m)",
"up": "Dostępny",
"down": "Niedostępny",
"bytesTx": "Transmitted",
"bytesTx": "Przesłane",
"bytesRx": "Odebrane"
},
"uptimerobot": {
"status": "Stan",
"uptime": "Czas działania",
"lastDown": "Last Downtime",
"downDuration": "Downtime Duration",
"lastDown": "Ostatni downtime",
"downDuration": "Długość downtime'u",
"sitesUp": "Działające",
"sitesDown": "Niedziałające",
"paused": "Zatrzymane",
"notyetchecked": "Not Yet Checked",
"notyetchecked": "Nie sprawdzono",
"up": "Dostępny",
"seemsdown": "Seems Down",
"seemsdown": "Możliwe, że wyłączony",
"down": "Niedostępny",
"unknown": "Nieznany"
},
"calendar": {
"inCinemas": "W kinach",
"physicalRelease": "Wydanie fizyczne",
"digitalRelease": "Digital release",
"noEventsToday": "No events for today!",
"noEventsFound": "No events found"
"digitalRelease": "Wydanie cyfrowe",
"noEventsToday": "Brak wydarzeń na dziś!",
"noEventsFound": "Nie znaleziono wydarzeń"
},
"romm": {
"platforms": "Platformy",
"totalRoms": "Total ROMs"
"totalRoms": "Łącznie ROM"
},
"netdata": {
"warnings": "Ostrzeżenia",
"criticals": "Criticals"
"criticals": "Krytyczny"
},
"plantit": {
"events": "Events",
"plants": "Plants",
"events": "Wydarzenia",
"plants": "Rośliny",
"photos": "Zdjęcia",
"species": "Species"
"species": "Gatunki"
},
"gitea": {
"notifications": "Notifications",
"notifications": "Powiadomienia",
"issues": "Zgłoszenia",
"pulls": "Pull Requests"
"pulls": "Żądania Pull"
},
"stash": {
"scenes": "Scenes",
"scenesPlayed": "Scenes Played",
"playCount": "Total Plays",
"playDuration": "Time Watched",
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"images": "Images",
"imageSize": "Images Size",
"galleries": "Galleries",
"performers": "Performers",
"studios": "Studios",
"scenes": "Sceny",
"scenesPlayed": "Odgrane sceny",
"playCount": "Łącznie odtworzone",
"playDuration": "Łączny czas oglądania",
"sceneSize": "Rozmiar scen",
"sceneDuration": "Czas trwania scen",
"images": "Obrazy",
"imageSize": "Rozmiar obrazów",
"galleries": "Galerie",
"performers": "Artyści",
"studios": "Studia",
"movies": "Filmy",
"tags": "Tagi",
"oCount": "O Count"
"oCount": "O Licznik"
},
"tandoor": {
"users": "Użytkownicy",
"recipes": "Recipes",
"keywords": "Keywords"
"recipes": "Przepisy",
"keywords": "Słowa kluczowe"
},
"homebox": {
"items": "Items",
"totalWithWarranty": "With Warranty",
"locations": "Locations",
"labels": "Labels",
"items": "Elementy",
"totalWithWarranty": "Z gwarancją",
"locations": "Lokalizacje",
"labels": "Etykiety",
"users": "Użytkownicy",
"totalValue": "Total Value"
"totalValue": "Wartość całkowita"
},
"crowdsec": {
"alerts": "Alarmy",
"bans": "Bany"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Desconexão pendente",
"connectionStatusDisconnecting": "Desconectando",
"connectionStatusDisconnected": "Desconectado",
"connectionStatusConnected": "Conectado",
"connectionStatusConnected": "Connected",
"uptime": "Ligado",
"maxDown": "Máx. de Descarga",
"maxUp": "Max. de Envio",
@@ -277,11 +277,11 @@
"approved": "Aprovada",
"available": "Disponível"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Conectado",
"new_devices": "Novos dispositivos",
"down_alerts": "Alertas de Baixo"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Consultas",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Utilizadores",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alertas",
"bans": "Bans"
}
}

View File

@@ -14,7 +14,7 @@
"date": "{{value, date}}",
"relativeDate": "{{value, relativeDate}}",
"uptime": "{{value, uptime}}",
"months": "mo",
"months": "M",
"days": "d",
"hours": "h",
"minutes": "m",
@@ -22,11 +22,11 @@
},
"widget": {
"missing_type": "Tipo de Widget ausente: {{type}}",
"api_error": "Erro da API",
"api_error": "Erros de API",
"information": "Informação",
"status": "Estado",
"url": "Endereço URL",
"raw_error": "Erro",
"raw_error": "Erro Raw",
"response_data": "Dados da Resposta"
},
"weather": {
@@ -85,23 +85,23 @@
"ping": {
"error": "Erro",
"ping": "Tempo de resposta",
"down": "Down",
"up": "Up",
"not_available": "Not Available"
"down": "Inativo",
"up": "Ativo",
"not_available": "Não Disponível"
},
"siteMonitor": {
"http_status": "HTTP status",
"http_status": "Estado HTTP",
"error": "Erro",
"response": "Response",
"down": "Down",
"up": "Up",
"not_available": "Not Available"
"response": "Resposta",
"down": "Inativo",
"up": "Ativo",
"not_available": "Não Disponível"
},
"emby": {
"playing": "A reproduzir",
"transcoding": "Transcodificação",
"bitrate": "Taxa de bits",
"no_active": "Sem streams ativas",
"no_active": "Sem Streams Ativos",
"movies": "Filmes",
"series": "Séries",
"episodes": "Episódios",
@@ -110,7 +110,7 @@
"esphome": {
"offline": "Desligado",
"offline_alt": "Desligado",
"online": "Online",
"online": "Disponível",
"total": "Total",
"unknown": "Desconhecido"
},
@@ -134,24 +134,24 @@
},
"fritzbox": {
"connectionStatus": "Estado",
"connectionStatusUnconfigured": "Unconfigured",
"connectionStatusConnecting": "Connecting",
"connectionStatusAuthenticating": "Authenticating",
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusDisconnecting": "Disconnecting",
"connectionStatusDisconnected": "Disconnected",
"connectionStatusConnected": "Connected",
"connectionStatusUnconfigured": "Não configurado",
"connectionStatusConnecting": "Conectando",
"connectionStatusAuthenticating": "Autenticando",
"connectionStatusPendingDisconnect": "Desconexão Pendente",
"connectionStatusDisconnecting": "Desconectando",
"connectionStatusDisconnected": "Desconectado",
"connectionStatusConnected": "Conectado",
"uptime": "Ligado",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
"down": "Down",
"up": "Up",
"received": "Received",
"sent": "Sent",
"externalIPAddress": "Ext. IP"
"down": "Inativo",
"up": "Ativo",
"received": "Recebido",
"sent": "Enviado",
"externalIPAddress": "IP Externo"
},
"caddy": {
"upstreams": "Upstreams",
"upstreams": "Streams de Envio",
"requests": "Solicitações atuais",
"requests_failed": "Solicitações com falha"
},
@@ -160,24 +160,24 @@
"diffsDetected": "Diferenças Detetadas"
},
"channelsdvrserver": {
"shows": "Shows",
"shows": "Programas",
"recordings": "Gravações",
"scheduled": "Scheduled",
"scheduled": "Agendado",
"passes": "Passes"
},
"tautulli": {
"playing": "A reproduzir",
"transcoding": "Transcodificação",
"bitrate": "Taxa de bits",
"no_active": "Sem streams ativas",
"plex_connection_error": "Check Plex Connection"
"no_active": "Sem Streams Ativos",
"plex_connection_error": "Verifique a conexão do Plex"
},
"omada": {
"connectedAp": "APs Ligados",
"activeUser": "Dispositivos activos",
"activeUser": "Dispositivos ativos",
"alerts": "Alertas",
"connectedGateway": "Gateways ligados",
"connectedSwitches": "Switches ligados"
"connectedGateway": "Gateways conectados",
"connectedSwitches": "Switches conectados"
},
"nzbget": {
"rate": "Taxa",
@@ -186,7 +186,7 @@
},
"plex": {
"streams": "Streams Ativas",
"albums": "Albums",
"albums": "Álbuns",
"movies": "Filmes",
"tv": "Series de TV"
},
@@ -213,12 +213,12 @@
"seed": "Semente"
},
"qnap": {
"cpuUsage": "CPU Usage",
"memUsage": "MEM Usage",
"systemTempC": "System Temp",
"cpuUsage": "Uso de CPU",
"memUsage": "Uso de Memória",
"systemTempC": "Temp. do Sistema",
"poolUsage": "Pool Usage",
"volumeUsage": "Volume Usage",
"invalid": "Invalid"
"volumeUsage": "Uso do volume",
"invalid": "Inválido"
},
"deluge": {
"download": "Descarregar",
@@ -250,7 +250,7 @@
"lidarr": {
"wanted": "Desejada",
"queued": "Em fila",
"artists": "Artists"
"artists": "Artistas"
},
"readarr": {
"wanted": "Desejada",
@@ -277,16 +277,16 @@
"approved": "Aprovada",
"available": "Disponível"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
"connected": "Conectado",
"new_devices": "Novos dispositivos",
"down_alerts": "Alertas de Inatividade"
},
"pihole": {
"queries": "Consultas",
"blocked": "Bloqueado",
"blocked_percent": "Blocked %",
"blocked_percent": "Bloqueado %",
"gravity": "Gravidade"
},
"adguard": {
@@ -306,11 +306,11 @@
"total": "Total"
},
"tailscale": {
"address": "Address",
"expires": "Expires",
"never": "Never",
"last_seen": "Last Seen",
"now": "Now",
"address": "Endereço",
"expires": "Expira em",
"never": "Nunca",
"last_seen": "Visto por último",
"now": "Agora",
"years": "{{number}}y",
"weeks": "{{number}}w",
"days": "{{number}}d",
@@ -331,8 +331,8 @@
"middleware": "Middleware"
},
"navidrome": {
"nothing_streaming": "Sem streams ativas",
"please_wait": "Por favor aguarde"
"nothing_streaming": "Sem Streams Ativos",
"please_wait": "Por favor, aguarde"
},
"npm": {
"enabled": "Ativo",
@@ -382,7 +382,7 @@
"players": "Reprodutores",
"version": "Versão",
"status": "Estado",
"up": "Online",
"up": "Disponível",
"down": "Desligado"
},
"miniflux": {
@@ -405,20 +405,20 @@
"load": "Carga",
"wait": "Por favor aguarde",
"temp": "TEMP",
"_temp": "Temp",
"warn": "Warn",
"_temp": "Temperatura",
"warn": "Aviso",
"uptime": "CIMA",
"total": "Total",
"free": "Livre",
"used": "Utilizado",
"days": "d",
"hours": "h",
"crit": "Crit",
"crit": "Crítico",
"read": "Lido",
"write": "Write",
"write": "Escrita",
"gpu": "GPU",
"mem": "Mem",
"swap": "Swap"
"mem": "Memória",
"swap": "Temporário"
},
"quicklaunch": {
"bookmark": "Marcador",
@@ -427,7 +427,7 @@
"custom": "Personalizado",
"visit": "Visitar",
"url": "Endereço URL",
"searchsuggestion": "Suggestion"
"searchsuggestion": "Sugestão"
},
"wmo": {
"0-day": "Solarengo",
@@ -494,15 +494,15 @@
"up_to_date": "Atualizado",
"child_bridges": "Pontes Filhas",
"child_bridges_status": "{{ok}}/{{total}}",
"up": "Up",
"up": "Ativo",
"pending": "Pendente",
"down": "Down"
"down": "Inativo"
},
"healthchecks": {
"new": "Novo",
"up": "Up",
"up": "Ativo",
"grace": "Em Período Gratuito",
"down": "Down",
"down": "Inativo",
"paused": "Pausado",
"status": "Estado",
"last_ping": "Ultimo Ping",
@@ -544,14 +544,14 @@
"hdhomerun": {
"channels": "Canais",
"hd": "HD",
"tunerCount": "Tuners",
"channelNumber": "Channel",
"channelNetwork": "Network",
"signalStrength": "Strength",
"signalQuality": "Quality",
"symbolQuality": "Quality",
"tunerCount": "Sintonizadores",
"channelNumber": "Canal",
"channelNetwork": "Rede",
"signalStrength": "Potência",
"signalQuality": "Qualidade",
"symbolQuality": "Qualidade",
"networkRate": "Taxa de bits",
"clientIP": "Client"
"clientIP": "Cliente"
},
"scrutiny": {
"passed": "Aprovado",
@@ -563,15 +563,15 @@
"total": "Total"
},
"peanut": {
"battery_charge": "Battery Charge",
"ups_load": "UPS Load",
"ups_status": "UPS Status",
"online": "Online",
"on_battery": "On Battery",
"low_battery": "Low Battery"
"battery_charge": "Carga da bateria",
"ups_load": "Carga do UPS",
"ups_status": "Estado UPS",
"online": "Disponível",
"on_battery": "Na bateria",
"low_battery": "Bateria Fraca"
},
"nextdns": {
"wait": "Por favor aguarde",
"wait": "Por favor, aguarde",
"no_devices": "Nenhum dado do dispositivo recebido"
},
"mikrotik": {
@@ -586,10 +586,10 @@
"streams_xepg": "Canais XEPG"
},
"opendtu": {
"yieldDay": "Today",
"absolutePower": "Power",
"relativePower": "Power %",
"limit": "Limit"
"yieldDay": "Hoje",
"absolutePower": "Energia",
"relativePower": "Energia %",
"limit": "Limite"
},
"opnsense": {
"cpu": "Carga do CPU",
@@ -614,14 +614,14 @@
"status": "Estado"
},
"pfsense": {
"load": "Load Avg",
"memory": "Mem Usage",
"wanStatus": "WAN Status",
"up": "Up",
"down": "Down",
"temp": "Temp",
"disk": "Disk Usage",
"wanIP": "WAN IP"
"load": "Carga Média",
"memory": "Uso de memória",
"wanStatus": "Estado WAN",
"up": "Ativo",
"down": "Inativo",
"temp": "Temperatura",
"disk": "Uso do disco",
"wanIP": "IP WAN"
},
"proxmoxbackupserver": {
"datastore_usage": "Armaz. de Dados",
@@ -644,9 +644,9 @@
},
"atsumeru": {
"series": "Séries",
"archives": "Archives",
"chapters": "Chapters",
"categories": "Categories"
"archives": "Arquivos",
"chapters": "Capítulos",
"categories": "Categorias"
},
"komga": {
"libraries": "Bibliotecas",
@@ -664,7 +664,7 @@
"wanted": "Desejada"
},
"photoprism": {
"albums": "Albums",
"albums": "Álbuns",
"photos": "Fotos",
"videos": "Vídeos",
"people": "Pessoa"
@@ -686,8 +686,8 @@
"memoryusage": "Memória Utilizada",
"freespace": "Espaço Livre",
"activeusers": "Utilizadores Ativos",
"numfiles": "Files",
"numshares": "Shared Items"
"numfiles": "Arquivos",
"numshares": "Itens compartilhados"
},
"kopia": {
"status": "Estado",
@@ -698,7 +698,7 @@
},
"unmanic": {
"active_workers": "Workers Ativos",
"total_workers": "Total Workers",
"total_workers": "Total de trabalhadores",
"records_total": "Comprimento da Fila"
},
"pterodactyl": {
@@ -716,7 +716,7 @@
"uptime": "Ligado"
},
"ghostfolio": {
"gross_percent_today": "Today",
"gross_percent_today": "Hoje",
"gross_percent_1y": "Um ano",
"gross_percent_max": "Todo o tempo"
},
@@ -732,13 +732,13 @@
"switches_on": "Interruptores Ligados"
},
"whatsupdocker": {
"monitoring": "Monitoring",
"monitoring": "Monitorando",
"updates": "Atualizações"
},
"calibreweb": {
"books": "Livros",
"authors": "Authors",
"categories": "Categories",
"authors": "Autores",
"categories": "Categorias",
"series": "Séries"
},
"jdownloader": {
@@ -749,47 +749,47 @@
},
"kavita": {
"seriesCount": "Séries",
"totalFiles": "Files"
"totalFiles": "Arquivos"
},
"azuredevops": {
"result": "Result",
"result": "Resultado",
"status": "Estado",
"buildId": "Build ID",
"succeeded": "Succeeded",
"notStarted": "Not Started",
"buildId": "ID Compilação",
"succeeded": "Bem-sucedido",
"notStarted": "Não iniciado",
"failed": "Falhou",
"canceled": "Canceled",
"inProgress": "In Progress",
"totalPrs": "Total PRs",
"myPrs": "My PRs",
"canceled": "Cancelado",
"inProgress": "Em Progresso",
"totalPrs": "Total de PRs",
"myPrs": "Minhas PRs",
"approved": "Aprovada"
},
"gamedig": {
"status": "Estado",
"online": "Online",
"online": "Disponível",
"offline": "Desligado",
"name": "Name",
"map": "Map",
"currentPlayers": "Current players",
"name": "Nome",
"map": "Mapa",
"currentPlayers": "Jogadores atuais",
"players": "Reprodutores",
"maxPlayers": "Max players",
"bots": "Bots",
"maxPlayers": "Número Máximo de Jogadores",
"bots": "Robôs",
"ping": "Tempo de resposta"
},
"urbackup": {
"ok": "Ok",
"errored": "Errors",
"noRecent": "Out of Date",
"totalUsed": "Used Storage"
"errored": "Erros",
"noRecent": "Desatualizado",
"totalUsed": "Armazanamento Utilizado"
},
"mealie": {
"recipes": "Recipes",
"recipes": "Receitas",
"users": "Utilizadores",
"categories": "Categories",
"tags": "Tags"
"categories": "Categorias",
"tags": "Marcadores"
},
"openmediavault": {
"downloading": "Downloading",
"downloading": "Baixando",
"total": "Total",
"running": "A correr",
"stopped": "Parado",
@@ -798,79 +798,83 @@
},
"openwrt": {
"uptime": "Ligado",
"cpuLoad": "CPU Load Avg (5m)",
"up": "Up",
"down": "Down",
"bytesTx": "Transmitted",
"bytesRx": "Received"
"cpuLoad": "Carga da CPU média (5m)",
"up": "Ativo",
"down": "Inativo",
"bytesTx": "Transmitido",
"bytesRx": "Recebido"
},
"uptimerobot": {
"status": "Estado",
"uptime": "Ligado",
"lastDown": "Last Downtime",
"downDuration": "Downtime Duration",
"lastDown": "Última inatividade",
"downDuration": "Duração de inatividade",
"sitesUp": "Sites no Ar",
"sitesDown": "Sites Fora do Ar",
"paused": "Pausado",
"notyetchecked": "Not Yet Checked",
"up": "Up",
"seemsdown": "Seems Down",
"down": "Down",
"notyetchecked": "Não conferidos ainda",
"up": "Ativo",
"seemsdown": "Parece Desconectado",
"down": "Inativo",
"unknown": "Desconhecido"
},
"calendar": {
"inCinemas": "In cinemas",
"physicalRelease": "Physical release",
"digitalRelease": "Digital release",
"noEventsToday": "No events for today!",
"noEventsFound": "No events found"
"inCinemas": "Nos cinemas",
"physicalRelease": "Versão física",
"digitalRelease": "Versão digital",
"noEventsToday": "Nenhum evento para hoje!",
"noEventsFound": "Nenhum evento encontrado"
},
"romm": {
"platforms": "Platforms",
"totalRoms": "Total ROMs"
"platforms": "Plataformas",
"totalRoms": "Total de ROMs"
},
"netdata": {
"warnings": "Warnings",
"criticals": "Criticals"
"warnings": "Alertas",
"criticals": "Críticos"
},
"plantit": {
"events": "Events",
"plants": "Plants",
"events": "Eventos",
"plants": "Plantas",
"photos": "Fotos",
"species": "Species"
"species": "Espécies"
},
"gitea": {
"notifications": "Notifications",
"notifications": "Notificações",
"issues": "Problemas",
"pulls": "Pull Requests"
"pulls": "Solicitações de Envio"
},
"stash": {
"scenes": "Scenes",
"scenesPlayed": "Scenes Played",
"playCount": "Total Plays",
"playDuration": "Time Watched",
"sceneSize": "Scenes Size",
"sceneDuration": "Scenes Duration",
"images": "Images",
"imageSize": "Images Size",
"galleries": "Galleries",
"performers": "Performers",
"studios": "Studios",
"scenes": "Cenas",
"scenesPlayed": "Cenas Reproduzidas",
"playCount": "Total de Reproduções",
"playDuration": "Tempo Assistido",
"sceneSize": "Tamanho das cenas",
"sceneDuration": "Duração das cenas",
"images": "Imagens",
"imageSize": "Tamanho da Imagem",
"galleries": "Galerias",
"performers": "Atores",
"studios": "Estúdios",
"movies": "Filmes",
"tags": "Tags",
"oCount": "O Count"
"tags": "Marcadores",
"oCount": "Contagem 0"
},
"tandoor": {
"users": "Utilizadores",
"recipes": "Recipes",
"keywords": "Keywords"
"recipes": "Receitas",
"keywords": "Palavras-chave"
},
"homebox": {
"items": "Items",
"totalWithWarranty": "With Warranty",
"locations": "Locations",
"labels": "Labels",
"items": "Itens",
"totalWithWarranty": "Com Garantia",
"locations": "Localização",
"labels": "Rótulos",
"users": "Utilizadores",
"totalValue": "Total Value"
"totalValue": "Valor Total"
},
"crowdsec": {
"alerts": "Alertas",
"bans": "Banimentos"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Aprobate",
"available": "Disponibile"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Utilizatori",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Ожидает отключения",
"connectionStatusDisconnecting": "Отключение",
"connectionStatusDisconnected": "Отключено",
"connectionStatusConnected": "Подключено",
"connectionStatusConnected": "Connected",
"uptime": "Время работы",
"maxDown": "Макс. Загрузка",
"maxUp": "Макс. Отдача",
@@ -277,11 +277,11 @@
"approved": "Одобрено",
"available": "Доступно"
},
"pialert": {
"netalertx": {
"total": "Всего",
"connected": "Подключено",
"new_devices": "Новые устройства",
"down_alerts": "Оповещение о недоступности"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Запросы",
@@ -872,5 +872,9 @@
"labels": "Ярлыки",
"users": "Пользователи",
"totalValue": "Общая стоимость"
},
"crowdsec": {
"alerts": "Предупреждения",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Čakám na odpojenie",
"connectionStatusDisconnecting": "Odpájanie",
"connectionStatusDisconnected": "Odpojené",
"connectionStatusConnected": "Pripojené",
"connectionStatusConnected": "Connected",
"uptime": "Prevádzka",
"maxDown": "Max. sťahovanie",
"maxUp": "Max. nahrávanie",
@@ -277,11 +277,11 @@
"approved": "Schválené",
"available": "Dostupné"
},
"pialert": {
"netalertx": {
"total": "Celkovo",
"connected": "Pripojené",
"new_devices": "Nové zariadenia",
"down_alerts": "Upozornenia o výpadkoch"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Dopyty",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Používatelia",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Upozornenia",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Čakanje na prekinitev",
"connectionStatusDisconnecting": "Prekinitev",
"connectionStatusDisconnected": "Prekinjeno",
"connectionStatusConnected": "Povezanih",
"connectionStatusConnected": "Povezan",
"uptime": "Čas delovanja",
"maxDown": "Maks. dol",
"maxUp": "Maks. gor",
@@ -277,11 +277,11 @@
"approved": "Odobreno",
"available": "Na voljo"
},
"pialert": {
"netalertx": {
"total": "Skupaj",
"connected": "Povezanih",
"new_devices": "Nove naprave",
"down_alerts": "Izključeno"
"connected": "Povezan",
"new_devices": "Nova naprave",
"down_alerts": "Alarmi nedelovanja"
},
"pihole": {
"queries": "Poizvedbe",
@@ -872,5 +872,9 @@
"labels": "Oznake",
"users": "Uporabniki",
"totalValue": "Skupna vrednost"
},
"crowdsec": {
"alerts": "Opozorila",
"bans": "Prepovedi"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -104,7 +104,7 @@
"no_active": "Inga aktiva strömmar",
"movies": "Movies",
"series": "Series",
"episodes": "Episodes",
"episodes": "Avsnitt",
"songs": "Songs"
},
"esphome": {
@@ -277,7 +277,7 @@
"approved": "Godkända",
"available": "Tillgänglig"
},
"pialert": {
"netalertx": {
"total": "Total",
"connected": "Connected",
"new_devices": "New Devices",
@@ -423,7 +423,7 @@
"quicklaunch": {
"bookmark": "Bookmark",
"service": "Service",
"search": "Search",
"search": "Sök",
"custom": "Custom",
"visit": "Visit",
"url": "URL",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Användare",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "ఆమోదించబడింది",
"available": "అందుబాటులో వున్నవి"
},
"pialert": {
"netalertx": {
"total": "మొత్తం",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "వినియోగదారులు",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Approved",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "ทั้งหมด",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "ผู้ใช้",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -21,7 +21,7 @@
"seconds": "s"
},
"widget": {
"missing_type": "Kayıp Araç Türü: {{type}}",
"missing_type": "Eksik Araç Türü: {{type}}",
"api_error": "API Hatası",
"information": "Bilgi",
"status": "Durum",
@@ -42,12 +42,12 @@
"cpu": "CPU",
"mem": "MEM",
"total": "Toplam",
"free": "Boşta",
"free": "Boş",
"used": "Kullanımda",
"load": "Yük",
"temp": "Geçici",
"temp": "Sıcaklık",
"max": "En Yüksek",
"uptime": "Çalışma Süresi"
"uptime": "Çalışıyor"
},
"unifi": {
"users": "Kullanıcılar",
@@ -61,7 +61,7 @@
"wlan_devices": "WLAN Aygıtları",
"lan_users": "LAN Kullanıcıları",
"wlan_users": "WLAN Kullanıcıları",
"up": "Çalışma Süresi",
"up": "Çalışıyor",
"down": "Aşağı",
"wait": "Lütfen bekleyin",
"empty_data": "Alt sistem durumu bilinmiyor"
@@ -71,15 +71,15 @@
"tx": "Giden Veri",
"mem": "MEM",
"cpu": "CPU",
"running": "Çalışan",
"running": "Çalışıyor",
"offline": "Çevrimdışı",
"error": "Hata",
"unknown": "Bilinmiyor",
"healthy": "Sağlık",
"healthy": "Sağlıklı",
"starting": "Başlatılıyor",
"unhealthy": "Sağlıksız",
"not_found": "Bulunamadı",
"exited": "Durduruldu",
"exited": "Kapandı",
"partial": "Parçalı"
},
"ping": {
@@ -123,8 +123,8 @@
"watt_hour": "Watt/Saat"
},
"flood": {
"download": "İndir",
"upload": "Yükle",
"download": "İndirme",
"upload": "Yükleme",
"leech": "Tüketici",
"seed": "Sağlayıcı"
},
@@ -197,18 +197,18 @@
},
"rutorrent": {
"active": "Aktif",
"upload": "Yükle",
"download": "İndir"
"upload": "Yükleme",
"download": "İndirme"
},
"transmission": {
"download": "İndir",
"upload": "Yükle",
"download": "İndirme",
"upload": "Yükleme",
"leech": "Tüketici",
"seed": "Sağlayıcı"
},
"qbittorrent": {
"download": "İndir",
"upload": "Yükle",
"download": "İndirme",
"upload": "Yükleme",
"leech": "Tüketici",
"seed": "Sağlayıcı"
},
@@ -221,87 +221,87 @@
"invalid": "Geçersiz"
},
"deluge": {
"download": "İndir",
"upload": "Yükle",
"download": "İndirme",
"upload": "Yükleme",
"leech": "Tüketici",
"seed": "Sağlayıcı"
},
"downloadstation": {
"download": "İndir",
"upload": "Yükle",
"download": "İndirme",
"upload": "Yükleme",
"leech": "Tüketici",
"seed": "Sağlayıcı"
},
"sonarr": {
"wanted": "Aranan",
"queued": "Kuyrukta",
"wanted": "İstendi",
"queued": "Sırada",
"series": "Diziler",
"queue": "Kuyruk",
"unknown": "Bilinmiyor"
},
"radarr": {
"wanted": "Aranan",
"missing": "Kayıp",
"queued": "Kuyrukta",
"wanted": "İstendi",
"missing": "Eksik",
"queued": "Sırada",
"movies": "Filmler",
"queue": "Kuyruk",
"unknown": "Bilinmiyor"
},
"lidarr": {
"wanted": "Aranan",
"queued": "Kuyrukta",
"wanted": "İstendi",
"queued": "Sırada",
"artists": "Sanatçılar"
},
"readarr": {
"wanted": "Aranan",
"queued": "Kuyrukta",
"wanted": "İstendi",
"queued": "Sırada",
"books": "Kitaplar"
},
"bazarr": {
"missingEpisodes": "Kayıp Bölümler",
"missingMovies": "Kayıp Filmler"
"missingEpisodes": "Eksik Bölümler",
"missingMovies": "Eksik Filmler"
},
"ombi": {
"pending": "Bekliyor",
"pending": "Bekleyen",
"approved": "Onaylı",
"available": "Kullanılabilir"
},
"jellyseerr": {
"pending": "Bekliyor",
"pending": "Bekleyen",
"approved": "Onaylı",
"available": "Kullanılabilir"
},
"overseerr": {
"pending": "Bekliyor",
"pending": "Bekleyen",
"processing": "İşleniyor",
"approved": "Onaylı",
"available": "Kullanılabilir"
},
"pialert": {
"netalertx": {
"total": "Toplam",
"connected": "Bağlandı",
"new_devices": "Yeni Cihazlar",
"down_alerts": "Düşme Uyarıları"
"down_alerts": "Hata Uyarıları"
},
"pihole": {
"queries": "Sorgular",
"blocked": "Engellenen",
"blocked_percent": "Engellenen %",
"gravity": "Yer Çekimi"
"gravity": "Gravity"
},
"adguard": {
"queries": "Sorgular",
"blocked": "Engellenen",
"filtered": "Filtrelenen",
"filtered": "Filtrelendi",
"latency": "Gecikme"
},
"speedtest": {
"upload": "Yükle",
"download": "İndir",
"upload": "Yükleme",
"download": "İndirme",
"ping": "Gecikme"
},
"portainer": {
"running": "Çalışan",
"running": "Çalışıyor",
"stopped": "Durduruldu",
"total": "Toplam"
},
@@ -353,10 +353,10 @@
},
"prowlarr": {
"enableIndexers": "Dizin Oluşturucular",
"numberOfGrabs": "Yakalama Sayısı",
"numberOfGrabs": "Yakalamalar",
"numberOfQueries": "Sorgular",
"numberOfFailGrabs": "Başarısız Yakalama Sayısı",
"numberOfFailQueries": "Başarısız Sorgu Sayısı"
"numberOfFailGrabs": "Başarısız Yakalamalar",
"numberOfFailQueries": "Başarısız Sorgular"
},
"jackett": {
"configured": "Yapılandırılmış",
@@ -374,8 +374,8 @@
"domain_count": "Etki Alanları"
},
"medusa": {
"wanted": "Aranan",
"queued": "Kuyrukta",
"wanted": "İstendi",
"queued": "Sırada",
"series": "Diziler"
},
"minecraft": {
@@ -386,7 +386,7 @@
"down": "Çevrimdışı"
},
"miniflux": {
"read": "Oku",
"read": "Okunan",
"unread": "Okunmamış"
},
"authentik": {
@@ -404,17 +404,17 @@
"cpu": "CPU",
"load": "Yük",
"wait": "Lütfen bekleyin",
"temp": "Geçici",
"temp": "Sıcaklık",
"_temp": "Sıcaklık",
"warn": "Uyarı",
"uptime": "Çalışma Süresi",
"uptime": "Çalışıyor",
"total": "Toplam",
"free": "Boşta",
"free": "Boş",
"used": "Kullanımda",
"days": "g",
"hours": "sa",
"crit": "Kritik",
"read": "Oku",
"read": "Okunan",
"write": "Yazma",
"gpu": "GPU",
"mem": "Hafıza",
@@ -495,7 +495,7 @@
"child_bridges": "Alt Köprüler",
"child_bridges_status": "{{ok}}/{{total}}",
"up": "Yükleme",
"pending": "Bekliyor",
"pending": "Bekleyen",
"down": "İndirme"
},
"healthchecks": {
@@ -503,7 +503,7 @@
"up": "Yükleme",
"grace": "Tolerans Döneminde",
"down": "İndirme",
"paused": "Durduruldu",
"paused": "Duraklatıldı",
"status": "Durum",
"last_ping": "Son Ping",
"never": "Henüz ping yok"
@@ -636,8 +636,8 @@
"storage": "Depo"
},
"uptimekuma": {
"up": "Sites Up",
"down": "Sites Down",
"up": "Siteler Çalışıyor",
"down": "Siteler Çalışmıyor",
"uptime": "Çalışma Süresi",
"incident": "Olay",
"m": "dk"
@@ -661,7 +661,7 @@
"mylar": {
"series": "Diziler",
"issues": "Sorunlar",
"wanted": "Aranan"
"wanted": "İstendi"
},
"photoprism": {
"albums": "Albümler",
@@ -706,13 +706,13 @@
"nodes": "Düğümler"
},
"prometheus": {
"targets_up": "Targets Up",
"targets_down": "Targets Down",
"targets_up": "Hedef Çalışıyor",
"targets_down": "Hedef Çalışmıyor",
"targets_total": "Toplam Hedef"
},
"gatus": {
"up": "Sites Up",
"down": "Sites Down",
"up": "Siteler Çalışıyor",
"down": "Siteler Çalışmıyor",
"uptime": "Çalışma Süresi"
},
"ghostfolio": {
@@ -727,7 +727,7 @@
"booksDuration": "Süre"
},
"homeassistant": {
"people_home": "People Home",
"people_home": "Evdeki İnsanlar",
"lights_on": "Işıklar Açık",
"switches_on": "Aç"
},
@@ -754,14 +754,14 @@
"azuredevops": {
"result": "Sonuç",
"status": "Durum",
"buildId": "Build ID",
"buildId": "Yapı Kimliği",
"succeeded": "Başarılı",
"notStarted": "Henüz Başlamadı",
"failed": "Başarısız",
"canceled": "İptal edildi",
"inProgress": "Sürüyor",
"totalPrs": "Total PRs",
"myPrs": "My PRs",
"totalPrs": "Toplam Çekme İstekleri",
"myPrs": "Benim Çekme İsteklerim",
"approved": "Onaylı"
},
"gamedig": {
@@ -791,7 +791,7 @@
"openmediavault": {
"downloading": "İndiriliyor",
"total": "Toplam",
"running": "Çalışan",
"running": "Çalışıyor",
"stopped": "Durduruldu",
"passed": "Geçti",
"failed": "Başarısız"
@@ -809,9 +809,9 @@
"uptime": "Çalışma Süresi",
"lastDown": "Son Kesinti",
"downDuration": "Kesinti Süresi",
"sitesUp": "Sites Up",
"sitesDown": "Sites Down",
"paused": "Durduruldu",
"sitesUp": "Siteler Çalışıyor",
"sitesDown": "Siteler Çalışmıyor",
"paused": "Duraklatıldı",
"notyetchecked": "Henüz Kontrol Edilmedi",
"up": "Yükleme",
"seemsdown": "Kapalı görünüyor",
@@ -821,7 +821,7 @@
"calendar": {
"inCinemas": "Sinemalarda",
"physicalRelease": "Fiziksel Yayınlanan",
"digitalRelease": "Dijital Yayınlanan",
"digitalRelease": "Dijitalde Yayınlandı",
"noEventsToday": "Bugün için etkinlik yok!",
"noEventsFound": "Etkinlik bulunamadı"
},
@@ -835,7 +835,7 @@
},
"plantit": {
"events": "Etkinlikler",
"plants": "Plants",
"plants": "Bitkiler",
"photos": "Fotoğraflar",
"species": "Türler"
},
@@ -854,11 +854,11 @@
"images": "Görseller",
"imageSize": "Görsel Boyutu",
"galleries": "Galeriler",
"performers": "Performers",
"performers": "Oyuncu",
"studios": "Stüdyolar",
"movies": "Filmler",
"tags": "Etiketler",
"oCount": "O Count"
"oCount": "O Sayısı"
},
"tandoor": {
"users": "Kullanıcılar",
@@ -872,5 +872,9 @@
"labels": "Etiketler",
"users": "Kullanıcılar",
"totalValue": "Toplam Değer"
},
"crowdsec": {
"alerts": "Alarmlar",
"bans": "Yasaklar"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "Pending Disconnect",
"connectionStatusDisconnecting": "Disconnecting",
"connectionStatusDisconnected": "Disconnected",
"connectionStatusConnected": "Підключено",
"connectionStatusConnected": "Connected",
"uptime": "Час роботи",
"maxDown": "Max. Down",
"maxUp": "Max. Up",
@@ -277,11 +277,11 @@
"approved": "Затверджено",
"available": "Доступно"
},
"pialert": {
"netalertx": {
"total": "Усього",
"connected": "Підключено",
"new_devices": "Нові пристрої",
"down_alerts": "Сповіщення про збій"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "Запити",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Користувачі",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Оповіщення",
"bans": "Bans"
}
}

View File

@@ -277,7 +277,7 @@
"approved": "Đã duyệt",
"available": "Available"
},
"pialert": {
"netalertx": {
"total": "Tổng",
"connected": "Connected",
"new_devices": "New Devices",
@@ -872,5 +872,9 @@
"labels": "Labels",
"users": "Users",
"totalValue": "Total Value"
},
"crowdsec": {
"alerts": "Alerts",
"bans": "Bans"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "待辦的斷開",
"connectionStatusDisconnecting": "正在中斷連線",
"connectionStatusDisconnected": "連接已中斷",
"connectionStatusConnected": "已連線",
"connectionStatusConnected": "Connected",
"uptime": "運行時間",
"maxDown": "最大下載速率",
"maxUp": "最大上傳速率",
@@ -277,11 +277,11 @@
"approved": "批准",
"available": "可用"
},
"pialert": {
"netalertx": {
"total": "全部",
"connected": "已連線",
"new_devices": "新裝置",
"down_alerts": "離線警告"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "查詢",
@@ -858,7 +858,7 @@
"studios": "工作室",
"movies": "電影",
"tags": "標籤",
"oCount": "O Count"
"oCount": "0 個"
},
"tandoor": {
"users": "使用者",
@@ -867,10 +867,14 @@
},
"homebox": {
"items": "項目",
"totalWithWarranty": "With Warranty",
"totalWithWarranty": "有保証",
"locations": "位置",
"labels": "標籤",
"users": "使用者",
"totalValue": "總共"
},
"crowdsec": {
"alerts": "警示",
"bans": "禁止"
}
}

View File

@@ -54,13 +54,13 @@
"uptime": "运行时间",
"days": "天",
"wan": "WAN",
"lan": "LAN",
"wlan": "WLAN",
"lan": "局域网",
"wlan": "无线局域网",
"devices": "设备",
"lan_devices": "LAN设备",
"lan_devices": "有线设备",
"wlan_devices": "WLAN 设备",
"lan_users": "LAN 用户",
"wlan_users": "WLAN 用户",
"lan_users": "有线用户",
"wlan_users": "无线用户",
"up": "运行时间",
"down": "离线",
"wait": "请稍候",
@@ -277,7 +277,7 @@
"approved": "已批准",
"available": "可用"
},
"pialert": {
"netalertx": {
"total": "总计",
"connected": "已连接",
"new_devices": "新设备",
@@ -858,7 +858,7 @@
"studios": "工作室",
"movies": "电影",
"tags": "标签",
"oCount": "O Count"
"oCount": "O "
},
"tandoor": {
"users": "用户数",
@@ -866,11 +866,15 @@
"keywords": "关键词"
},
"homebox": {
"items": "Items",
"totalWithWarranty": "With Warranty",
"locations": "Locations",
"labels": "Labels",
"items": "条目",
"totalWithWarranty": "有保证",
"locations": "位置",
"labels": "标签",
"users": "用户数",
"totalValue": "Total Value"
"totalValue": "总计"
},
"crowdsec": {
"alerts": "警告",
"bans": "禁用"
}
}

View File

@@ -140,7 +140,7 @@
"connectionStatusPendingDisconnect": "待辦的斷開",
"connectionStatusDisconnecting": "正在中斷連線",
"connectionStatusDisconnected": "連接已中斷",
"connectionStatusConnected": "已連線",
"connectionStatusConnected": "Connected",
"uptime": "運行時間",
"maxDown": "最大下載速率",
"maxUp": "最大上傳速率",
@@ -277,11 +277,11 @@
"approved": "已核准",
"available": "可觀看"
},
"pialert": {
"netalertx": {
"total": "全部",
"connected": "已連線",
"new_devices": "新裝置",
"down_alerts": "離線警告"
"connected": "Connected",
"new_devices": "New Devices",
"down_alerts": "Down Alerts"
},
"pihole": {
"queries": "查詢",
@@ -858,7 +858,7 @@
"studios": "工作室",
"movies": "電影",
"tags": "標籤",
"oCount": "O Count"
"oCount": "0 個"
},
"tandoor": {
"users": "用戶",
@@ -867,10 +867,14 @@
},
"homebox": {
"items": "項目",
"totalWithWarranty": "With Warranty",
"totalWithWarranty": "有保証",
"locations": "位置",
"labels": "標籤",
"users": "用戶",
"totalValue": "總共"
},
"crowdsec": {
"alerts": "警示",
"bans": "禁止"
}
}

View File

@@ -12,7 +12,7 @@ export default function SiteMonitor({ group, service, style }) {
let statusTitle = t("siteMonitor.http_status");
let statusText = "";
if (error) {
if (error || (data && data.error)) {
colorClass = "text-rose-500";
statusText = t("siteMonitor.error");
statusTitle += ` ${t("siteMonitor.error")}`;

View File

@@ -9,7 +9,7 @@ function convertToFahrenheit(t) {
return (t * 9) / 5 + 32;
}
export default function CpuTemp({ expanded, units, refresh = 1500 }) {
export default function CpuTemp({ expanded, units, refresh = 1500, tempmin = 0, tempmax = -1 }) {
const { t } = useTranslation();
const { data, error } = useSWR(`/api/widgets/resources?type=cputemp`, {
@@ -39,7 +39,12 @@ export default function CpuTemp({ expanded, units, refresh = 1500 }) {
}
const unit = units === "imperial" ? "fahrenheit" : "celsius";
mainTemp = unit === "celsius" ? mainTemp : convertToFahrenheit(mainTemp);
const maxTemp = unit === "celsius" ? data.cputemp.max : convertToFahrenheit(data.cputemp.max);
const minTemp = tempmin < mainTemp ? tempmin : mainTemp;
let maxTemp = tempmax;
if (maxTemp < minTemp) {
maxTemp = unit === "celsius" ? data.cputemp.max : convertToFahrenheit(data.cputemp.max);
}
return (
<Resource
@@ -58,7 +63,7 @@ export default function CpuTemp({ expanded, units, refresh = 1500 }) {
unit,
})}
expandedLabel={t("resources.max")}
percentage={Math.round((mainTemp / maxTemp) * 100)}
percentage={Math.round(((mainTemp - minTemp) / (maxTemp - minTemp)) * 100)}
expanded={expanded}
/>
);

View File

@@ -8,7 +8,7 @@ import CpuTemp from "./cputemp";
import Uptime from "./uptime";
export default function Resources({ options }) {
const { expanded, units, diskUnits } = options;
const { expanded, units, diskUnits, tempmin, tempmax } = options;
let { refresh } = options;
if (!refresh) refresh = 1500;
refresh = Math.max(refresh, 1000);
@@ -23,7 +23,9 @@ export default function Resources({ options }) {
<Disk key={disk} options={{ disk }} expanded={expanded} diskUnits={diskUnits} refresh={refresh} />
))
: options.disk && <Disk options={options} expanded={expanded} diskUnits={diskUnits} refresh={refresh} />}
{options.cputemp && <CpuTemp expanded={expanded} units={units} refresh={refresh} />}
{options.cputemp && (
<CpuTemp expanded={expanded} units={units} refresh={refresh} tempmin={tempmin} tempmax={tempmax} />
)}
{options.uptime && <Uptime refresh={refresh} />}
</div>
{options.label && (

View File

@@ -2,11 +2,7 @@ import { getAllClasses, getInnerBlock, getBottomBlock } from "./container";
export default function ContainerForm({ children = [], options, additionalClassNames = "", callback }) {
return (
<form
type="button"
onSubmit={callback}
className={`${getAllClasses(options, additionalClassNames)} information-widget-form`}
>
<form onSubmit={callback} className={`${getAllClasses(options, additionalClassNames)} information-widget-form`}>
{getInnerBlock(children)}
{getBottomBlock(children)}
</form>

View File

@@ -3,7 +3,7 @@ import { getAllClasses, getInnerBlock, getBottomBlock } from "./container";
export default function ContainerLink({ children = [], options, additionalClassNames = "", target }) {
return (
<a
href={options.url}
href={options.href || options.url}
target={target}
className={`${getAllClasses(options, additionalClassNames)} information-widget-link`}
>

View File

@@ -13,7 +13,7 @@ async function retrieveFromGlancesAPI(privateWidgetOptions, endpoint) {
throw new Error(errorMessage);
}
const apiUrl = `${url}/api/3/${endpoint}`;
const apiUrl = `${url}/api/${privateWidgetOptions.version}/${endpoint}`;
const headers = {
"Accept-Encoding": "application/json",
};
@@ -42,9 +42,10 @@ async function retrieveFromGlancesAPI(privateWidgetOptions, endpoint) {
}
export default async function handler(req, res) {
const { index, cputemp: includeCpuTemp, uptime: includeUptime, disk: includeDisks } = req.query;
const { index, cputemp: includeCpuTemp, uptime: includeUptime, disk: includeDisks, version } = req.query;
const privateWidgetOptions = await getPrivateWidgetOptions("glances", index);
privateWidgetOptions.version = version ?? 3;
try {
const cpuData = await retrieveFromGlancesAPI(privateWidgetOptions, "cpu");

View File

@@ -393,6 +393,14 @@ export function cleanServiceGroups(groups) {
enableBlocks,
enableNowPlaying,
// emby, jellyfin, tautulli
enableUser,
expandOneStreamToTwoRows,
showEpisodeNumber,
// glances, pihole
version,
// glances
chart,
metric,
@@ -442,11 +450,15 @@ export function cleanServiceGroups(groups) {
// proxmox
node,
// speedtest
bitratePrecision,
// sonarr, radarr
enableQueue,
// truenas
enablePools,
nasType,
// unifi
site,
@@ -514,11 +526,19 @@ export function cleanServiceGroups(groups) {
if (enableBlocks !== undefined) cleanedService.widget.enableBlocks = JSON.parse(enableBlocks);
if (enableNowPlaying !== undefined) cleanedService.widget.enableNowPlaying = JSON.parse(enableNowPlaying);
}
if (["emby", "jellyfin", "tautulli"].includes(type)) {
if (expandOneStreamToTwoRows !== undefined)
cleanedService.widget.expandOneStreamToTwoRows = !!JSON.parse(expandOneStreamToTwoRows);
if (showEpisodeNumber !== undefined)
cleanedService.widget.showEpisodeNumber = !!JSON.parse(showEpisodeNumber);
if (enableUser !== undefined) cleanedService.widget.enableUser = !!JSON.parse(enableUser);
}
if (["sonarr", "radarr"].includes(type)) {
if (enableQueue !== undefined) cleanedService.widget.enableQueue = JSON.parse(enableQueue);
}
if (type === "truenas") {
if (enablePools !== undefined) cleanedService.widget.enablePools = JSON.parse(enablePools);
if (nasType !== undefined) cleanedService.widget.nasType = nasType;
}
if (["diskstation", "qnap"].includes(type)) {
if (volume) cleanedService.widget.volume = volume;
@@ -527,6 +547,9 @@ export function cleanServiceGroups(groups) {
if (snapshotHost) cleanedService.widget.snapshotHost = snapshotHost;
if (snapshotPath) cleanedService.widget.snapshotPath = snapshotPath;
}
if (["glances", "pihole"].includes(type)) {
if (version) cleanedService.widget.version = version;
}
if (type === "glances") {
if (metric) cleanedService.widget.metric = metric;
if (chart !== undefined) {
@@ -568,6 +591,11 @@ export function cleanServiceGroups(groups) {
if (type === "healthchecks") {
if (uuid !== undefined) cleanedService.widget.uuid = uuid;
}
if (type === "speedtest") {
if (bitratePrecision !== undefined) {
cleanedService.widget.bitratePrecision = parseInt(bitratePrecision, 10);
}
}
}
return cleanedService;

View File

@@ -57,7 +57,7 @@ export function jsonArrayFilter(data, filter) {
export function sanitizeErrorURL(errorURL) {
// Dont display sensitive params on frontend
const url = new URL(errorURL);
["apikey", "api_key", "token", "t", "access_token"].forEach((key) => {
["apikey", "api_key", "token", "t", "access_token", "auth"].forEach((key) => {
if (url.searchParams.has(key)) url.searchParams.set(key, "***");
});
return url.toString();

View File

@@ -5,6 +5,7 @@ import { createUnzip, constants as zlibConstants } from "node:zlib";
import { http, https } from "follow-redirects";
import { addCookieToJar, setCookieHeader } from "./cookie-jar";
import { sanitizeErrorURL } from "./api-helpers";
import createLogger from "utils/logger";
@@ -113,6 +114,11 @@ export async function httpProxy(url, params = {}) {
constructedUrl.pathname,
);
if (err) logger.error(err);
return [500, "application/json", { error: { message: err?.message ?? "Unknown error", url, rawError: err } }, null];
return [
500,
"application/json",
{ error: { message: err?.message ?? "Unknown error", url: sanitizeErrorURL(url), rawError: err } },
null,
];
}
}

View File

@@ -15,6 +15,7 @@ const components = {
channelsdvrserver: dynamic(() => import("./channelsdvrserver/component")),
cloudflared: dynamic(() => import("./cloudflared/component")),
coinmarketcap: dynamic(() => import("./coinmarketcap/component")),
crowdsec: dynamic(() => import("./crowdsec/component")),
iframe: dynamic(() => import("./iframe/component")),
customapi: dynamic(() => import("./customapi/component")),
deluge: dynamic(() => import("./deluge/component")),
@@ -62,6 +63,7 @@ const components = {
moonraker: dynamic(() => import("./moonraker/component")),
mylar: dynamic(() => import("./mylar/component")),
navidrome: dynamic(() => import("./navidrome/component")),
netalertx: dynamic(() => import("./netalertx/component")),
netdata: dynamic(() => import("./netdata/component")),
nextcloud: dynamic(() => import("./nextcloud/component")),
nextdns: dynamic(() => import("./nextdns/component")),
@@ -79,7 +81,7 @@ const components = {
pfsense: dynamic(() => import("./pfsense/component")),
photoprism: dynamic(() => import("./photoprism/component")),
proxmoxbackupserver: dynamic(() => import("./proxmoxbackupserver/component")),
pialert: dynamic(() => import("./pialert/component")),
pialert: dynamic(() => import("./netalertx/component")),
pihole: dynamic(() => import("./pihole/component")),
plantit: dynamic(() => import("./plantit/component")),
plex: dynamic(() => import("./plex/component")),

View File

@@ -0,0 +1,34 @@
import { useTranslation } from "next-i18next";
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 { t } = useTranslation();
const { widget } = service;
const { data: alerts, error: alertsError } = useWidgetAPI(widget, "alerts");
const { data: bans, error: bansError } = useWidgetAPI(widget, "bans");
if (alertsError || bansError) {
return <Container service={service} error={alertsError ?? bansError} />;
}
if (!alerts && !bans) {
return (
<Container service={service}>
<Block label="crowdsec.alerts" />
<Block label="crowdsec.bans" />
</Container>
);
}
return (
<Container service={service}>
<Block label="crowdsec.alerts" value={t("common.number", { value: alerts?.length ?? 0 })} />
<Block label="crowdsec.bans" value={t("common.number", { value: bans?.length ?? 0 })} />
</Container>
);
}

View File

@@ -0,0 +1,86 @@
import cache from "memory-cache";
import { httpProxy } from "utils/proxy/http";
import { formatApiCall } from "utils/proxy/api-helpers";
import getServiceWidget from "utils/config/service-helpers";
import createLogger from "utils/logger";
import widgets from "widgets/widgets";
const proxyName = "crowdsecProxyHandler";
const logger = createLogger(proxyName);
const sessionTokenCacheKey = `${proxyName}__sessionToken`;
async function login(widget, service) {
const url = formatApiCall(widgets[widget.type].loginURL, widget);
const [status, , data] = await httpProxy(url, {
method: "POST",
headers: {
"Content-Type": "application/json",
"User-Agent": "Mozilla/5.0", // Crowdsec requires a user-agent
},
body: JSON.stringify({
machine_id: widget.username,
password: widget.password,
scenarios: [],
}),
});
const dataParsed = JSON.parse(data);
if (!(status === 200) || !dataParsed.token) {
logger.error("Failed to login to Crowdsec API, status: %d", status);
cache.del(`${sessionTokenCacheKey}.${service}`);
}
cache.put(`${sessionTokenCacheKey}.${service}`, dataParsed.token, new Date(dataParsed.expire) - new Date());
}
export default async function crowdsecProxyHandler(req, res) {
const { group, service, endpoint } = req.query;
if (!group || !service) {
logger.error("Invalid or missing service '%s' or group '%s'", service, group);
return res.status(400).json({ error: "Invalid proxy service type" });
}
const widget = await getServiceWidget(group, service);
if (!widget || !widgets[widget.type].api) {
logger.error("Invalid or missing widget for service '%s' in group '%s'", service, group);
return res.status(400).json({ error: "Invalid widget configuration" });
}
if (!cache.get(`${sessionTokenCacheKey}.${service}`)) {
await login(widget, service);
}
const token = cache.get(`${sessionTokenCacheKey}.${service}`);
if (!token) {
return res.status(500).json({ error: "Failed to authenticate with Crowdsec" });
}
const url = new URL(formatApiCall(widgets[widget.type].api, { endpoint, ...widget }));
try {
const params = {
method: "GET",
headers: {
"User-Agent": "Mozilla/5.0", // Crowdsec requires a user-agent
"Content-Type": "application/json",
Authorization: `Bearer ${token}`,
},
};
logger.debug("Calling Crowdsec API endpoint: %s", endpoint);
const [status, , data] = await httpProxy(url, params);
if (status !== 200) {
logger.error("Error calling Crowdsec API: %d. Data: %s", status, data);
return res.status(status).json({ error: "Crowdsec API Error", data });
}
return res.status(status).send(data);
} catch (error) {
logger.error("Exception calling Crowdsec API: %s", error.message);
return res.status(500).json({ error: "Crowdsec API Error", message: error.message });
}
}

View File

@@ -0,0 +1,18 @@
import crowdsecProxyHandler from "./proxy";
const widget = {
api: "{url}/v1/{endpoint}",
loginURL: "{url}/v1/watchers/login",
proxyHandler: crowdsecProxyHandler,
mappings: {
alerts: {
endpoint: "alerts",
},
bans: {
endpoint: "alerts?decision_type=ban&origin=crowdsec&has_active_decision=1",
},
},
};
export default widget;

View File

@@ -27,9 +27,26 @@ function ticksToString(ticks) {
return parts.map((part) => part.toString().padStart(2, "0")).join(":");
}
function SingleSessionEntry({ playCommand, session }) {
function generateStreamTitle(session, enableUser, showEpisodeNumber) {
const {
NowPlayingItem: { Name, SeriesName, Type, ParentIndexNumber, IndexNumber },
UserName,
} = session;
let streamTitle = "";
if (Type === "Episode" && showEpisodeNumber) {
const seasonStr = `S${ParentIndexNumber.toString().padStart(2, "0")}`;
const episodeStr = `E${IndexNumber.toString().padStart(2, "0")}`;
streamTitle = `${SeriesName}: ${seasonStr} · ${episodeStr} - ${Name}`;
} else {
streamTitle = `${Name}${SeriesName ? ` - ${SeriesName}` : ""}`;
}
return enableUser ? `${streamTitle} (${UserName})` : streamTitle;
}
function SingleSessionEntry({ playCommand, session, enableUser, showEpisodeNumber }) {
const {
NowPlayingItem: { Name, SeriesName },
PlayState: { PositionTicks, IsPaused, IsMuted },
} = session;
@@ -42,13 +59,13 @@ function SingleSessionEntry({ playCommand, session }) {
const percent = Math.min(1, PositionTicks / RunTimeTicks) * 100;
const streamTitle = generateStreamTitle(session, enableUser, showEpisodeNumber);
return (
<>
<div className="text-theme-700 dark:text-theme-200 relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1 flex">
<div className="grow text-xs z-10 self-center ml-2 relative w-full h-4 mr-2">
<div className="absolute w-full whitespace-nowrap text-ellipsis overflow-hidden">
{Name}
{SeriesName && ` - ${SeriesName}`}
<div className="absolute w-full whitespace-nowrap text-ellipsis overflow-hidden" title={streamTitle}>
{streamTitle}
</div>
</div>
<div className="self-center text-xs flex justify-end mr-1.5 pl-1">
@@ -97,9 +114,8 @@ function SingleSessionEntry({ playCommand, session }) {
);
}
function SessionEntry({ playCommand, session }) {
function SessionEntry({ playCommand, session, enableUser, showEpisodeNumber }) {
const {
NowPlayingItem: { Name, SeriesName },
PlayState: { PositionTicks, IsPaused, IsMuted },
} = session;
@@ -110,6 +126,8 @@ function SessionEntry({ playCommand, session }) {
IsVideoDirect: true,
}; // if no transcodinginfo its videodirect
const streamTitle = generateStreamTitle(session, enableUser, showEpisodeNumber);
const percent = Math.min(1, PositionTicks / RunTimeTicks) * 100;
return (
@@ -139,9 +157,8 @@ function SessionEntry({ playCommand, session }) {
)}
</div>
<div className="grow text-xs z-10 self-center relative w-full h-4">
<div className="absolute w-full whitespace-nowrap text-ellipsis overflow-hidden">
{Name}
{SeriesName && ` - ${SeriesName}`}
<div className="absolute w-full whitespace-nowrap text-ellipsis overflow-hidden" title={streamTitle}>
{streamTitle}
</div>
</div>
<div className="self-center text-xs flex justify-end mr-1 z-10">{IsMuted && <BsVolumeMuteFill />}</div>
@@ -215,6 +232,9 @@ export default function Component({ service }) {
const enableBlocks = service.widget?.enableBlocks;
const enableNowPlaying = service.widget?.enableNowPlaying ?? true;
const enableUser = !!service.widget?.enableUser; // default is false
const expandOneStreamToTwoRows = service.widget?.expandOneStreamToTwoRows !== false; // default is true
const showEpisodeNumber = !!service.widget?.showEpisodeNumber; // default is false
if (!sessionsData || !countData) {
return (
@@ -225,9 +245,11 @@ export default function Component({ service }) {
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">-</span>
</div>
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">-</span>
</div>
{expandOneStreamToTwoRows && (
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">-</span>
</div>
)}
</div>
)}
</>
@@ -255,15 +277,17 @@ export default function Component({ service }) {
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">{t("emby.no_active")}</span>
</div>
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">-</span>
</div>
{expandOneStreamToTwoRows && (
<div className="text-theme-700 dark:text-theme-200 text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
<span className="absolute left-2 text-xs mt-[2px]">-</span>
</div>
)}
</div>
</>
);
}
if (playing.length === 1) {
if (expandOneStreamToTwoRows && playing.length === 1) {
const session = playing[0];
return (
<>
@@ -272,27 +296,30 @@ export default function Component({ service }) {
<SingleSessionEntry
playCommand={(currentSession, command) => handlePlayCommand(currentSession, command)}
session={session}
enableUser={enableUser}
showEpisodeNumber={showEpisodeNumber}
/>
</div>
</>
);
}
if (playing.length > 0)
return (
<>
{enableBlocks && <CountBlocks service={service} countData={countData} />}
<div className="flex flex-col pb-1 mx-1">
{playing.map((session) => (
<SessionEntry
key={session.Id}
playCommand={(currentSession, command) => handlePlayCommand(currentSession, command)}
session={session}
/>
))}
</div>
</>
);
return (
<>
{enableBlocks && <CountBlocks service={service} countData={countData} />}
<div className="flex flex-col pb-1 mx-1">
{playing.map((session) => (
<SessionEntry
key={session.Id}
playCommand={(currentSession, command) => handlePlayCommand(currentSession, command)}
session={session}
enableUser={enableUser}
showEpisodeNumber={showEpisodeNumber}
/>
))}
</div>
</>
);
}
if (enableBlocks) {

View File

@@ -16,15 +16,15 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit } = widget;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit, version = 3 } = widget;
const [dataPoints, setDataPoints] = useState(new Array(pointsLimit).fill({ value: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(service.widget, "cpu", {
const { data, error } = useWidgetAPI(service.widget, `${version}/cpu`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});
const { data: quicklookData, error: quicklookError } = useWidgetAPI(service.widget, "quicklook");
const { data: quicklookData, error: quicklookError } = useWidgetAPI(service.widget, `${version}/quicklook`);
useEffect(() => {
if (data) {

View File

@@ -16,7 +16,7 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit } = widget;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit, version = 3 } = widget;
const [, diskName] = widget.metric.split(":");
const [dataPoints, setDataPoints] = useState(
@@ -24,7 +24,7 @@ export default function Component({ service }) {
);
const [ratePoints, setRatePoints] = useState(new Array(pointsLimit).fill({ a: 0, b: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(service.widget, "diskio", {
const { data, error } = useWidgetAPI(service.widget, `${version}/diskio`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});

View File

@@ -11,11 +11,11 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval } = widget;
const { chart, refreshInterval = defaultInterval, version = 3 } = widget;
const [, fsName] = widget.metric.split("fs:");
const diskUnits = widget.diskUnits === "bbytes" ? "common.bbytes" : "common.bytes";
const { data, error } = useWidgetAPI(widget, "fs", {
const { data, error } = useWidgetAPI(widget, `${version}/fs`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});

View File

@@ -16,12 +16,12 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit } = widget;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit, version = 3 } = widget;
const [, gpuName] = widget.metric.split(":");
const [dataPoints, setDataPoints] = useState(new Array(pointsLimit).fill({ a: 0, b: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(widget, "gpu", {
const { data, error } = useWidgetAPI(widget, `${version}/gpu`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});

View File

@@ -74,13 +74,13 @@ const defaultSystemInterval = 30000; // This data (OS, hostname, distribution) i
export default function Component({ service }) {
const { widget } = service;
const { chart, refreshInterval = defaultInterval } = widget;
const { chart, refreshInterval = defaultInterval, version = 3 } = widget;
const { data: quicklookData, errorL: quicklookError } = useWidgetAPI(service.widget, "quicklook", {
const { data: quicklookData, errorL: quicklookError } = useWidgetAPI(service.widget, `${version}/quicklook`, {
refreshInterval,
});
const { data: systemData, errorL: systemError } = useWidgetAPI(service.widget, "system", {
const { data: systemData, errorL: systemError } = useWidgetAPI(service.widget, `${version}/system`, {
refreshInterval: defaultSystemInterval,
});

View File

@@ -17,11 +17,11 @@ export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart } = widget;
const { refreshInterval = defaultInterval(chart), pointsLimit = defaultPointsLimit } = widget;
const { refreshInterval = defaultInterval(chart), pointsLimit = defaultPointsLimit, version = 3 } = widget;
const [dataPoints, setDataPoints] = useState(new Array(pointsLimit).fill({ value: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(service.widget, "mem", {
const { data, error } = useWidgetAPI(service.widget, `${version}/mem`, {
refreshInterval: Math.max(defaultInterval(chart), refreshInterval),
});

View File

@@ -17,13 +17,16 @@ export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, metric } = widget;
const { refreshInterval = defaultInterval(chart), pointsLimit = defaultPointsLimit } = widget;
const { refreshInterval = defaultInterval(chart), pointsLimit = defaultPointsLimit, version = 3 } = widget;
const rxKey = version === 3 ? "rx" : "bytes_recv";
const txKey = version === 3 ? "tx" : "bytes_sent";
const [, interfaceName] = metric.split(":");
const [dataPoints, setDataPoints] = useState(new Array(pointsLimit).fill({ value: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(widget, "network", {
const { data, error } = useWidgetAPI(widget, `${version}/network`, {
refreshInterval: Math.max(defaultInterval(chart), refreshInterval),
});
@@ -36,8 +39,8 @@ export default function Component({ service }) {
const newDataPoints = [
...prevDataPoints,
{
a: (interfaceData.rx * 8) / interfaceData.time_since_update,
b: (interfaceData.tx * 8) / interfaceData.time_since_update,
a: (interfaceData[rxKey] * 8) / interfaceData.time_since_update,
b: (interfaceData[txKey] * 8) / interfaceData.time_since_update,
},
];
if (newDataPoints.length > pointsLimit) {
@@ -47,7 +50,7 @@ export default function Component({ service }) {
});
}
}
}, [data, interfaceName, pointsLimit]);
}, [data, interfaceName, pointsLimit, rxKey, txKey]);
if (error) {
return (
@@ -97,7 +100,7 @@ export default function Component({ service }) {
<div className="text-xs opacity-75">
{t("common.bitrate", {
value: (interfaceData.rx * 8) / interfaceData.time_since_update,
value: (interfaceData[rxKey] * 8) / interfaceData.time_since_update,
maximumFractionDigits: 0,
})}{" "}
{t("docker.rx")}
@@ -115,7 +118,7 @@ export default function Component({ service }) {
<Block position="bottom-3 right-3">
<div className="text-xs opacity-75">
{t("common.bitrate", {
value: (interfaceData.tx * 8) / interfaceData.time_since_update,
value: (interfaceData[txKey] * 8) / interfaceData.time_since_update,
maximumFractionDigits: 0,
})}{" "}
{t("docker.tx")}

View File

@@ -22,9 +22,11 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval } = widget;
const { chart, refreshInterval = defaultInterval, version = 3 } = widget;
const { data, error } = useWidgetAPI(service.widget, "processlist", {
const memoryInfoKey = version === 3 ? 0 : "data";
const { data, error } = useWidgetAPI(service.widget, `${version}/processlist`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});
@@ -66,7 +68,7 @@ export default function Component({ service }) {
<div className="opacity-25 w-14 text-right">{item.cpu_percent.toFixed(1)}%</div>
<div className="opacity-25 w-14 text-right">
{t("common.bytes", {
value: item.memory_info[0],
value: item.memory_info[memoryInfoKey],
maximumFractionDigits: 0,
})}
</div>

View File

@@ -16,12 +16,12 @@ const defaultInterval = 1000;
export default function Component({ service }) {
const { t } = useTranslation();
const { widget } = service;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit } = widget;
const { chart, refreshInterval = defaultInterval, pointsLimit = defaultPointsLimit, version = 3 } = widget;
const [, sensorName] = widget.metric.split(":");
const [dataPoints, setDataPoints] = useState(new Array(pointsLimit).fill({ value: 0 }, 0, pointsLimit));
const { data, error } = useWidgetAPI(service.widget, "sensors", {
const { data, error } = useWidgetAPI(service.widget, `${version}/sensors`, {
refreshInterval: Math.max(defaultInterval, refreshInterval),
});

View File

@@ -1,7 +1,7 @@
import credentialedProxyHandler from "utils/proxy/handlers/credentialed";
const widget = {
api: "{url}/api/3/{endpoint}",
api: "{url}/api/{endpoint}",
proxyHandler: credentialedProxyHandler,
};

View File

@@ -14,7 +14,7 @@ async function login(widget, service) {
const endpoint = "auth/login";
const api = widgets?.[widget.type]?.api;
const loginUrl = new URL(formatApiCall(api, { endpoint, ...widget }));
const loginBody = { username: widget.username, password: widget.password };
const loginBody = { username: widget.username.toString(), password: widget.password.toString() };
const headers = { "Content-Type": "application/json" };
// eslint-disable-next-line no-unused-vars
const [status, contentType, data, responseHeaders] = await httpProxy(loginUrl, {

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