Merge branch 'master' into api-only

This commit is contained in:
Omar Roth 2019-05-31 10:12:07 -05:00
commit 6722f14b72
No known key found for this signature in database
GPG Key ID: B8254FB7EC3D37F2
40 changed files with 7532 additions and 3935 deletions

View File

@ -1,3 +1,122 @@
# 0.17.0 (2019-05-06)
# Version 0.17.0: Player and Authentication API
Hello everyone! This past month there have been [130 commits](https://github.com/omarroth/invidious/compare/0.16.0..0.17.0) from 11 contributors. Large focus has been on improving the player as well as adding API access for other projects to make use of Invidious.
There have also been significant changes in preparation of native notifications (see [#195](https://github.com/omarroth/invidious/issues/195), [#469](https://github.com/omarroth/invidious/issues/469), [#473](https://github.com/omarroth/invidious/issues/473), and [#502](https://github.com/omarroth/invidious/issues/502)), and playlists. I expect to see both of these to be added in the next release.
I'm quite happy to mention that new translations have been added for Esperanto (`eo`) and Ukranian (`uk`). Support for pluralization has also been added, so it should now be possible to make a more native experience for speakers in other languages. The system currently in place is a bit cumbersome, so for any help using this feature please get in touch!
## For Administrators
A `check_tables` option has been added to automatically migrate without the use of custom scripts. This method will likely prove to be much more robust, and is currently enabled for the official instance. To prevent any unintended changes to the DB, `check_tables` is disabled by default and will print commands before executing. Having this makes features that require schema changes much easier to implement, and also makes it easier to upgrade from older instances.
As part of [#303](https://github.com/omarroth/invidious/issues/303), a `cache_annotations` option has been added to speed up access from `/api/v1/annotations/:id`. This vastly improves the experience for videos with annotations. Currently, only videos that contain legacy annotations will be cached, which should help keep down the size of the cache. `cache_annotations` is disabled by default.
## For Developers
An authorization API has been added which allows other applications to read and modify user subscriptions and preferences (see [#473](https://github.com/omarroth/invidious/issues/473)). Support for accessing user feeds and notifications is also planned. I believe this feature is a large step forward in supporting syncing subscriptions and preferences with other services, and I'm excited to see what other developers do with this functionality.
Support for server-to-client push notifications is currently underway. This allows Invidious users, as well as applications using the Invidious API, to receive notifications about uploads in near real-time (see #469). An `/api/v1/auth/notifications` endpoint is currently available. I'm very excited for this to be integrated into the site, and to see how other developers use it in their own projects.
An `/api/v1/storyboards/:id` endpoint has been added for accessing storyboard URLs, which allows developers to add video previews to their players (see below).
## Player
Support for annotations has been merged into master with [#303](https://github.com/omarroth/invidious/issues/303), thanks @glmdgrielson! Annotations can be enabled by default or only for subscribed channels, and can also be toggled per video. I'm extremely proud of the progress made here, and I'm so thankful to everyone that has made this possible. I expect this to be the last update with regards to supporting annotations, but I do plan on continuing to improve the experience as much as possible.
The Invidious player now supports video previews and a corresponding API endpoint `/api/v1/storyboards/:id` has been added for developers looking to add similar functionality to their own players. Not much else to say here. Overall it's a very nice quality of life improvement and an attractive addition to the site.
It is now possible to select specific sources for videos provided using DASH (see [#34](https://github.com/omarroth/invidious/issues/34)). I would consider support largely feature complete, although there are still several issues to be fixed before I would consider it ready for larger rollout. You can watch videos in 1080p by setting `Default quality` to `dash` in your preferences, or by adding `&quality=dash` to the end of video URLs.
## Finances
### Donations
- [Patreon](https://www.patreon.com/omarroth) : \$49.73
- [Liberapay](https://liberapay.com/omarroth) : \$63.03
- Crypto : ~\$0.00 (converted from BCH, BTC)
- Total : \$112.76
### Expenses
- invidious-load1 (nyc1) : \$10.00 (load balancer)
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
- Total : \$80.00
That's all for now. Thanks!
# 0.16.0 (2019-04-06)
# Version 0.16.0: API Improvements and Annotations
Hello again! This past month has seen [116 commits](https://github.com/omarroth/invidious/compare/0.15.0..0.16.0) from 13 contributors and a couple important changes I'd like to announce.
A privacy policy is now available [here](https://invidio.us/privacy). I've done my best to explain things as clearly as possible without oversimplifying, and would very much recommend reading it if you're concerned about your privacy and want to learn more about how Invidious uses your data. Please let me know if there is anything that needs clarification.
I'm also very happy to announce that a Spanish translation has been added to the site. You can use it with `?hl=es` or by setting `es` as your default locale. As always I'm extremely grateful to translators for making the site accessible to more people.
## For Administrators
Invidious now supports server-to-server [push notifications](https://developers.google.com/youtube/v3/guides/push_notifications). This uses [PubSubHubbub](https://pubsubhubbub.github.io/PubSubHubbub/pubsubhubbub-core-0.4.html) to automatically handle new videos sent to an instance, which is less resource intensive and generally faster. Note that it will not pull all videos from a subscribed channel, so recommended usage is in addition to `channel_threads`. Using PubSub requires a valid `domain` that updates can be sent to, and a random string that can be used to sign updates sent to the instance. You can enable it by adding `use_pubsub_feeds: true` to your `config.yml`. See [Configuration](https://github.com/omarroth/invidious/wiki/Configuration) for more info.
Unfortunately there are a couple necessary changes to the DB to support `liveNow` and `premiereTimestamp` in subscription feeds. Migration scripts have been provided that should be used automatically if following the instructions [here](https://github.com/omarroth/invidious/wiki/Updating).
You can now configure default user preferences for your instance. This allows you to set default locale, player preferences, and more. See [#415](https://github.com/omarroth/invidious/issues/415) for more details and example usage.
## For Developers
The [fields](https://developers.google.com/youtube/v3/getting-started#fields) API has been added with [#429](https://github.com/omarroth/invidious/pull/429) and is now supported on all JSON endpoints, thanks [**@afrmtbl**](https://github.com/afrmtbl)! Synax is straight-forward and can be used to reduce data transfer and create a simpler response for debugging. You can see an example [here](https://invidio.us/api/v1/videos/CvFH_6DNRCY?pretty=1&fields=title,recommendedVideos/title). I've been quite happy using it and hope it is similarly useful for others.
An `/api/v1/annotations/:id` endpoint has been added for pulling legacy annotation data from [this](https://archive.org/details/youtubeannotations) archive, see below for more details. You can also access annotation data available on YouTube using `?source=youtube`, although this will only return card data as legacy annotations were deleted on January 15th.
A couple minor changes to existing endpoints:
- A `premiereTimestamp` field has been added to `/api/v1/videos/:id`
- A `sort_by` param has been added to `/api/v1/comments/:id`, supports `new`, `top`.
More info is available in the [documentation](https://github.com/omarroth/invidious/wiki/API).
## Annotations
I'm pleased to announce that annotation data is finally available from the roughly 1.4 billion videos archived as part of [this](https://www.reddit.com/r/DataHoarder/comments/aa6czg/youtube_annotation_archive/) project. They are accessible from the Internet Archive [here](https://archive.org/details/youtubeannotations) or as a 355GB torrent, see [here](https://www.reddit.com/r/DataHoarder/comments/b7imx9/youtube_annotation_archive_annotation_data_from/) for more details. A corresponding `/api/v1/annotations/:id` endpoint has been added to Invidious which uses the collection from IA to provide legacy annotations.
Support for them in the player is possible thanks to [this](https://github.com/afrmtbl/videojs-youtube-annotations) plugin developed by [**@afrmtbl**](https://github.com/afrmtbl). A PR for adding support to the site is available as [#303](https://github.com/omarroth/invidious/pull/303). There's also an [extension](https://github.com/afrmtbl/AnnotationsRestored) for overlaying them on top of the YouTube player (again thanks to [**@afrmtbl**](https://github.com/afrmtbl)), and an [extension](https://tech234a.bitbucket.io/AnnotationsReloaded?src=invidious) for hooking into code still present in the YouTube player itself, developed by [**@tech234a**](https://github.com/tech234a).
I would recommend reading the [official announcement](https://www.reddit.com/r/DataHoarder/comments/b7imx9/youtube_annotation_archive_annotation_data_from/) for more details. I would like to again thank everyone that helped contribute to this project.
## Finances
### Donations
- [Patreon](https://www.patreon.com/omarroth) : \$42.42
- [Liberapay](https://liberapay.com/omarroth) : \$70.11
- Crypto : ~\$1.76 (converted from BCH, BTC, BSV)
- Total : \$114.29
### Expenses
- invidious-load1 (nyc1) : \$10.00 (load balancer)
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
- Total : \$80.00
This past month the site saw a couple abnormal peaks in traffic, so an additional webserver has been added to match the increased load. The goal on Patreon has been updated to match the above expenses.
Thanks everyone!
# 0.15.0 (2019-03-06) # 0.15.0 (2019-03-06)
## Version 0.15.0: Preferences and Channel Playlists ## Version 0.15.0: Preferences and Channel Playlists
@ -42,21 +161,21 @@ There's also more discussion on improving Invidious for streaming music in [#304
### Donations ### Donations
[Patreon](https://www.patreon.com/omarroth) : \$42.42 - [Patreon](https://www.patreon.com/omarroth) : \$42.42
[Liberapay](https://liberapay.com/omarroth) : \$30.97 - [Liberapay](https://liberapay.com/omarroth) : \$30.97
Crypto : ~\$0.00 (converted from BCH, BTC) - Crypto : ~\$0.00 (converted from BCH, BTC)
Total : \$73.39 - Total : \$73.39
### Expenses ### Expenses
invidious-load1 (nyc1) : \$10.00 (load balancer) - invidious-load1 (nyc1) : \$10.00 (load balancer)
invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds) - invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server) - invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server) - invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server) - invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server) - invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-db1 (s-4vcpu-8gb) : \$40.00 (database) - invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
Total : \$75.00 - Total : \$75.00
It's been very humbling to see how fast the project has grown, and I look forward to making the site even better. Thank you everyone. It's been very humbling to see how fast the project has grown, and I look forward to making the site even better. Thank you everyone.
@ -121,14 +240,14 @@ Organizing this project has unfortunately taken up quite a bit of my time, and I
### Expenses ### Expenses
invidious-load1 (nyc1) : $10.00 (load balancer) - invidious-load1 (nyc1) : \$10.00 (load balancer)
invidious-update1 (s-1vcpu-1gb) : $5.00 (updates feeds) - invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
invidious-node1 (s-1vcpu-1gb) : $5.00 (web server) - invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node2 (s-1vcpu-1gb) : $5.00 (web server) - invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node3 (s-1vcpu-1gb) : $5.00 (web server) - invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-node4 (s-1vcpu-1gb) : $5.00 (web server) - invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
invidious-db1 (s-4vcpu-8gb) : $40.00 (database) - invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
Total : $75.00 - Total : \$75.00
As always I'm grateful for everyone's contributions and support. I'll see you all in March. As always I'm grateful for everyone's contributions and support. I'll see you all in March.

View File

@ -101,14 +101,15 @@ $ exit
$ sudo systemctl enable postgresql $ sudo systemctl enable postgresql
$ sudo systemctl start postgresql $ sudo systemctl start postgresql
$ sudo -i -u postgres $ sudo -i -u postgres
$ psql -c "CREATE USER kemal WITH PASSWORD 'kemal';" $ psql -c "CREATE USER kemal WITH PASSWORD 'kemal';" # Change 'kemal' here to a stronger password, and update `password` in config/config.yml
$ createdb -O kemal invidious $ createdb -O kemal invidious
$ psql invidious < /home/invidious/invidious/config/sql/channels.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/channels.sql
$ psql invidious < /home/invidious/invidious/config/sql/videos.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/videos.sql
$ psql invidious < /home/invidious/invidious/config/sql/channel_videos.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/channel_videos.sql
$ psql invidious < /home/invidious/invidious/config/sql/users.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/users.sql
$ psql invidious < /home/invidious/invidious/config/sql/session_ids.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/session_ids.sql
$ psql invidious < /home/invidious/invidious/config/sql/nonces.sql $ psql invidious kemal < /home/invidious/invidious/config/sql/nonces.sql
$ psql invidious kemal < /home/invidious/invidious/config/sql/annotations.sql
$ exit $ exit
``` ```
@ -143,14 +144,15 @@ $ brew install shards crystal-lang postgres imagemagick librsvg
$ git clone https://github.com/omarroth/invidious $ git clone https://github.com/omarroth/invidious
$ cd invidious $ cd invidious
$ brew services start postgresql $ brew services start postgresql
$ psql -c "CREATE ROLE kemal WITH LOGIN PASSWORD 'kemal';" $ psql -c "CREATE ROLE kemal WITH PASSWORD 'kemal';" # Change 'kemal' here to a stronger password, and update `password` in config/config.yml
$ createdb invidious -U kemal $ createdb -O kemal invidious
$ psql invidious < config/sql/channels.sql $ psql invidious kemal < config/sql/channels.sql
$ psql invidious < config/sql/videos.sql $ psql invidious kemal < config/sql/videos.sql
$ psql invidious < config/sql/channel_videos.sql $ psql invidious kemal < config/sql/channel_videos.sql
$ psql invidious < config/sql/users.sql $ psql invidious kemal < config/sql/users.sql
$ psql invidious < config/sql/session_ids.sql $ psql invidious kemal < config/sql/session_ids.sql
$ psql invidious < config/sql/nonces.sql $ psql invidious kemal < config/sql/nonces.sql
$ psql invidious kemal < config/sql/annotations.sql
# Setup Invidious # Setup Invidious
$ shards update && shards install $ shards update && shards install
@ -172,15 +174,12 @@ Usage: invidious [arguments]
--ssl-key-file FILE SSL key file --ssl-key-file FILE SSL key file
--ssl-cert-file FILE SSL certificate file --ssl-cert-file FILE SSL certificate file
-h, --help Shows this help -h, --help Shows this help
-t THREADS, --crawl-threads=THREADS
Number of threads for crawling YouTube (default: 0)
-c THREADS, --channel-threads=THREADS -c THREADS, --channel-threads=THREADS
Number of threads for refreshing channels (default: 1) Number of threads for refreshing channels (default: 1)
-f THREADS, --feed-threads=THREADS -f THREADS, --feed-threads=THREADS
Number of threads for refreshing feeds (default: 1) Number of threads for refreshing feeds (default: 1)
-v THREADS, --video-threads=THREADS
Number of threads for refreshing videos (default: 0)
-o OUTPUT, --output=OUTPUT Redirect output (default: STDOUT) -o OUTPUT, --output=OUTPUT Redirect output (default: STDOUT)
-v, --version Print version
``` ```
Or for development: Or for development:
@ -188,6 +187,7 @@ Or for development:
```bash ```bash
$ curl -fsSLo- https://raw.githubusercontent.com/samueleaton/sentry/master/install.cr | crystal eval $ curl -fsSLo- https://raw.githubusercontent.com/samueleaton/sentry/master/install.cr | crystal eval
$ ./sentry $ ./sentry
🤖 Your SentryBot is vigilant. beep-boop...
``` ```
## Documentation ## Documentation
@ -201,7 +201,7 @@ $ ./sentry
## Made with Invidious ## Made with Invidious
- [FreeTube](https://github.com/FreeTubeApp/FreeTube): An Open Source YouTube app for privacy. - [FreeTube](https://github.com/FreeTubeApp/FreeTube): An Open Source YouTube app for privacy.
- [CloudTube](https://github.com/cloudrac3r/cadencegq): Website featuring pastebin, image host, and YouTube player - [CloudTube](https://cadence.moe/cloudtube/subscriptions): A JS-rich alternate YouTube player
- [PeerTubeify](https://gitlab.com/Ealhad/peertubeify): On YouTube, displays a link to the same video on PeerTube, if it exists. - [PeerTubeify](https://gitlab.com/Ealhad/peertubeify): On YouTube, displays a link to the same video on PeerTube, if it exists.
- [MusicPiped](https://github.com/deep-gaurav/MusicPiped): A materialistic music player that streams music from YouTube. - [MusicPiped](https://github.com/deep-gaurav/MusicPiped): A materialistic music player that streams music from YouTube.

View File

@ -1,7 +1,7 @@
FROM archlinux/base FROM archlinux/base
RUN pacman -Sy --noconfirm shards crystal imagemagick librsvg \ RUN pacman -Sy --noconfirm shards crystal imagemagick librsvg \
which pkgconf gcc ttf-liberation which pkgconf gcc ttf-liberation glibc
# base-devel contains many other basic packages, that are normally assumed to already exist on a clean arch system # base-devel contains many other basic packages, that are normally assumed to already exist on a clean arch system
ADD . /invidious ADD . /invidious

View File

@ -12,12 +12,13 @@ if [ ! -f /var/lib/postgresql/data/setupFinished ]; then
>&2 echo "### importing table schemas" >&2 echo "### importing table schemas"
su postgres -c 'createdb invidious' su postgres -c 'createdb invidious'
su postgres -c 'psql -c "CREATE USER kemal WITH PASSWORD '"'kemal'"'"' su postgres -c 'psql -c "CREATE USER kemal WITH PASSWORD '"'kemal'"'"'
su postgres -c 'psql invidious < config/sql/channels.sql' su postgres -c 'psql invidious kemal < config/sql/channels.sql'
su postgres -c 'psql invidious < config/sql/videos.sql' su postgres -c 'psql invidious kemal < config/sql/videos.sql'
su postgres -c 'psql invidious < config/sql/channel_videos.sql' su postgres -c 'psql invidious kemal < config/sql/channel_videos.sql'
su postgres -c 'psql invidious < config/sql/users.sql' su postgres -c 'psql invidious kemal < config/sql/users.sql'
su postgres -c 'psql invidious < config/sql/session_ids.sql' su postgres -c 'psql invidious kemal < config/sql/session_ids.sql'
su postgres -c 'psql invidious < config/sql/nonces.sql' su postgres -c 'psql invidious kemal < config/sql/nonces.sql'
su postgres -c 'psql invidious kemal < config/sql/annotations.sql'
touch /var/lib/postgresql/data/setupFinished touch /var/lib/postgresql/data/setupFinished
echo "### invidious database setup finished" echo "### invidious database setup finished"
exit exit

View File

@ -1,297 +1,315 @@
{ {
"`x` subscribers": "`x` المشتركين", "`x` subscribers": "`x` المشتركين",
"`x` videos": "`x` الفيديوهات", "`x` videos": "`x` الفيديوهات",
"LIVE": "مباشر", "LIVE": "مباشر",
"Shared `x` ago": "تم رفع الفيديو منذ `x`", "Shared `x` ago": "تم رفع الفيديو منذ `x`",
"Unsubscribe": "إلغاء الإشتراك", "Unsubscribe": "إلغاء الإشتراك",
"Subscribe": "إشتراك", "Subscribe": "إشتراك",
"Login to subscribe to `x`": "سجل الدخول للإشتراك فى `x`", "View channel on YouTube": "زيارة القناة على موقع يوتيوب",
"View channel on YouTube": "زيارة القناة على موقع يوتيوب", "View playlist on YouTube": "",
"newest": "الأجدد", "newest": "الأجدد",
"oldest": "الأقدم", "oldest": "الأقدم",
"popular": "الاكثر شعبية", "popular": "الاكثر شعبية",
"last": "اخر الفيديوهات المعدلة", "last": "اخر قوائم التشغيل المعدلة",
"Next page": "الصفحة الثانية", "Next page": "الصفحة الثانية",
"Previous page": "الصفحة السابقة", "Previous page": "الصفحة السابقة",
"Clear watch history?": "مسح السجل ؟", "Clear watch history?": "مسح السجل ؟",
"Yes": "نعم", "New password": "الرقم السرى الجديد",
"No": "لا", "New passwords must match": "الأرقام السرية يجب ان تكون متطابقة",
"Import and Export Data": "استخراج و إضافة البيانات", "Cannot change password for Google accounts": "لا يستطيع تغيير الرقم السرى لحساب جوجل",
"Import": "إضافة", "Authorize token?": "رمز الإذن ؟",
"Import Invidious data": "إضافة بيانات Invidious", "Authorize token for `x`?": "رمز الإذن لـ `x` ?",
"Import YouTube subscriptions": "إضافةالإشتراكات من موقع يوتيوب", "Yes": "نعم",
"Import FreeTube subscriptions (.db)": "إضافةالمشتركين من FreeTube (.db)", "No": "لا",
"Import NewPipe subscriptions (.json)": "إضافة المشتركين من NewPipe (.json)", "Import and Export Data": "استخراج و إضافة البيانات",
"Import NewPipe data (.zip)": "إضافة بيانات NewPipe (.zip)", "Import": "إضافة",
"Export": "استخراج", "Import Invidious data": "إضافة بيانات Invidious",
"Export subscriptions as OPML": "استخراج المشتركين كـ OPML", "Import YouTube subscriptions": "إضافةالإشتراكات من موقع يوتيوب",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "استخراج المشتركين كـ OPML (لـ NewPipe و FreeTube)", "Import FreeTube subscriptions (.db)": "إضافةالمشتركين من FreeTube (.db)",
"Export data as JSON": "استخراج البيانات كـ JSON", "Import NewPipe subscriptions (.json)": "إضافة المشتركين من NewPipe (.json)",
"Delete account?": "حذف الحساب ؟", "Import NewPipe data (.zip)": "إضافة بيانات NewPipe (.zip)",
"History": "السجل", "Export": "استخراج",
"An alternative front-end to YouTube": "البديل الكامل لموقع يوتيوب", "Export subscriptions as OPML": "استخراج المشتركين كـ OPML",
"JavaScript license information": "معلومات ترخيص JavaScript", "Export subscriptions as OPML (for NewPipe & FreeTube)": "استخراج المشتركين كـ OPML (لـ NewPipe و FreeTube)",
"source": "المصدر", "Export data as JSON": "استخراج البيانات كـ JSON",
"Login": "تسجيل الدخول", "Delete account?": "حذف الحساب ؟",
"Login/Register": "تسجيل الدخول\\إنشاء حساب", "History": "السجل",
"Login to Google": "تسجيل الدخول بإستخدام جوجل", "An alternative front-end to YouTube": "البديل الكامل لموقع يوتيوب",
"User ID:": "إسم المستخدم:", "JavaScript license information": "معلومات ترخيص JavaScript",
"Password:": "الرقم السرى:", "source": "المصدر",
"Time (h:mm:ss):": "(يجب ان يكتب مثل هذا التنسيق) الوقت (h(ساعات):mm(دقائق):ss(ثوانى)):", "Log in": "تسجيل الدخول",
"Text CAPTCHA": "CAPTCHA كلامية", "Log in/register": "تسجيل الدخول\\إنشاء حساب",
"Image CAPTCHA": "CAPTCHA صورية", "Log in with Google": "تسجيل الدخول بإستخدام جوجل",
"Sign In": "تسجيل الدخول", "User ID": "إسم المستخدم",
"Register": "انشاء الحساب", "Password": "الرقم السرى",
"Email:": "الإيميل:", "Time (h:mm:ss):": "(يجب ان يكتب مثل هذا التنسيق) الوقت (h(ساعات):mm(دقائق):ss(ثوانى)):",
"Google verification code:": "رمز تحقق جوجل:", "Text CAPTCHA": "CAPTCHA كلامية",
"Preferences": "التفضيلات", "Image CAPTCHA": "CAPTCHA صورية",
"Player preferences": "التفضيلات المشغل", "Sign In": "تسجيل الدخول",
"Always loop: ": "كرر الفيديو دائما: ", "Register": "انشاء الحساب",
"Autoplay: ": "تشغيل تلقائى: ", "E-mail": "الإيميل",
"Autoplay next video: ": "شغل الفيديو التالى تلقائى: ", "Google verification code": "رمز تحقق جوجل",
"Listen by default: ": "تشغيل النسخة السمعية تلقائى: ", "Preferences": "التفضيلات",
"Proxy videos? ": "عرض الفيديوهات عن طريق الوكيل(proxy) ؟", "Player preferences": "التفضيلات المشغل",
"Default speed: ": "السرعة الإفتراضية: ", "Always loop: ": "كرر الفيديو دائما: ",
"Preferred video quality: ": "الجودة المفضلة للفيديوهات: ", "Autoplay: ": "تشغيل تلقائى: ",
"Player volume: ": "صوت المشغل: ", "Play next by default: ": "شغل الفيديو التالى تلقائيا",
"Default comments: ": "إضهار التعليقات الإفتراضية لـ: ", "Autoplay next video: ": " شغل الفيديو التالى تلقائيا (فى قوائم التشغيل)",
"youtube": "يوتيوب", "Listen by default: ": "تشغيل النسخة السمعية تلقائى: ",
"reddit": "Reddit", "Proxy videos? ": "عرض الفيديوهات عن طريق الوكيل(proxy) ؟",
"Default captions: ": "الترجمات الإفتراضية: ", "Default speed: ": "السرعة الإفتراضية: ",
"Fallback captions: ": "الترجمات المصاحبة: ", "Preferred video quality: ": "الجودة المفضلة للفيديوهات: ",
"Show related videos? ": "عرض مقاطع الفيديو ذات الصلة؟", "Player volume: ": "صوت المشغل: ",
"Visual preferences": "التفضيلات المرئية", "Default comments: ": "إضهار التعليقات الإفتراضية لـ: ",
"Dark mode: ": "الوضع الليلى: ", "youtube": "يوتيوب",
"Thin mode: ": "الوضع الخفيف: ", "reddit": "Reddit",
"Subscription preferences": "تفضيلات الإشتراك", "Default captions: ": "الترجمات الإفتراضية: ",
"Redirect homepage to feed: ": "إعادة التوجية من الصفحة الرئيسية لصفحة المشتركين (لرؤية اخر فيديوهات المشتركين): ", "Fallback captions: ": "الترجمات المصاحبة: ",
"Number of videos shown in feed: ": "عدد الفيديوهات التى ستظهر فى صفحة المشتركين: ", "Show related videos? ": "عرض مقاطع الفيديو ذات الصلة؟",
"Sort videos by: ": "ترتيب الفيديو بـ: ", "Show annotations by default? ": "عرض الملاحظات فى الفيديو تلقائيا ؟",
"published": "احدث فيديو", "Visual preferences": "التفضيلات المرئية",
"published - reverse": "احدث فيديو - عكسى", "Dark mode: ": "الوضع الليلى: ",
"alphabetically": "ترتيب ابجدى", "Thin mode: ": "الوضع الخفيف: ",
"alphabetically - reverse": "ابجدى - عكسى", "Subscription preferences": "تفضيلات الإشتراك",
"channel name": "بإسم القناة", "Show annotations by default for subscribed channels? ": "عرض الملاحظات فى الفيديوهات تلقائيا فى القنوات المشترك بها فقط ؟",
"channel name - reverse": "بإسم القناة - عكسى", "Redirect homepage to feed: ": "إعادة التوجية من الصفحة الرئيسية لصفحة المشتركين (لرؤية اخر فيديوهات المشتركين): ",
"Only show latest video from channel: ": "فقط إظهر اخر فيديو من القناة: ", "Number of videos shown in feed: ": "عدد الفيديوهات التى ستظهر فى صفحة المشتركين: ",
"Only show latest unwatched video from channel: ": "فقط اظهر اخر فيديو لم يتم رؤيتة من القناة: ", "Sort videos by: ": "ترتيب الفيديو بـ: ",
"Only show unwatched: ": "فقط اظهر الذى لم يتم رؤيتة: ", "published": "احدث فيديو",
"Only show notifications (if there are any): ": "إظهار الإشعارات فقط (إذا كان هناك أي): ", "published - reverse": "احدث فيديو - عكسى",
"Data preferences": "إعدادات التفضيلات", "alphabetically": "ترتيب ابجدى",
"Clear watch history": "حذف سجل المشاهدة", "alphabetically - reverse": "ابجدى - عكسى",
"Import/Export data": "إضافة\\إستخراج البيانات", "channel name": "بإسم القناة",
"Manage subscriptions": "إدارة المشتركين", "channel name - reverse": "بإسم القناة - عكسى",
"Watch history": "سجل المشاهدة", "Only show latest video from channel: ": "فقط إظهر اخر فيديو من القناة: ",
"Delete account": "حذف الحساب", "Only show latest unwatched video from channel: ": "فقط اظهر اخر فيديو لم يتم رؤيتة من القناة: ",
"Administrator preferences": "إعدادات المدير", "Only show unwatched: ": "فقط اظهر الذى لم يتم رؤيتة: ",
"Default homepage: ": "الصفحة الرئيسية الافتراضية ", "Only show notifications (if there are any): ": "إظهار الإشعارات فقط (إذا كان هناك أي): ",
"Feed menu: ": "قائمة التغذية", "Data preferences": "إعدادات التفضيلات",
"Top enabled? ": "تفعيل 'الأفضل' ؟ ", "Clear watch history": "حذف سجل المشاهدة",
"CAPTCHA enabled? ": "تفعيل الكابتشا ؟", "Import/export data": "إضافة\\إستخراج البيانات",
"Login enabled? ": "تفعيل تسجيل الدخول ؟", "Change password": "غير الرقم السرى",
"Registration enabled? ": "تفعيل التسجيل ؟", "Manage subscriptions": "إدارة المشتركين",
"Report statistics? ": "إبلاغ الإحصائيات", "Manage tokens": "إدارة الرموز",
"Save preferences": "حفظ التفضيلات", "Watch history": "سجل المشاهدة",
"Subscription manager": "مدير الإشتراكات", "Delete account": "حذف الحساب",
"`x` subscriptions": "`x` مشتركين", "Administrator preferences": "إعدادات المدير",
"Import/Export": "إضافة\\إستخراج", "Default homepage: ": "الصفحة الرئيسية الافتراضية ",
"unsubscribe": "إلغاء الإشتراك", "Feed menu: ": "قائمة التغذية",
"Subscriptions": "الإشتراكات", "Top enabled? ": "تفعيل 'الأفضل' ؟ ",
"`x` unseen notifications": "`x` إشعارات لم تشاهدها بعد ", "CAPTCHA enabled? ": "تفعيل الكابتشا ؟",
"search": "بحث", "Login enabled? ": "تفعيل تسجيل الدخول ؟",
"Sign out": "تسجيل الخروج", "Registration enabled? ": "تفعيل التسجيل ؟",
"Released under the AGPLv3 by Omar Roth.": "تم الإنشاء تحت AGPLv3 بواسطة عمر روث.", "Report statistics? ": "إبلاغ الإحصائيات",
"Source available here.": "الأكواد متوفرة هنا.", "Save preferences": "حفظ التفضيلات",
"View JavaScript license information.": "مشاهدة معلومات حول تراخيص الجافاسكريبت.", "Subscription manager": "مدير الإشتراكات",
"View privacy policy.": "عرض سياسة الخصوصية", "Token manager": "إداره الرمز",
"Trending": "الشائع", "Token": "الرمز",
"Unlisted": "غير مصنف", "`x` subscriptions": "`x` مشتركين",
"Watch video on Youtube": "مشاهدة الفيديو على اليوتيوب", "`x` tokens": "`x` رموز",
"Genre: ": "النوع: ", "Import/export": "إضافة\\إستخراج",
"License: ": "التراخيص: ", "unsubscribe": "إلغاء الإشتراك",
"Family friendly? ": "محتوى عائلى? ", "revoke": "مسح",
"Wilson score: ": "درجة ويلسون: ", "Subscriptions": "الإشتراكات",
"Engagement: ": "نسبة المشاركة (عدد المشاهدات\\عدد الإعجابات): ", "`x` unseen notifications": "`x` إشعارات لم تشاهدها بعد ",
"Whitelisted regions: ": "الدول المسموح فيها هذا الفيديو: ", "search": "بحث",
"Blacklisted regions: ": "الدول الحظور فيها هذا الفيديو: ", "Log out": "تسجيل الخروج",
"Shared `x`": "شارك منذ `x`", "Released under the AGPLv3 by Omar Roth.": "تم الإنشاء تحت AGPLv3 بواسطة عمر روث.",
"Premieres in `x`": "يعرض فى 'x'", "Source available here.": "الأكواد متوفرة هنا.",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "اهلا! يبدو ان الجافاسكريبت معطلة. اضغط هنا لعرض التعليقات, ضع فى إعتبارك انها ستأخذ وقت اطول للعرض.", "View JavaScript license information.": "مشاهدة معلومات حول تراخيص الجافاسكريبت.",
"View YouTube comments": "عرض تعليقات اليوتيوب", "View privacy policy.": "عرض سياسة الخصوصية",
"View more comments on Reddit": "عرض المزيد من التعليقات على\\من موقع Reddit", "Trending": "الشائع",
"View `x` comments": "عرض `x` تعليقات", "Unlisted": "غير مصنف",
"View Reddit comments": "عرض تعليقات ريدإت Reddit", "Watch on YouTube": "مشاهدة الفيديو على اليوتيوب",
"Hide replies": "إخفاء الردود", "Hide annotations": "إخفاء الملاحظات فى الفيديو",
"Show replies": "عرض الردود", "Show annotations": "عرض الملاحظات فى الفيديو",
"Incorrect password": "الرقم السرى غير صحيح", "Genre: ": "النوع: ",
"Quota exceeded, try again in a few hours": "تم تجاوز عدد المرات المسموح بها, حاول مرة اخرى بعد عدة ساعات", "License: ": "التراخيص: ",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "غير قادر على تسجيل الدخول, تأكد من تشغيل المصادقة الثنائية 2FA.", "Family friendly? ": "محتوى عائلى? ",
"Invalid TFA code": "كود مصادقة ثنائية 2FA غير صحيح", "Wilson score: ": "درجة ويلسون: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "لم يتم تسجيل الدخول. هذا ربما بسبب ان المصادقة الثنائية 2FA معطلة فى حسابك.", "Engagement: ": "نسبة المشاركة (عدد المشاهدات\\عدد الإعجابات): ",
"Invalid answer": "إجابة خاطئة", "Whitelisted regions: ": "الدول المسموح فيها هذا الفيديو: ",
"Invalid CAPTCHA": "الكابتشا CAPTCHA غير صاحلة", "Blacklisted regions: ": "الدول الحظور فيها هذا الفيديو: ",
"CAPTCHA is a required field": "مكان الكابتشا CAPTCHA مطلوب", "Shared `x`": "شارك منذ `x`",
"User ID is a required field": "مكان إسم المستخدم مطلوب", "`x` views": "`x` مشاهدون",
"Password is a required field": "مكان الرقم السرى مطلوب", "Premieres in `x`": "يعرض فى `x`",
"Invalid username or password": "إسم المستخدم او الرقم السرى غير صحيح", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "اهلا! يبدو ان الجافاسكريبت معطلة. اضغط هنا لعرض التعليقات, ضع فى إعتبارك انها ستأخذ وقت اطول للعرض.",
"Please sign in using 'Sign in with Google'": "الرجاء تسجيل الدخول 'تسجيل الدخول بواسطة جوجل'", "View YouTube comments": "عرض تعليقات اليوتيوب",
"Password cannot be empty": "الرقم السرى لايمكن ان يكون فارغ", "View more comments on Reddit": "عرض المزيد من التعليقات على\\من موقع Reddit",
"Password cannot be longer than 55 characters": "الرقم السرى لا يتعدى 55 حرف", "View `x` comments": "عرض `x` تعليقات",
"Please sign in": "الرجاء تسجيل الدخول", "View Reddit comments": "عرض تعليقات ريدإت Reddit",
"Invidious Private Feed for `x`": "صفحة Invidious للمشتركين الخاصة\\مخفية لـ `x`", "Hide replies": "إخفاء الردود",
"channel:`x`": "قناة:`x`", "Show replies": "عرض الردود",
"Deleted or invalid channel": "قناة ممسوحة او غير صالحة", "Incorrect password": "الرقم السرى غير صحيح",
"This channel does not exist.": "القناة غير موجودة.", "Quota exceeded, try again in a few hours": "تم تجاوز عدد المرات المسموح بها, حاول مرة اخرى بعد عدة ساعات",
"Could not get channel info.": "لم يستطع الحصول على معلومات القناة.", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "غير قادر على تسجيل الدخول, تأكد من تشغيل المصادقة الثنائية 2FA.",
"Could not fetch comments": "لم يتمكن من إحضار التعليقات", "Invalid TFA code": "كود مصادقة ثنائية 2FA غير صحيح",
"View `x` replies": "عرض `x` ردود", "Login failed. This may be because two-factor authentication is not turned on for your account.": "لم يتم تسجيل الدخول. هذا ربما بسبب ان المصادقة الثنائية 2FA معطلة فى حسابك.",
"`x` ago": "`x` منذ", "Wrong answer": "إجابة خاطئة",
"Load more": "عرض المزيد", "Erroneous CAPTCHA": "الكابتشا CAPTCHA غير صاحلة",
"`x` points": "`x` نقاط", "CAPTCHA is a required field": "مكان الكابتشا CAPTCHA مطلوب",
"Could not create mix.": "لم يستطع عمل خلط.", "User ID is a required field": "مكان إسم المستخدم مطلوب",
"Playlist is empty": "قائمة التشغيل فارغة", "Password is a required field": "مكان الرقم السرى مطلوب",
"Invalid playlist.": "قائمة التشغيل غير صالحة.", "Wrong username or password": "إسم المستخدم او الرقم السرى غير صحيح",
"Playlist does not exist.": "قائمة التشغيل غير موجودة.", "Please sign in using 'Log in with Google'": "الرجاء تسجيل الدخول 'تسجيل الدخول بواسطة جوجل'",
"Could not pull trending pages.": "لم يستطع عرض الصفحات الراجئة.", "Password cannot be empty": "الرقم السرى لايمكن ان يكون فارغ",
"Hidden field \"challenge\" is a required field": "مكان مخفى \"تحدى\" مكان مطلوب", "Password cannot be longer than 55 characters": "الرقم السرى لا يتعدى 55 حرف",
"Hidden field \"token\" is a required field": "مكان مخفى \"رمز\" مكان مطلوب", "Please log in": "الرجاء تسجيل الدخول",
"Invalid challenge": "تحدى غير صالح", "Invidious Private Feed for `x`": "صفحة Invidious للمشتركين الخاصة\\مخفية لـ `x`",
"Invalid token": "روز غير صالح", "channel:`x`": "قناة:`x`",
"Invalid user": "مستخدم غير صالح", "Deleted or invalid channel": "قناة ممسوحة او غير صالحة",
"Token is expired, please try again": "الرمز منتهى الصلاحية , الرجاء المحاولة مرة اخرى", "This channel does not exist.": "القناة غير موجودة.",
"English": "إنجليزى", "Could not get channel info.": "لم يستطع الحصول على معلومات القناة.",
"English (auto-generated)": "إنجليزى (تم إنشائة تلقائى)", "Could not fetch comments": "لم يتمكن من إحضار التعليقات",
"Afrikaans": "الأفريكانية", "View `x` replies": "عرض `x` ردود",
"Albanian": "الألبانية", "`x` ago": "`x` منذ",
"Amharic": "الأمهرية", "Load more": "عرض المزيد",
"Arabic": "العربية", "`x` points": "`x` نقاط",
"Armenian": "الأرميني", "Could not create mix.": "لم يستطع عمل خلط.",
"Azerbaijani": "أذربيجان", "Empty playlist": "قائمة التشغيل فارغة",
"Bangla": "البنغالية", "Not a playlist.": "قائمة التشغيل غير صالحة.",
"Basque": "الباسكي", "Playlist does not exist.": "قائمة التشغيل غير موجودة.",
"Belarusian": "البيلاروسية", "Could not pull trending pages.": "لم يستطع عرض الصفحات الراجئة.",
"Bosnian": "البوسنية", "Hidden field \"challenge\" is a required field": "مكان مخفى \"تحدى\" مكان مطلوب",
"Bulgarian": "البلغارية", "Hidden field \"token\" is a required field": "مكان مخفى \"رمز\" مكان مطلوب",
"Burmese": "البورمية", "Erroneous challenge": "تحدى غير صالح",
"Catalan": "الكاتالونية", "Erroneous token": "روز غير صالح",
"Cebuano": "السيبيونو", "No such user": "مستخدم غير صالح",
"Chinese (Simplified)": "الصينية (المبسطة)", "Token is expired, please try again": "الرمز منتهى الصلاحية , الرجاء المحاولة مرة اخرى",
"Chinese (Traditional)": "الصينية (التقليدية)", "English": "إنجليزى",
"Corsican": "الكورسيكية", "English (auto-generated)": "إنجليزى (تم إنشائة تلقائى)",
"Croatian": "الكرواتية", "Afrikaans": "الأفريكانية",
"Czech": "تشيكي", "Albanian": "الألبانية",
"Danish": "دانماركي", "Amharic": "الأمهرية",
"Dutch": "هولندي", "Arabic": "العربية",
"Esperanto": "الاسبرانتو", "Armenian": "الأرميني",
"Estonian": "الإستونية", "Azerbaijani": "أذربيجان",
"Filipino": "الفلبينية", "Bangla": "البنغالية",
"Finnish": "الفنلندية", "Basque": "الباسكي",
"French": "الفرنسية", "Belarusian": "البيلاروسية",
"Galician": "الجاليكية", "Bosnian": "البوسنية",
"Georgian": "الجورجية", "Bulgarian": "البلغارية",
"German": "ألمانية", "Burmese": "البورمية",
"Greek": "الإغريقي", "Catalan": "الكاتالونية",
"Gujarati": "الغوجاراتية", "Cebuano": "السيبيونو",
"Haitian Creole": "الكاثوليكية الهايتية", "Chinese (Simplified)": "الصينية (المبسطة)",
"Hausa": "الهوسا", "Chinese (Traditional)": "الصينية (التقليدية)",
"Hawaiian": "هاواي", "Corsican": "الكورسيكية",
"Hebrew": "العبرية", "Croatian": "الكرواتية",
"Hindi": "الهندية", "Czech": "تشيكي",
"Hmong": "همونغ", "Danish": "دانماركي",
"Hungarian": "الهنغارية", "Dutch": "هولندي",
"Icelandic": "أيسلندي", "Esperanto": "الاسبرانتو",
"Igbo": "الإيبو", "Estonian": "الإستونية",
"Indonesian": "الأندونيسية", "Filipino": "الفلبينية",
"Irish": "الأيرلندية", "Finnish": "الفنلندية",
"Italian": "الإيطالي", "French": "الفرنسية",
"Japanese": "اليابانية", "Galician": "الجاليكية",
"Javanese": "جاوي", "Georgian": "الجورجية",
"Kannada": "الكانادا", "German": "ألمانية",
"Kazakh": "الكازاخية", "Greek": "الإغريقي",
"Khmer": "الخمير", "Gujarati": "الغوجاراتية",
"Korean": "الكورية", "Haitian Creole": "الكاثوليكية الهايتية",
"Kurdish": "كردي", "Hausa": "الهوسا",
"Kyrgyz": "قيرغيزستان", "Hawaiian": "هاواي",
"Lao": "لاو", "Hebrew": "العبرية",
"Latin": "لاتينية", "Hindi": "الهندية",
"Latvian": "اللاتفية", "Hmong": "همونغ",
"Lithuanian": "اللتوانية", "Hungarian": "الهنغارية",
"Luxembourgish": "اللوكسمبرجية", "Icelandic": "أيسلندي",
"Macedonian": "المقدونية", "Igbo": "الإيبو",
"Malagasy": "مدجشقر\\مدغشقر", "Indonesian": "الأندونيسية",
"Malay": "الملايو", "Irish": "الأيرلندية",
"Malayalam": "المالايالامية", "Italian": "الإيطالي",
"Maltese": "المالطية", "Japanese": "اليابانية",
"Maori": "الماوري", "Javanese": "جاوي",
"Marathi": "المهاراتية", "Kannada": "الكانادا",
"Mongolian": "المنغولية", "Kazakh": "الكازاخية",
"Nepali": "النيبالية", "Khmer": "الخمير",
"Norwegian": "النرويجية", "Korean": "الكورية",
"Nyanja": "نيانجا", "Kurdish": "كردي",
"Pashto": "الباشتو", "Kyrgyz": "قيرغيزستان",
"Persian": "الفارسية", "Lao": "لاو",
"Polish": "البولندي", "Latin": "لاتينية",
"Portuguese": "البرتغالية", "Latvian": "اللاتفية",
"Punjabi": "البنجابية", "Lithuanian": "اللتوانية",
"Romanian": "روماني", "Luxembourgish": "اللوكسمبرجية",
"Russian": "الروسية", "Macedonian": "المقدونية",
"Samoan": "ساموا", "Malagasy": "مدجشقر\\مدغشقر",
"Scottish Gaelic": "الغيلية الاسكتلندية", "Malay": "الملايو",
"Serbian": "صربي", "Malayalam": "المالايالامية",
"Shona": "شونا", "Maltese": "المالطية",
"Sindhi": "السندية", "Maori": "الماوري",
"Sinhala": "السنهالية", "Marathi": "المهاراتية",
"Slovak": "السلوفاكية", "Mongolian": "المنغولية",
"Slovenian": "سلوفيني", "Nepali": "النيبالية",
"Somali": "الصومالية", "Norwegian Bokmål": "النرويجية",
"Southern Sotho": "جنوب سوثو", "Nyanja": "نيانجا",
"Spanish": "الأسبانية", "Pashto": "الباشتو",
"Spanish (Latin America)": "الأسبانية (أمريكا اللاتينية)", "Persian": "الفارسية",
"Sundanese": "السودانية", "Polish": "البولندي",
"Swahili": "السواحلية", "Portuguese": "البرتغالية",
"Swedish": "السويدية", "Punjabi": "البنجابية",
"Tajik": "الطاجيكية", "Romanian": "روماني",
"Tamil": "التاميل", "Russian": "الروسية",
"Telugu": "التيلجو", "Samoan": "ساموا",
"Thai": "التايلاندية", "Scottish Gaelic": "الغيلية الاسكتلندية",
"Turkish": "التركية", "Serbian": "صربي",
"Ukrainian": "الأوكراني", "Shona": "شونا",
"Urdu": "الأردية", "Sindhi": "السندية",
"Uzbek": "الأوزبكي", "Sinhala": "السنهالية",
"Vietnamese": "الفيتنامية", "Slovak": "السلوفاكية",
"Welsh": "الولزية", "Slovenian": "سلوفيني",
"Western Frisian": "الفريزية الغربية", "Somali": "الصومالية",
"Xhosa": "زوسا", "Southern Sotho": "جنوب سوثو",
"Yiddish": "اليديشية", "Spanish": "الأسبانية",
"Yoruba": "اليوروبا", "Spanish (Latin America)": "الأسبانية (أمريكا اللاتينية)",
"Zulu": "الزولو", "Sundanese": "السودانية",
"`x` years": "`x` سنوات", "Swahili": "السواحلية",
"`x` months": "`x` شهور", "Swedish": "السويدية",
"`x` weeks": "`x` اسابيع", "Tajik": "الطاجيكية",
"`x` days": "`x` ايام", "Tamil": "التاميل",
"`x` hours": "`x` ساعات", "Telugu": "التيلجو",
"`x` minutes": "`x` دقائق", "Thai": "التايلاندية",
"`x` seconds": "`x` ثوانى", "Turkish": "التركية",
"Fallback comments: ": "التعليقات المصاحبة", "Ukrainian": "الأوكراني",
"Popular": "لاكثر شعبية", "Urdu": "الأردية",
"Top": "الأفضل", "Uzbek": "الأوزبكي",
"About": "حول", "Vietnamese": "الفيتنامية",
"Rating: ": "التقييم", "Welsh": "الولزية",
"Language: ": "اللغة", "Western Frisian": "الفريزية الغربية",
"Default": "الكل", "Xhosa": "زوسا",
"Music": "الاغانى", "Yiddish": "اليديشية",
"Gaming": "الألعاب", "Yoruba": "اليوروبا",
"News": "الأخبار", "Zulu": "الزولو",
"Movies": "الأفلام", "`x` years": "`x` سنوات",
"Download as: ": "تحميل كـ", "`x` months": "`x` شهور",
"Download": "تحميل", "`x` weeks": "`x` اسابيع",
"%A %B %-d, %Y": "", "`x` days": "`x` ايام",
"(edited)": "(تم تعديلة)", "`x` hours": "`x` ساعات",
"Youtube permalink of the comment": "رابط التعليق على اليوتيوب", "`x` minutes": "`x` دقائق",
"`x` marked it with a ❤": "'x' اعجب بهذا", "`x` seconds": "`x` ثوانى",
"Audio mode": "الوضع الصوتى", "Fallback comments: ": "التعليقات المصاحبة",
"Video mode": "وضع الفيديو", "Popular": "لاكثر شعبية",
"Videos": "الفيديوهات", "Top": "الأفضل",
"Playlists": "قوائم التشغيل", "About": "حول",
"Current version: ": "الإصدار الحالى" "Rating: ": "التقييم",
} "Language: ": "اللغة",
"View as playlist": "عرض كا قائمة التشغيل",
"Default": "الكل",
"Music": "الاغانى",
"Gaming": "الألعاب",
"News": "الأخبار",
"Movies": "الأفلام",
"Download": "تحميل كـ",
"Download as: ": "تحميل",
"%A %B %-d, %Y": "",
"(edited)": "(تم تعديلة)",
"YouTube comment permalink": "رابط التعليق على اليوتيوب",
"`x` marked it with a ❤": "`x` اعجب بهذا",
"Audio mode": "الوضع الصوتى",
"Video mode": "وضع الفيديو",
"Videos": "الفيديوهات",
"Playlists": "قوائم التشغيل",
"Current version: ": "الإصدار الحالى"
}

View File

@ -1,297 +1,315 @@
{ {
"`x` subscribers": "`x` Abonnenten", "`x` subscribers": "`x` Abonnenten",
"`x` videos": "`x` Videos", "`x` videos": "`x` Videos",
"LIVE": "LIVE", "LIVE": "LIVE",
"Shared `x` ago": "Vor `x` geteilt", "Shared `x` ago": "Vor `x` geteilt",
"Unsubscribe": "Abbestellen", "Unsubscribe": "Abbestellen",
"Subscribe": "Abonnieren", "Subscribe": "Abonnieren",
"Login to subscribe to `x`": "Einloggen um `x` zu abonnieren", "View channel on YouTube": "Kanal auf YouTube anzeigen",
"View channel on YouTube": "Kanal auf YouTube anzeigen", "View playlist on YouTube": "",
"newest": "neueste", "newest": "neueste",
"oldest": "älteste", "oldest": "älteste",
"popular": "beliebt", "popular": "beliebt",
"last": "", "last": "letzte",
"Next page": "Nächste Seite", "Next page": "Nächste Seite",
"Previous page": "Vorherige Seite", "Previous page": "Vorherige Seite",
"Clear watch history?": "Verlauf löschen?", "Clear watch history?": "Verlauf löschen?",
"Yes": "Ja", "New password": "Neues Passwort",
"No": "Nein", "New passwords must match": "Neue Passwörter müssen übereinstimmen",
"Import and Export Data": "Import und Export Daten", "Cannot change password for Google accounts": "Das Passwort für Google -Konten kann nicht geändert werden",
"Import": "Importieren", "Authorize token?": "Token autorisieren?",
"Import Invidious data": "Invidious Daten importieren", "Authorize token for `x`?": "Token für `x` autorisieren?",
"Import YouTube subscriptions": "YouTube Abonnements importieren", "Yes": "Ja",
"Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)", "No": "Nein",
"Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)", "Import and Export Data": "Import und Export Daten",
"Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)", "Import": "Importieren",
"Export": "Exportieren", "Import Invidious data": "Invidious Daten importieren",
"Export subscriptions as OPML": "Abonnements als OPML exportieren", "Import YouTube subscriptions": "YouTube Abonnements importieren",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnements als OPML exportieren (für NewPipe & FreeTube)", "Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)",
"Export data as JSON": "Daten als JSON exportieren", "Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)",
"Delete account?": "Account löschen?", "Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)",
"History": "Verlauf", "Export": "Exportieren",
"An alternative front-end to YouTube": "Eine alternative Oberfläche für YouTube", "Export subscriptions as OPML": "Abonnements als OPML exportieren",
"JavaScript license information": "JavaScript Lizenzinformationen", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnements als OPML exportieren (für NewPipe & FreeTube)",
"source": "Quelle", "Export data as JSON": "Daten als JSON exportieren",
"Login": "Einloggen", "Delete account?": "Account löschen?",
"Login/Register": "Einloggen/Registrieren", "History": "Verlauf",
"Login to Google": "In Google einloggen", "An alternative front-end to YouTube": "Eine alternative Oberfläche für YouTube",
"User ID:": "Benutzer ID:", "JavaScript license information": "JavaScript Lizenzinformationen",
"Password:": "Passwort:", "source": "Quelle",
"Time (h:mm:ss):": "Zeit (h:mm:ss):", "Log in": "Einloggen",
"Text CAPTCHA": "Text CAPTCHA", "Log in/register": "Einloggen/Registrieren",
"Image CAPTCHA": "Image CAPTCHA", "Log in with Google": "In Google einloggen",
"Sign In": "Einloggen", "User ID": "Benutzer ID",
"Register": "Registrieren", "Password": "Passwort",
"Email:": "Email:", "Time (h:mm:ss):": "Zeit (h:mm:ss):",
"Google verification code:": "Google Bestätigungscode:", "Text CAPTCHA": "Text CAPTCHA",
"Preferences": "Einstellungen", "Image CAPTCHA": "Image CAPTCHA",
"Player preferences": "Playereinstellungen", "Sign In": "Einloggen",
"Always loop: ": "Immer wiederholen: ", "Register": "Registrieren",
"Autoplay: ": "Automatisch abspielen: ", "E-mail": "Email",
"Autoplay next video: ": "nächstes Video automatisch abspielen: ", "Google verification code": "Google Bestätigungscode",
"Listen by default: ": "Nur Ton als Standard: ", "Preferences": "Einstellungen",
"Proxy videos? ": "", "Player preferences": "Playereinstellungen",
"Default speed: ": "Standardgeschwindigkeit: ", "Always loop: ": "Immer wiederholen: ",
"Preferred video quality: ": "Bevorzugte Videoqualität: ", "Autoplay: ": "Automatisch abspielen: ",
"Player volume: ": "Playerlautstärke: ", "Play next by default: ": "Standardmäßig als nächstes abspielen: ",
"Default comments: ": "Standardkommentare: ", "Autoplay next video: ": "nächstes Video automatisch abspielen: ",
"youtube": "youtube", "Listen by default: ": "Nur Ton als Standard: ",
"reddit": "reddit", "Proxy videos? ": "Proxy-Videos? ",
"Default captions: ": "Standarduntertitel: ", "Default speed: ": "Standardgeschwindigkeit: ",
"Fallback captions: ": "Ersatzuntertitel: ", "Preferred video quality: ": "Bevorzugte Videoqualität: ",
"Show related videos? ": "Ähnliche Videos anzeigen? ", "Player volume: ": "Playerlautstärke: ",
"Visual preferences": "Anzeigeeinstellungen", "Default comments: ": "Standardkommentare: ",
"Dark mode: ": "Nachtmodus: ", "youtube": "youtube",
"Thin mode: ": "Schlanker Modus: ", "reddit": "reddit",
"Subscription preferences": "Abonnementeinstellungen", "Default captions: ": "Standarduntertitel: ",
"Redirect homepage to feed: ": "Startseite zu Feed umleiten: ", "Fallback captions: ": "Ersatzuntertitel: ",
"Number of videos shown in feed: ": "Anzahl von Videos die im Feed angezeigt werden: ", "Show related videos? ": "Ähnliche Videos anzeigen? ",
"Sort videos by: ": "Videos sortieren nach: ", "Show annotations by default? ": "Standardmäßig Anmerkungen anzeigen? ",
"published": "veröffentlicht", "Visual preferences": "Anzeigeeinstellungen",
"published - reverse": "veröffentlicht - invertiert", "Dark mode: ": "Nachtmodus: ",
"alphabetically": "alphabetisch", "Thin mode: ": "Schlanker Modus: ",
"alphabetically - reverse": "alphabetisch - invertiert", "Subscription preferences": "Abonnementeinstellungen",
"channel name": "Kanalname", "Show annotations by default for subscribed channels? ": "Anmerkungen für abonnierte Kanäle standardmäßig anzeigen? ",
"channel name - reverse": "Kanalname - invertiert", "Redirect homepage to feed: ": "Startseite zu Feed umleiten: ",
"Only show latest video from channel: ": "Nur neueste Videos des Kanals anzeigen: ", "Number of videos shown in feed: ": "Anzahl von Videos die im Feed angezeigt werden: ",
"Only show latest unwatched video from channel: ": "Nur neueste ungesehene Videos des Kanals anzeigen: ", "Sort videos by: ": "Videos sortieren nach: ",
"Only show unwatched: ": "Nur ungesehene anzeigen: ", "published": "veröffentlicht",
"Only show notifications (if there are any): ": "Nur Benachrichtigungen anzeigen (wenn es welche gibt): ", "published - reverse": "veröffentlicht - invertiert",
"Data preferences": "Dateneinstellungen", "alphabetically": "alphabetisch",
"Clear watch history": "Verlauf löschen", "alphabetically - reverse": "alphabetisch - invertiert",
"Import/Export data": "Daten im- exportieren", "channel name": "Kanalname",
"Manage subscriptions": "Abonnements verwalten", "channel name - reverse": "Kanalname - invertiert",
"Watch history": "Verlauf", "Only show latest video from channel: ": "Nur neueste Videos des Kanals anzeigen: ",
"Delete account": "Account löschen", "Only show latest unwatched video from channel: ": "Nur neueste ungesehene Videos des Kanals anzeigen: ",
"Administrator preferences": "", "Only show unwatched: ": "Nur ungesehene anzeigen: ",
"Default homepage: ": "", "Only show notifications (if there are any): ": "Nur Benachrichtigungen anzeigen (wenn es welche gibt): ",
"Feed menu: ": "", "Data preferences": "Dateneinstellungen",
"Top enabled? ": "", "Clear watch history": "Verlauf löschen",
"CAPTCHA enabled? ": "", "Import/export data": "Daten im- exportieren",
"Login enabled? ": "", "Change password": "Passwort ändern",
"Registration enabled? ": "", "Manage subscriptions": "Abonnements verwalten",
"Report statistics? ": "", "Manage tokens": "Token verwalten",
"Save preferences": "Einstellungen speichern", "Watch history": "Verlauf",
"Subscription manager": "Abonnementverwaltung", "Delete account": "Account löschen",
"`x` subscriptions": "`x` Abonnements", "Administrator preferences": "Administratoreinstellungen",
"Import/Export": "Importieren/Exportieren", "Default homepage: ": "Standard-Homepage: ",
"unsubscribe": "abbestellen", "Feed menu: ": "Feed-Menü: ",
"Subscriptions": "Abonnements", "Top enabled? ": "Top aktiviert? ",
"`x` unseen notifications": "`x` ungesehene Benachrichtigungen", "CAPTCHA enabled? ": "CAPTCHA aktiviert? ",
"search": "Suchen", "Login enabled? ": "Login aktiviert? ",
"Sign out": "Abmelden", "Registration enabled? ": "Registrierung aktiviert? ",
"Released under the AGPLv3 by Omar Roth.": "Veröffentlicht unter AGPLv3 von Omar Roth.", "Report statistics? ": "Statistiken berichten? ",
"Source available here.": "Quellcode verfügbar hier.", "Save preferences": "Einstellungen speichern",
"View JavaScript license information.": "Javascript Lizenzinformationen anzeigen.", "Subscription manager": "Abonnementverwaltung",
"View privacy policy.": "", "Token manager": "Token-Manager",
"Trending": "Trending", "Token": "Token",
"Unlisted": "", "`x` subscriptions": "`x` Abonnements",
"Watch video on Youtube": "Video auf YouTube ansehen", "`x` tokens": "`x` Tokens",
"Genre: ": "Genre: ", "Import/export": "Importieren/Exportieren",
"License: ": "Lizenz: ", "unsubscribe": "abbestellen",
"Family friendly? ": "Familienfreundlich? ", "revoke": "widerrufen",
"Wilson score: ": "Wilson-Score: ", "Subscriptions": "Abonnements",
"Engagement: ": "Engagement: ", "`x` unseen notifications": "`x` ungesehene Benachrichtigungen",
"Whitelisted regions: ": "Erlaubte Regionen: ", "search": "Suchen",
"Blacklisted regions: ": "Unerlaubte Regionen: ", "Log out": "Abmelden",
"Shared `x`": "Geteilt `x`", "Released under the AGPLv3 by Omar Roth.": "Veröffentlicht unter AGPLv3 von Omar Roth.",
"Premieres in `x`": "", "Source available here.": "Quellcode verfügbar hier.",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hallo! Anscheinend haben Sie JavaScript deaktiviert. Klicken Sie hier um Kommentare anzuzeigen, beachten sie dass es etwas länger dauern kann um sie zu laden.", "View JavaScript license information.": "Javascript Lizenzinformationen anzeigen.",
"View YouTube comments": "YouTube Kommentare anzeigen", "View privacy policy.": "Datenschutzerklärung einsehen.",
"View more comments on Reddit": "Mehr Kommentare auf Reddit anzeigen", "Trending": "Trending",
"View `x` comments": "`x` Kommentare anzeigen", "Unlisted": "Nicht aufgeführt",
"View Reddit comments": "Reddit Kommentare anzeigen", "Watch on YouTube": "Video auf YouTube ansehen",
"Hide replies": "Antworten verstecken", "Hide annotations": "Anmerkungen ausblenden",
"Show replies": "Antworten anzeigen", "Show annotations": "Anmerkungen anzeigen",
"Incorrect password": "Falsches Passwort", "Genre: ": "Genre: ",
"Quota exceeded, try again in a few hours": "Kontingent überschritten, versuche es in ein paar Stunden erneut", "License: ": "Lizenz: ",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Login nicht möglich, stellen Sie sicher dass two-factor Authentifikation (Authentifizierung oder SMS) aktiviert ist.", "Family friendly? ": "Familienfreundlich? ",
"Invalid TFA code": "Ungültiger TFA Code", "Wilson score: ": "Wilson-Score: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login fehlgeschlagen. Das kann daran liegen dass two-factor Authentifizierung in ihrem Account nicht aktiviert ist.", "Engagement: ": "Engagement: ",
"Invalid answer": "Ungültige Antwort", "Whitelisted regions: ": "Erlaubte Regionen: ",
"Invalid CAPTCHA": "Ungültiges CAPTCHA", "Blacklisted regions: ": "Unerlaubte Regionen: ",
"CAPTCHA is a required field": "CAPTCHA ist eine erforderliche Eingabe", "Shared `x`": "Geteilt `x`",
"User ID is a required field": "Benutzer ID ist eine erforderliche Eingabe", "`x` views": "`x` Ansichten",
"Password is a required field": "Passwort ist eine erforderliche Eingabe", "Premieres in `x`": "Premieren in `x`",
"Invalid username or password": "Ungültiger Benutzername oder Passwort", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hallo! Anscheinend haben Sie JavaScript deaktiviert. Klicken Sie hier um Kommentare anzuzeigen, beachten sie dass es etwas länger dauern kann um sie zu laden.",
"Please sign in using 'Sign in with Google'": "Bitte melden sie sich mit 'Mit Google anmelden' an", "View YouTube comments": "YouTube Kommentare anzeigen",
"Password cannot be empty": "Passwort darf nicht leer sein", "View more comments on Reddit": "Mehr Kommentare auf Reddit anzeigen",
"Password cannot be longer than 55 characters": "Passwort darf nicht länger als 55 Zeichen sein", "View `x` comments": "`x` Kommentare anzeigen",
"Please sign in": "Bitte anmelden", "View Reddit comments": "Reddit Kommentare anzeigen",
"Invidious Private Feed for `x`": "Invidious Persönlicher Feed für `x`", "Hide replies": "Antworten verstecken",
"channel:`x`": "Kanal:`x`", "Show replies": "Antworten anzeigen",
"Deleted or invalid channel": "Gelöschter oder ungültiger Kanal", "Incorrect password": "Falsches Passwort",
"This channel does not exist.": "Dieser Kanal existiert nicht.", "Quota exceeded, try again in a few hours": "Kontingent überschritten, versuche es in ein paar Stunden erneut",
"Could not get channel info.": "Kanalinformationen konnten nicht geladen werden.", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Login nicht möglich, stellen Sie sicher dass two-factor Authentifikation (Authentifizierung oder SMS) aktiviert ist.",
"Could not fetch comments": "Kommentare konnten nicht geladen werden", "Invalid TFA code": "Ungültiger TFA Code",
"View `x` replies": "Zeige `x` Antworten", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Login fehlgeschlagen. Das kann daran liegen dass two-factor Authentifizierung in ihrem Account nicht aktiviert ist.",
"`x` ago": "vor `x`", "Wrong answer": "Ungültige Antwort",
"Load more": "Mehr laden", "Erroneous CAPTCHA": "Ungültiges CAPTCHA",
"`x` points": "`x` Punkte", "CAPTCHA is a required field": "CAPTCHA ist eine erforderliche Eingabe",
"Could not create mix.": "Mix konnte nicht erstellt werden.", "User ID is a required field": "Benutzer ID ist eine erforderliche Eingabe",
"Playlist is empty": "Playlist ist leer", "Password is a required field": "Passwort ist eine erforderliche Eingabe",
"Invalid playlist.": "Ungültige Playlist.", "Wrong username or password": "Ungültiger Benutzername oder Passwort",
"Playlist does not exist.": "Playlist existiert nicht.", "Please sign in using 'Log in with Google'": "Bitte melden sie sich mit 'Mit Google anmelden' an",
"Could not pull trending pages.": "Trending Seiten konnten nicht geladen werden.", "Password cannot be empty": "Passwort darf nicht leer sein",
"Hidden field \"challenge\" is a required field": "Verstecktes Feld \"challenge\" ist eine erforderliche Eingabe", "Password cannot be longer than 55 characters": "Passwort darf nicht länger als 55 Zeichen sein",
"Hidden field \"token\" is a required field": "Verstecktes Feld \"token\" ist eine erforderliche Eingabe", "Please log in": "Bitte anmelden",
"Invalid challenge": "Ungültiger Test", "Invidious Private Feed for `x`": "Invidious Persönlicher Feed für `x`",
"Invalid token": "Ungöltige Marke", "channel:`x`": "Kanal:`x`",
"Invalid user": "Ungültiger Benutzer", "Deleted or invalid channel": "Gelöschter oder ungültiger Kanal",
"Token is expired, please try again": "Marke ist abgelaufen, bitte erneut versuchen", "This channel does not exist.": "Dieser Kanal existiert nicht.",
"English": "Englisch", "Could not get channel info.": "Kanalinformationen konnten nicht geladen werden.",
"English (auto-generated)": "Englisch (automatisch erzeugt)", "Could not fetch comments": "Kommentare konnten nicht geladen werden",
"Afrikaans": "Afrikaans", "View `x` replies": "Zeige `x` Antworten",
"Albanian": "Albanisch", "`x` ago": "vor `x`",
"Amharic": "Amharisch", "Load more": "Mehr laden",
"Arabic": "Arabisch", "`x` points": "`x` Punkte",
"Armenian": "Armenisch", "Could not create mix.": "Mix konnte nicht erstellt werden.",
"Azerbaijani": "Aserbaidschanisch", "Empty playlist": "Playlist ist leer",
"Bangla": "Bengalisch", "Not a playlist.": "Ungültige Playlist.",
"Basque": "Baskisch", "Playlist does not exist.": "Playlist existiert nicht.",
"Belarusian": "Weißrussisch", "Could not pull trending pages.": "Trending Seiten konnten nicht geladen werden.",
"Bosnian": "Bosnisch", "Hidden field \"challenge\" is a required field": "Verstecktes Feld \"challenge\" ist eine erforderliche Eingabe",
"Bulgarian": "Bulgarisch", "Hidden field \"token\" is a required field": "Verstecktes Feld \"token\" ist eine erforderliche Eingabe",
"Burmese": "Burmesisch", "Erroneous challenge": "Ungültiger Test",
"Catalan": "Katalanisch", "Erroneous token": "Ungöltige Marke",
"Cebuano": "Cebuano", "No such user": "Ungültiger Benutzer",
"Chinese (Simplified)": "Chinesisch (vereinfacht)", "Token is expired, please try again": "Marke ist abgelaufen, bitte erneut versuchen",
"Chinese (Traditional)": "Chinesisch (traditionell)", "English": "Englisch",
"Corsican": "Korsisch", "English (auto-generated)": "Englisch (automatisch erzeugt)",
"Croatian": "Kroatisch", "Afrikaans": "Afrikaans",
"Czech": "Tschechisch", "Albanian": "Albanisch",
"Danish": "Dänisch", "Amharic": "Amharisch",
"Dutch": "Niederländisch", "Arabic": "Arabisch",
"Esperanto": "Esperanto", "Armenian": "Armenisch",
"Estonian": "Estnisch", "Azerbaijani": "Aserbaidschanisch",
"Filipino": "Philippinisch", "Bangla": "Bengalisch",
"Finnish": "Finnisch", "Basque": "Baskisch",
"French": "Französisch", "Belarusian": "Weißrussisch",
"Galician": "Galizisch", "Bosnian": "Bosnisch",
"Georgian": "Georgisch", "Bulgarian": "Bulgarisch",
"German": "Deutsch", "Burmese": "Burmesisch",
"Greek": "Griechisch", "Catalan": "Katalanisch",
"Gujarati": "Gujarati", "Cebuano": "Cebuano",
"Haitian Creole": "Haitianisches Kreolisch", "Chinese (Simplified)": "Chinesisch (vereinfacht)",
"Hausa": "Hausa", "Chinese (Traditional)": "Chinesisch (traditionell)",
"Hawaiian": "Hawaiianisch", "Corsican": "Korsisch",
"Hebrew": "Hebräisch", "Croatian": "Kroatisch",
"Hindi": "Hindi", "Czech": "Tschechisch",
"Hmong": "Hmong", "Danish": "Dänisch",
"Hungarian": "Ungarisch", "Dutch": "Niederländisch",
"Icelandic": "Isländisch", "Esperanto": "Esperanto",
"Igbo": "Igbo", "Estonian": "Estnisch",
"Indonesian": "Indonesisch", "Filipino": "Philippinisch",
"Irish": "Irisch", "Finnish": "Finnisch",
"Italian": "Italienisch", "French": "Französisch",
"Japanese": "Japanisch", "Galician": "Galizisch",
"Javanese": "Javanisch", "Georgian": "Georgisch",
"Kannada": "Kannada", "German": "Deutsch",
"Kazakh": "Kasachisch", "Greek": "Griechisch",
"Khmer": "Khmer", "Gujarati": "Gujarati",
"Korean": "Koreanisch", "Haitian Creole": "Haitianisches Kreolisch",
"Kurdish": "Kurdisch", "Hausa": "Hausa",
"Kyrgyz": "Kirgisisch", "Hawaiian": "Hawaiianisch",
"Lao": "Laotisch", "Hebrew": "Hebräisch",
"Latin": "Lateinisch", "Hindi": "Hindi",
"Latvian": "Lettisch", "Hmong": "Hmong",
"Lithuanian": "Litauisch", "Hungarian": "Ungarisch",
"Luxembourgish": "Luxemburgisch", "Icelandic": "Isländisch",
"Macedonian": "Mazedonisch", "Igbo": "Igbo",
"Malagasy": "Madagassisch", "Indonesian": "Indonesisch",
"Malay": "Malaiisch", "Irish": "Irisch",
"Malayalam": "Malayalam", "Italian": "Italienisch",
"Maltese": "Maltesisch", "Japanese": "Japanisch",
"Maori": "Maori", "Javanese": "Javanisch",
"Marathi": "Marathi", "Kannada": "Kannada",
"Mongolian": "Mongolisch", "Kazakh": "Kasachisch",
"Nepali": "Nepalesisch", "Khmer": "Khmer",
"Norwegian": "Norwegisch", "Korean": "Koreanisch",
"Nyanja": "Nyanja", "Kurdish": "Kurdisch",
"Pashto": "Paschtunisch", "Kyrgyz": "Kirgisisch",
"Persian": "Persisch", "Lao": "Laotisch",
"Polish": "Polnisch", "Latin": "Lateinisch",
"Portuguese": "Portugiesisch", "Latvian": "Lettisch",
"Punjabi": "Pandschabi", "Lithuanian": "Litauisch",
"Romanian": "Rumänisch", "Luxembourgish": "Luxemburgisch",
"Russian": "Russisch", "Macedonian": "Mazedonisch",
"Samoan": "Samoanisch", "Malagasy": "Madagassisch",
"Scottish Gaelic": "Schottisches Gälisch", "Malay": "Malaiisch",
"Serbian": "Serbisch", "Malayalam": "Malayalam",
"Shona": "Schona", "Maltese": "Maltesisch",
"Sindhi": "Sindhi", "Maori": "Maori",
"Sinhala": "Singhalesisch", "Marathi": "Marathi",
"Slovak": "Slowakisch", "Mongolian": "Mongolisch",
"Slovenian": "Slowenisch", "Nepali": "Nepalesisch",
"Somali": "Somali", "Norwegian Bokmål": "Norwegisch",
"Southern Sotho": "Südliches Sotho", "Nyanja": "Nyanja",
"Spanish": "Spanisch", "Pashto": "Paschtunisch",
"Spanish (Latin America)": "Spanisch (Lateinamerika)", "Persian": "Persisch",
"Sundanese": "Sundanesisch", "Polish": "Polnisch",
"Swahili": "Suaheli", "Portuguese": "Portugiesisch",
"Swedish": "Schwedisch", "Punjabi": "Pandschabi",
"Tajik": "Tadschikisch", "Romanian": "Rumänisch",
"Tamil": "Tamilisch", "Russian": "Russisch",
"Telugu": "Telugu", "Samoan": "Samoanisch",
"Thai": "Thailändisch", "Scottish Gaelic": "Schottisches Gälisch",
"Turkish": "Türkisch", "Serbian": "Serbisch",
"Ukrainian": "Ukrainisch", "Shona": "Schona",
"Urdu": "Urdu", "Sindhi": "Sindhi",
"Uzbek": "Usbekisch", "Sinhala": "Singhalesisch",
"Vietnamese": "Vietnamesisch", "Slovak": "Slowakisch",
"Welsh": "Walisisch", "Slovenian": "Slowenisch",
"Western Frisian": "Westfriesisch", "Somali": "Somali",
"Xhosa": "Xhosa", "Southern Sotho": "Südliches Sotho",
"Yiddish": "Jiddisch", "Spanish": "Spanisch",
"Yoruba": "Joruba", "Spanish (Latin America)": "Spanisch (Lateinamerika)",
"Zulu": "Zulu", "Sundanese": "Sundanesisch",
"`x` years": "`x` Jahre", "Swahili": "Suaheli",
"`x` months": "`x` Monate", "Swedish": "Schwedisch",
"`x` weeks": "`x` Wochen", "Tajik": "Tadschikisch",
"`x` days": "`x` Tage", "Tamil": "Tamilisch",
"`x` hours": "`x` Stunden", "Telugu": "Telugu",
"`x` minutes": "`x` Minuten", "Thai": "Thailändisch",
"`x` seconds": "`x` Sekunden", "Turkish": "Türkisch",
"Fallback comments: ": "Alternative Kommentare: ", "Ukrainian": "Ukrainisch",
"Popular": "Populär", "Urdu": "Urdu",
"Top": "Top", "Uzbek": "Usbekisch",
"About": "Über", "Vietnamese": "Vietnamesisch",
"Rating: ": "Bewertung: ", "Welsh": "Walisisch",
"Language: ": "Sprache: ", "Western Frisian": "Westfriesisch",
"Default": "", "Xhosa": "Xhosa",
"Music": "", "Yiddish": "Jiddisch",
"Gaming": "", "Yoruba": "Joruba",
"News": "", "Zulu": "Zulu",
"Movies": "", "`x` years": "`x` Jahre",
"Download": "", "`x` months": "`x` Monate",
"Download as: ": "", "`x` weeks": "`x` Wochen",
"%A %B %-d, %Y": "", "`x` days": "`x` Tage",
"(edited)": "", "`x` hours": "`x` Stunden",
"Youtube permalink of the comment": "", "`x` minutes": "`x` Minuten",
"`x` marked it with a ❤": "", "`x` seconds": "`x` Sekunden",
"Audio mode": "", "Fallback comments: ": "Alternative Kommentare: ",
"Video mode": "", "Popular": "Populär",
"Videos": "", "Top": "Top",
"Playlists": "", "About": "Über",
"Current version: ": "" "Rating: ": "Bewertung: ",
} "Language: ": "Sprache: ",
"View as playlist": "Als Wiedergabeliste anzeigen",
"Default": "Standard",
"Music": "Musik",
"Gaming": "Videospiele",
"News": "Neuigkeiten",
"Movies": "Filme",
"Download": "Herunterladen",
"Download as: ": "Herunterladen als: ",
"%A %B %-d, %Y": "%A %B %-d, %Y",
"(edited)": "(editiert)",
"YouTube comment permalink": "YouTube-Kommentar Permalink",
"`x` marked it with a ❤": "`x` markierte es mit einem ❤",
"Audio mode": "Audiomodus",
"Video mode": "Videomodus",
"Videos": "Videos",
"Playlists": "Wiedergabelisten",
"Current version: ": "Aktuelle Version: "
}

360
locales/el.json Normal file
View File

@ -0,0 +1,360 @@
{
"`x` subscribers": {
"(\\D|^)1(\\D|$)": "`x` συνδρομητής",
"": "`x` συνδρομητές"
},
"`x` videos": {
"(\\D|^)1(\\D|$)": "`x` βίντεο",
"": "`x` βίντεο"
},
"LIVE": "ΖΩΝΤΑΝΑ",
"Shared `x` ago": "Μοιράστηκε πριν `x`",
"Unsubscribe": "Απεγγραφή",
"Subscribe": "Εγγραφή",
"View channel on YouTube": "Προβολή καναλιού στο YouTube",
"View playlist on YouTube": "",
"newest": "νεότερα",
"oldest": "παλιότερα",
"popular": "δημοφιλή",
"last": "τελευταία",
"Next page": "Επόμενη σελίδα",
"Previous page": "Προηγούμενη σελίδα",
"Clear watch history?": "Διαγραφή ιστορικού προβολής;",
"New password": "Νέος κωδικός πρόσβασης",
"New passwords must match": "Οι νέοι κωδικοί πρόσβασης πρέπει να ταιριάζουν",
"Cannot change password for Google accounts": "Δεν επιτρέπεται η αλλαγή κωδικού πρόσβασης λογαριασμών Google",
"Authorize token?": "Εξουσιοδότηση διασύνδεσης;",
"Authorize token for `x`?": "Εξουσιοδότηση διασύνδεσης με `x`;",
"Yes": "Ναι",
"No": "Όχι",
"Import and Export Data": "Εισαγωγή και Εξαγωγή Δεδομένων",
"Import": "Εισαγωγή",
"Import Invidious data": "Εισαγωγή δεδομένων Invidious",
"Import YouTube subscriptions": "Εισαγωγή συνδρομών YouTube",
"Import FreeTube subscriptions (.db)": "Εισαγωγή συνδρομών FreeTube (.db)",
"Import NewPipe subscriptions (.json)": "Εισαγωγή συνδρομών NewPipe (.json)",
"Import NewPipe data (.zip)": "Εισαγωγή δεδομένων NewPipe (.zip)",
"Export": "Εξαγωγή",
"Export subscriptions as OPML": "Εξαγωγή συνδρομών ως OPML",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Εξαγωγή συνδρομών ως OPML (για NewPipe & FreeTube)",
"Export data as JSON": "Εξαγωγή δεδομένων ως JSON",
"Delete account?": "Διαγραφή λογαριασμού;",
"History": "Ιστορικό",
"An alternative front-end to YouTube": "Μία εναλλακτική πλατφόρμα για το YouTube",
"JavaScript license information": "Πληροφορίες άδειας JavaScript",
"source": "πηγή",
"Log in": "Σύνδεση",
"Log in/register": "Σύνδεση/εγγραφή",
"Log in with Google": "Σύνδεση με Google",
"User ID": "Ταυτότητα χρήστη",
"Password": "Κωδικός πρόσβασης",
"Time (h:mm:ss):": "Ώρα (ω:λλ:δδ):",
"Text CAPTCHA": "Κείμενο CAPTCHA",
"Image CAPTCHA": "Εικόνα CAPTCHA",
"Sign In": "Σύνδεση",
"Register": "Εγγραφή",
"E-mail": "E-mail",
"Google verification code": "Κωδικός επαλήθευσης Google",
"Preferences": "Προτιμήσεις",
"Player preferences": "Προτιμήσεις αναπαραγωγής",
"Always loop: ": "Αυτόματη επανάληψη: ",
"Autoplay: ": "Αυτόματη αναπαραγωγή: ",
"Play next by default: ": "Αναπαραγωγή επόμενου: ",
"Autoplay next video: ": "Αυτόματη αναπαραγωγή επόμενου: ",
"Listen by default: ": "Φόρτωση μόνο ήχου: ",
"Proxy videos? ": "Αναπαραγωγή με διακομιστή μεσολάβησης (proxy): ",
"Default speed: ": "Προεπιλεγμένη ταχύτητα: ",
"Preferred video quality: ": "Προτιμώμενη ανάλυση: ",
"Player volume: ": "Ένταση αναπαραγωγής: ",
"Default comments: ": "Προεπιλεγμένα σχόλια: ",
"youtube": "youtube",
"reddit": "reddit",
"Default captions: ": "Προεπιλεγμένοι υπότιτλοι: ",
"Fallback captions: ": "Εναλλακτικοί υπότιτλοι: ",
"Show related videos? ": "Προβολή σχετικών βίντεο; ",
"Show annotations by default? ": "Αυτόματη προβολή σημειώσεων; :",
"Visual preferences": "Προτιμήσεις εμφάνισης",
"Dark mode: ": "Σκοτεινή λειτουργία: ",
"Thin mode: ": "Ελαφριά λειτουργία: ",
"Subscription preferences": "Προτιμήσεις συνδρομών",
"Show annotations by default for subscribed channels? ": "Προβολή σημειώσεων μόνο για κανάλια στα οποία είστε συνδρομητής; ",
"Redirect homepage to feed: ": "Ανακατεύθυνση αρχικής στη ροή συνδρομών: ",
"Number of videos shown in feed: ": "Αριθμός βίντεο ανά σελίδα ροής συνδρομών: ",
"Sort videos by: ": "Ταξινόμηση ανά: ",
"published": "ημερομηνία δημοσίευσης",
"published - reverse": "ημερομηνία δημοσίευσης - ανάποδα",
"alphabetically": "αλφαβητικά",
"alphabetically - reverse": "αλφαβητικά - ανάποδα",
"channel name": "όνομα καναλιού",
"channel name - reverse": "όνομα καναλιού - ανάποδα",
"Only show latest video from channel: ": "Προβολή μόνο του τελευταίου βίντεο του καναλιού: ",
"Only show latest unwatched video from channel: ": "Προβολή μόνο του τελευταίου μη-προβεβλημένου βίντεο του καναλιού: ",
"Only show unwatched: ": "Προβολή μόνο μη-προβεβλημένων: ",
"Only show notifications (if there are any): ": "Προβολή μόνο ειδοποιήσεων (αν υπάρχουν): ",
"Data preferences": "Προτιμήσεις δεδομένων",
"Clear watch history": "Εκκαθάριση ιστορικού προβολής",
"Import/export data": "Εισαγωγή/εξαγωγή δεδομένων",
"Change password": "Αλλαγή κωδικού πρόσβασης",
"Manage subscriptions": "Διαχείριση συνδρομών",
"Manage tokens": "Διαχείριση διασυνδέσεων",
"Watch history": "Ιστορικό προβολής",
"Delete account": "Διαγραφή λογαριασμού",
"Administrator preferences": "Προτιμήσεις διαχειριστή",
"Default homepage: ": "Προεπιλεγμένη αρχική: ",
"Feed menu: ": "Μενού ροής συνδρομών: ",
"Top enabled? ": "Ενεργοποίηση κορυφαίων; ",
"CAPTCHA enabled? ": "Ενεργοποίηση CAPTCHA; ",
"Login enabled? ": "Ενεργοποίηση σύνδεσης; ",
"Registration enabled? ": "Ενεργοποίηση εγγραφής; ",
"Report statistics? ": "Αναφορά στατιστικών; ",
"Save preferences": "Αποθήκευση προτιμήσεων",
"Subscription manager": "Διαχειριστής συνδρομών",
"Token manager": "Διαχειριστής διασυνδέσεων",
"Token": "Διασύνδεση",
"`x` subscriptions": {
"(\\D|^)1(\\D|$)": "`x` συνδρομή",
"": "`x` συνδρομές"
},
"`x` tokens": {
"(\\D|^)1(\\D|$)": "`x` διασύνδεση",
"": "`x` διασυνδέσεις"
},
"Import/export": "Εισαγωγή/εξαγωγή",
"unsubscribe": "κατάργηση συνδρομής",
"revoke": "ανάκληση",
"Subscriptions": "Συνδρομές",
"`x` unseen notifications": {
"(\\D|^)1(\\D|$)": "`x` καινούρια ειδοποίηση",
"": "`x` καινούριες ειδοποιήσεις"
},
"search": "αναζήτηση",
"Log out": "Αποσύνδεση",
"Released under the AGPLv3 by Omar Roth.": "Κυκλοφορεί υπό την άδεια AGPLv3 από τον Omar Roth.",
"Source available here.": "Προβολή πηγαίου κώδικα εδώ.",
"View JavaScript license information.": "Προβολή πληροφοριών άδειας JavaScript.",
"View privacy policy.": "Προβολή πολιτικής απορρήτου.",
"Trending": "Τάσεις",
"Unlisted": "Κρυφό",
"Watch on YouTube": "Προβολή στο YouTube",
"Hide annotations": "Απόκρυψη σημειώσεων",
"Show annotations": "Προβολή σημειώσεων",
"Genre: ": "Είδος: ",
"License: ": "Άδεια: ",
"Family friendly? ": "Φιλικό προς την οικογένεια; ",
"Wilson score: ": "Wilson score: ",
"Engagement: ": "Ενδιαφέρον: ",
"Whitelisted regions: ": "Επιτρεπτές περιοχές: ",
"Blacklisted regions: ": "Μη-επιτρεπτές περιοχές: ",
"Shared `x`": "Μοιράστηκε το `x`",
"`x` views": {
"(\\D|^)1(\\D|$)": "`x` προβολή",
"": "`x` προβολές"
},
"Premieres in `x`": "Πρώτη προβολή σε `x`",
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Γεια! Φαίνεται πως έχετε απενεργοποιήσει το JavaScript. Πατήστε εδώ για προβολή σχολίων, αλλά έχετε υπ'όψιν σας πως ίσως φορτώσουν πιο αργά. ",
"View YouTube comments": "Προβολή σχολίων από το YouTube",
"View more comments on Reddit": "Προβολή περισσότερων σχολίων στο Reddit",
"View `x` comments": "Προβολή `x` σχολίων",
"View Reddit comments": "Προβολή σχολίων από το Reddit",
"Hide replies": "Απόκρυψη απαντήσεων",
"Show replies": "Προβολή απαντήσεων",
"Incorrect password": "Λανθασμένος κωδικός πρόσβασης",
"Quota exceeded, try again in a few hours": "Έχετε υπερβεί το όριο προσπαθειών, δοκιμάστε ξανα σε λίγες ώρες",
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Αδυναμία σύνδεσης, βεβαιωθείτε πως ο έλεγχος ταυτότητας δύο παραγόντων (με Authenticator ή SMS) είναι ενεργοποιημένος.",
"Invalid TFA code": "Μη έγκυρος κωδικός ελέγχου ταυτότητας δύο παραγόντων",
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Αποτυχία σύνδεσης. Ίσως ευθύνεται η έλλειψη ελέγχου ταυτότητας δύο παραγόντων για το λογαριασμό σας.",
"Wrong answer": "Λανθασμένη απάντηση",
"Erroneous CAPTCHA": "Λανθασμένο CAPTCHA",
"CAPTCHA is a required field": "Το CAPTCHA είναι απαιτούμενο πεδίο",
"User ID is a required field": "Η ταυτότητα χρήστη είναι απαιτούμενο πεδίο",
"Password is a required field": "Ο κωδικός πρόσβασης είναι απαιτούμενο πεδίο",
"Wrong username or password": "Λανθασμένο όνομα χρήστη ή κωδικός πρόσβασης",
"Please sign in using 'Log in with Google'": "Συνδεθείτε με την επιλογή 'Σύνδεση με Google'",
"Password cannot be empty": "Ο κωδικός πρόσβασης δεν γίνεται να είναι κενός",
"Password cannot be longer than 55 characters": "Ο κωδικός πρόσβασης δεν γίνεται να υπερβαίνει τους 55 χαρακτήρες",
"Please log in": "Συνδεθείτε",
"Invidious Private Feed for `x`": "Ροή RSS του Invidious για το χρήστη `x`",
"channel:`x`": "κανάλι:`x`",
"Deleted or invalid channel": "Διαγραμμένο ή μη έγκυρο κανάλι",
"This channel does not exist.": "Αυτό το κανάλι δεν υπάρχει.",
"Could not get channel info.": "Αδύναμια εύρεσης πληροφοριών καναλιού.",
"Could not fetch comments": "Αδυναμία λήψης σχολίων",
"View `x` replies": {
"(\\D|^)1(\\D|$)": "Προβολή `x` απάντησης",
"": "Προβολή `x` απαντήσεων"
},
"`x` ago": "Πριν `x`",
"Load more": "Φόρτωση περισσότερων",
"`x` points": {
"(\\D|^)1(\\D|$)": "`x` βαθμός",
"": "`x` βαθμοί"
},
"Could not create mix.": "Αδυναμία δημιουργίας μίξης.",
"Empty playlist": "Κενή λίστα αναπαραγωγής",
"Not a playlist.": "Μη έγκυρη λίστα αναπαραγωγής",
"Playlist does not exist.": "Μη υπαρκτή λίστα αναπαραγωγής.",
"Could not pull trending pages.": "Αδυναμία λήψης σελίδας τάσεων.",
"Hidden field \"challenge\" is a required field": "Το Κρυφό πεδίο \"δοκιμασία\" είναι απαραίτητο",
"Hidden field \"token\" is a required field": "Το κρυφό πεδίο \"αναγνωριστικό διασύνδεσης\" είναι απαραίτητο",
"Erroneous challenge": "Λανθασμένη δοκιμασία",
"Erroneous token": "Λανθασμένο αναγνωριστικό διασύνδεσης",
"No such user": "Μη υπαρκτός χρήστης",
"Token is expired, please try again": "Το αναγνωριστικό διασύνδεσης έχει λήξει, παρακαλώ ξαναπροσπαθήστε",
"English": "Αγγλικά",
"English (auto-generated)": "Αγγλικά (αυτόματα)",
"Afrikaans": "Αφρικάανς",
"Albanian": "Αλβανικά",
"Amharic": "Αμχαρικά",
"Arabic": "Αραβικά",
"Armenian": "Αρμένικα",
"Azerbaijani": "Αζερικά",
"Bangla": "Μπενγκάλι",
"Basque": "Βασκικά",
"Belarusian": "Λευκορωσικά",
"Bosnian": "Βοσνιακά",
"Bulgarian": "Βουλγάρικα",
"Burmese": "Βιρμανικά",
"Catalan": "Καταλανικά",
"Cebuano": "Κεμπουάνο",
"Chinese (Simplified)": "Κινέζικα (Απλοποιημένα)",
"Chinese (Traditional)": "Κινέζικα (Παραδοσιακά)",
"Corsican": "Κορσικανικά",
"Croatian": "Κροατικά",
"Czech": "Τσέχικα",
"Danish": "Δανέζικα",
"Dutch": "Ολλανδικά",
"Esperanto": "Εσπεράντο",
"Estonian": "Εσθονικά",
"Filipino": "Φιλιππινέζικα",
"Finnish": "Φινλανδικά",
"French": "Γαλλικά",
"Galician": "Γαλικιακά",
"Georgian": "Γεωργιανά",
"German": "Γερμανικά",
"Greek": "Ελληνικά",
"Gujarati": "Γκουτζαρατικά",
"Haitian Creole": "Κρεόλ Αϊτής",
"Hausa": "Χάουσα",
"Hawaiian": "Χαβανέζικα",
"Hebrew": "Εβραϊκά",
"Hindi": "Χίντι",
"Hmong": "Χμονγκ",
"Hungarian": "Ουγγαρέζικα",
"Icelandic": "Ισλανδικά",
"Igbo": "Ιγκμπό",
"Indonesian": "Ινδονησιακά",
"Irish": "Ιρλανδικά",
"Italian": "Ιταλικά",
"Japanese": "Ιαπωνικά",
"Javanese": "Ιαβανέζικα",
"Kannada": "Κανάντα",
"Kazakh": "Καζακικά",
"Khmer": "Χμερ",
"Korean": "Κορεάτικα",
"Kurdish": "Κούρδικα",
"Kyrgyz": "Κιργιστανικά",
"Lao": "Lao",
"Latin": "Λατινικά",
"Latvian": "Λετονικά",
"Lithuanian": "Λιθουανικά",
"Luxembourgish": "Λουξεμβουργιανά",
"Macedonian": "Μακεδονικά",
"Malagasy": "Μαλαγασικά",
"Malay": "Μαλαισιανά",
"Malayalam": "Μαλαγιαλάμ",
"Maltese": "Μαλτέζικα",
"Maori": "Μαορί",
"Marathi": "Μαράτι",
"Mongolian": "Μογγολικά",
"Nepali": "Νεπαλικά",
"Norwegian Bokmål": "Νορβηγικά Μποκμάλ",
"Nyanja": "Νιάντζα",
"Pashto": "Αφγανικά",
"Persian": "Περσικά",
"Polish": "Πολωνικά",
"Portuguese": "Πορτογαλικά",
"Punjabi": "Παντζάμπι",
"Romanian": "Ρουμανικά",
"Russian": "Ρώσικα",
"Samoan": "Σαμόα",
"Scottish Gaelic": "Σκωτικά Γαελικά",
"Serbian": "Σέρβικα",
"Shona": "Σόνα",
"Sindhi": "Σίντι",
"Sinhala": "Σιναλεζικά",
"Slovak": "Σλοβακικά",
"Slovenian": "ΣΛοβενικά",
"Somali": "Σομαλικά",
"Southern Sotho": "Νότια Σούτου",
"Spanish": "Ισπανικά",
"Spanish (Latin America)": "Ισπανικά (Λατινική Αμερική)",
"Sundanese": "Σουντανέζικα",
"Swahili": "Σουαχίλι",
"Swedish": "Σουηδικά",
"Tajik": "Τατζικικά",
"Tamil": "Ταμίλ",
"Telugu": "Τελούγκου",
"Thai": "Ταϊλανδικά",
"Turkish": "Τούρκικα",
"Ukrainian": "Ουκρανικά",
"Urdu": "Ουρντού",
"Uzbek": "Ουζμπεκικά",
"Vietnamese": "Βιετναμέζικα",
"Welsh": "Ουαλικά",
"Western Frisian": "Δυτική Φριζική",
"Xhosa": "Xhosa",
"Yiddish": "Γίντις",
"Yoruba": "Γιορούμπα",
"Zulu": "Ζουλού",
"`x` years": {
"(\\D|^)1(\\D|$)": "`x` χρόνο",
"": "`x` χρόνια"
},
"`x` months": {
"(\\D|^)1(\\D|$)": "`x` μήνα",
"": "`x` μήνες"
},
"`x` weeks": {
"(\\D|^)1(\\D|$)": "`x` εβδομάδα",
"": "`x` εβδομάδες"
},
"`x` days": {
"(\\D|^)1(\\D|$)": "`x` ημέρα",
"": "`x` ημέρες"
},
"`x` hours": {
"(\\D|^)1(\\D|$)": "`x` ώρα",
"": "`x` ώρες"
},
"`x` minutes": {
"(\\D|^)1(\\D|$)": "`x` λεπτό",
"": "`x` λεπτά"
},
"`x` seconds": {
"(\\D|^)1(\\D|$)": "`x` δευτερόλεπτο",
"": "`x` δευτερόλεπτα"
},
"Fallback comments: ": "Εναλλακτικά σχόλια: ",
"Popular": "Δημοφιλή",
"Top": "Κορυφαία",
"About": "Σχετικά",
"Rating: ": "Aξιολόγηση: ",
"Language: ": "Γλώσσα: ",
"View as playlist": "Προβολή ως λίστα αναπαραγωγής",
"Default": "Προεπιλογή",
"Music": "Μουσική",
"Gaming": "Παιχνίδια",
"News": "Ειδήσεις",
"Movies": "Ταινίες",
"Download": "Λήψη",
"Download as: ": "Λήψη ως: ",
"%A %B %-d, %Y": "%A %B %-d, %Y",
"(edited)": "(τροποποιημένο)",
"YouTube comment permalink": "Σύνδεσμος YouTube σχολίου",
"`x` marked it with a ❤": "Ο χρηστης `x` έβαλε ❤",
"Audio mode": "Λειτουργία ήχου",
"Video mode": "Λειτουργία βίντεο",
"Videos": "Βίντεο",
"Playlists": "Λίστες Αναπαραγωγής",
"Current version: ": "Τρέχουσα έκδοση: "
}

View File

@ -1,295 +1,360 @@
{ {
"`x` subscribers": "`x` subscribers", "`x` subscribers": {
"`x` videos": "`x` videos", "(\\D|^)1(\\D|$)": "`x` subscriber",
"LIVE": "LIVE", "": "`x` subscribers"
"Shared `x` ago": "Shared `x` ago", },
"Unsubscribe": "Unsubscribe", "`x` videos": {
"Subscribe": "Subscribe", "(\\D|^)1(\\D|$)": "`x` video",
"Login to subscribe to `x`": "Login to subscribe to `x`", "": "`x` videos"
"View channel on YouTube": "View channel on YouTube", },
"newest": "newest", "LIVE": "LIVE",
"oldest": "oldest", "Shared `x` ago": "Shared `x` ago",
"popular": "popular", "Unsubscribe": "Unsubscribe",
"last": "last", "Subscribe": "Subscribe",
"Next page": "Next page", "View channel on YouTube": "View channel on YouTube",
"Previous page": "Previous page", "View playlist on YouTube": "View playlist on YouTube",
"Clear watch history?": "Clear watch history?", "newest": "newest",
"Yes": "Yes", "oldest": "oldest",
"No": "No", "popular": "popular",
"Import and Export Data": "Import and Export Data", "last": "last",
"Import": "Import", "Next page": "Next page",
"Import Invidious data": "Import Invidious data", "Previous page": "Previous page",
"Import YouTube subscriptions": "Import YouTube subscriptions", "Clear watch history?": "Clear watch history?",
"Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)", "New password": "New password",
"Import NewPipe subscriptions (.json)": "Import NewPipe subscriptions (.json)", "New passwords must match": "New passwords must match",
"Import NewPipe data (.zip)": "Import NewPipe data (.zip)", "Cannot change password for Google accounts": "Cannot change password for Google accounts",
"Export": "Export", "Authorize token?": "Authorize token?",
"Export subscriptions as OPML": "Export subscriptions as OPML", "Authorize token for `x`?": "Authorize token for `x`?",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Export subscriptions as OPML (for NewPipe & FreeTube)", "Yes": "Yes",
"Export data as JSON": "Export data as JSON", "No": "No",
"Delete account?": "Delete account?", "Import and Export Data": "Import and Export Data",
"History": "History", "Import": "Import",
"An alternative front-end to YouTube": "An alternative front-end to YouTube", "Import Invidious data": "Import Invidious data",
"JavaScript license information": "JavaScript license information", "Import YouTube subscriptions": "Import YouTube subscriptions",
"source": "source", "Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)",
"Login": "Login", "Import NewPipe subscriptions (.json)": "Import NewPipe subscriptions (.json)",
"Login/Register": "Login/Register", "Import NewPipe data (.zip)": "Import NewPipe data (.zip)",
"Login to Google": "Login to Google", "Export": "Export",
"User ID:": "User ID:", "Export subscriptions as OPML": "Export subscriptions as OPML",
"Password:": "Password:", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Export subscriptions as OPML (for NewPipe & FreeTube)",
"Time (h:mm:ss):": "Time (h:mm:ss):", "Export data as JSON": "Export data as JSON",
"Text CAPTCHA": "Text CAPTCHA", "Delete account?": "Delete account?",
"Image CAPTCHA": "Image CAPTCHA", "History": "History",
"Sign In": "Sign In", "An alternative front-end to YouTube": "An alternative front-end to YouTube",
"Register": "Register", "JavaScript license information": "JavaScript license information",
"Email:": "Email:", "source": "source",
"Google verification code:": "Google verification code:", "Log in": "Log in",
"Preferences": "Preferences", "Log in/register": "Log in/register",
"Player preferences": "Player preferences", "Log in with Google": "Log in with Google",
"Always loop: ": "Always loop: ", "User ID": "User ID",
"Autoplay: ": "Autoplay: ", "Password": "Password",
"Autoplay next video: ": "Autoplay next video: ", "Time (h:mm:ss):": "Time (h:mm:ss):",
"Listen by default: ": "Listen by default: ", "Text CAPTCHA": "Text CAPTCHA",
"Proxy videos? ": "Proxy videos? ", "Image CAPTCHA": "Image CAPTCHA",
"Default speed: ": "Default speed: ", "Sign In": "Sign In",
"Preferred video quality: ": "Preferred video quality: ", "Register": "Register",
"Player volume: ": "Player volume: ", "E-mail": "E-mail",
"Default comments: ": "Default comments: ", "Google verification code": "Google verification code",
"Default captions: ": "Default captions: ", "Preferences": "Preferences",
"Fallback captions: ": "Fallback captions: ", "Player preferences": "Player preferences",
"Show related videos? ": "Show related videos? ", "Always loop: ": "Always loop: ",
"Visual preferences": "Visual preferences", "Autoplay: ": "Autoplay: ",
"Dark mode: ": "Dark mode: ", "Play next by default: ": "Play next by default: ",
"Thin mode: ": "Thin mode: ", "Autoplay next video: ": "Autoplay next video: ",
"Subscription preferences": "Subscription preferences", "Listen by default: ": "Listen by default: ",
"Redirect homepage to feed: ": "Redirect homepage to feed: ", "Proxy videos? ": "Proxy videos? ",
"Number of videos shown in feed: ": "Number of videos shown in feed: ", "Default speed: ": "Default speed: ",
"Sort videos by: ": "Sort videos by: ", "Preferred video quality: ": "Preferred video quality: ",
"published": "published", "Player volume: ": "Player volume: ",
"published - reverse": "published - reverse", "Default comments: ": "Default comments: ",
"alphabetically": "alphabetically", "youtube": "youtube",
"alphabetically - reverse": "alphabetically - reverse", "reddit": "reddit",
"channel name": "channel name", "Default captions: ": "Default captions: ",
"channel name - reverse": "channel name - reverse", "Fallback captions: ": "Fallback captions: ",
"Only show latest video from channel: ": "Only show latest video from channel: ", "Show related videos? ": "Show related videos? ",
"Only show latest unwatched video from channel: ": "Only show latest unwatched video from channel: ", "Show annotations by default? ": "Show annotations by default? ",
"Only show unwatched: ": "Only show unwatched: ", "Visual preferences": "Visual preferences",
"Only show notifications (if there are any): ": "Only show notifications (if there are any): ", "Dark mode: ": "Dark mode: ",
"Data preferences": "Data preferences", "Thin mode: ": "Thin mode: ",
"Clear watch history": "Clear watch history", "Subscription preferences": "Subscription preferences",
"Import/Export data": "Import/Export data", "Show annotations by default for subscribed channels? ": "Show annotations by default for subscribed channels? ",
"Manage subscriptions": "Manage subscriptions", "Redirect homepage to feed: ": "Redirect homepage to feed: ",
"Watch history": "Watch history", "Number of videos shown in feed: ": "Number of videos shown in feed: ",
"Delete account": "Delete account", "Sort videos by: ": "Sort videos by: ",
"Administrator preferences": "Administrator preferences", "published": "published",
"Default homepage: ": "Default homepage: ", "published - reverse": "published - reverse",
"Feed menu: ": "Feed menu: ", "alphabetically": "alphabetically",
"Top enabled? ": "Top enabled? ", "alphabetically - reverse": "alphabetically - reverse",
"CAPTCHA enabled? ": "CAPTCHA enabled? ", "channel name": "channel name",
"Login enabled? ": "Login enabled? ", "channel name - reverse": "channel name - reverse",
"Registration enabled? ": "Registration enabled? ", "Only show latest video from channel: ": "Only show latest video from channel: ",
"Report statistics? ": "Report statistics? ", "Only show latest unwatched video from channel: ": "Only show latest unwatched video from channel: ",
"Save preferences": "Save preferences", "Only show unwatched: ": "Only show unwatched: ",
"Subscription manager": "Subscription manager", "Only show notifications (if there are any): ": "Only show notifications (if there are any): ",
"`x` subscriptions": "`x` subscriptions", "Data preferences": "Data preferences",
"Import/Export": "Import/Export", "Clear watch history": "Clear watch history",
"unsubscribe": "unsubscribe", "Import/export data": "Import/export data",
"Subscriptions": "Subscriptions", "Change password": "Change password",
"`x` unseen notifications": "`x` unseen notifications", "Manage subscriptions": "Manage subscriptions",
"search": "search", "Manage tokens": "Manage tokens",
"Sign out": "Sign out", "Watch history": "Watch history",
"Released under the AGPLv3 by Omar Roth.": "Released under the AGPLv3 by Omar Roth.", "Delete account": "Delete account",
"Source available here.": "Source available here.", "Administrator preferences": "Administrator preferences",
"View JavaScript license information.": "View JavaScript license information.", "Default homepage: ": "Default homepage: ",
"View privacy policy.": "View privacy policy.", "Feed menu: ": "Feed menu: ",
"Trending": "Trending", "Top enabled? ": "Top enabled? ",
"Unlisted": "", "CAPTCHA enabled? ": "CAPTCHA enabled? ",
"Watch video on Youtube": "Watch video on Youtube", "Login enabled? ": "Login enabled? ",
"Genre: ": "Genre: ", "Registration enabled? ": "Registration enabled? ",
"License: ": "License: ", "Report statistics? ": "Report statistics? ",
"Family friendly? ": "Family friendly? ", "Save preferences": "Save preferences",
"Wilson score: ": "Wilson score: ", "Subscription manager": "Subscription manager",
"Engagement: ": "Engagement: ", "Token manager": "Token manager",
"Whitelisted regions: ": "Whitelisted regions: ", "Token": "Token",
"Blacklisted regions: ": "Blacklisted regions: ", "`x` subscriptions": {
"Shared `x`": "Shared `x`", "(\\D|^)1(\\D|$)": "`x` subscription",
"Premieres in `x`": "", "": "`x` subscriptions"
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.", },
"View YouTube comments": "View YouTube comments", "`x` tokens": {
"View more comments on Reddit": "View more comments on Reddit", "(\\D|^)1(\\D|$)": "`x` token",
"View `x` comments": "View `x` comments", "": "`x` tokens"
"View Reddit comments": "View Reddit comments", },
"Hide replies": "Hide replies", "Import/export": "Import/export",
"Show replies": "Show replies", "unsubscribe": "unsubscribe",
"Incorrect password": "Incorrect password", "revoke": "revoke",
"Quota exceeded, try again in a few hours": "Quota exceeded, try again in a few hours", "Subscriptions": "Subscriptions",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.", "`x` unseen notifications": {
"Invalid TFA code": "Invalid TFA code", "(\\D|^)1(\\D|$)": "`x` unseen notification",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login failed. This may be because two-factor authentication is not enabled on your account.", "": "`x` unseen notifications"
"Invalid answer": "Invalid answer", },
"Invalid CAPTCHA": "Invalid CAPTCHA", "search": "search",
"CAPTCHA is a required field": "CAPTCHA is a required field", "Log out": "Log out",
"User ID is a required field": "User ID is a required field", "Released under the AGPLv3 by Omar Roth.": "Released under the AGPLv3 by Omar Roth.",
"Password is a required field": "Password is a required field", "Source available here.": "Source available here.",
"Invalid username or password": "Invalid username or password", "View JavaScript license information.": "View JavaScript license information.",
"Please sign in using 'Sign in with Google'": "Please sign in using 'Sign in with Google'", "View privacy policy.": "View privacy policy.",
"Password cannot be empty": "Password cannot be empty", "Trending": "Trending",
"Password cannot be longer than 55 characters": "Password cannot be longer than 55 characters", "Unlisted": "Unlisted",
"Please sign in": "Please sign in", "Watch on YouTube": "Watch on YouTube",
"Invidious Private Feed for `x`": "Invidious Private Feed for `x`", "Hide annotations": "Hide annotations",
"channel:`x`": "channel:`x`", "Show annotations": "Show annotations",
"Deleted or invalid channel": "Deleted or invalid channel", "Genre: ": "Genre: ",
"This channel does not exist.": "This channel does not exist.", "License: ": "License: ",
"Could not get channel info.": "Could not get channel info.", "Family friendly? ": "Family friendly? ",
"Could not fetch comments": "Could not fetch comments", "Wilson score: ": "Wilson score: ",
"View `x` replies": "View `x` replies", "Engagement: ": "Engagement: ",
"`x` ago": "`x` ago", "Whitelisted regions: ": "Whitelisted regions: ",
"Load more": "Load more", "Blacklisted regions: ": "Blacklisted regions: ",
"`x` points": "`x` points", "Shared `x`": "Shared `x`",
"Could not create mix.": "Could not create mix.", "`x` views": {
"Playlist is empty": "Playlist is empty", "(\\D|^)1(\\D|$)": "`x` views",
"Invalid playlist.": "Invalid playlist.", "": "`x` views"
"Playlist does not exist.": "Playlist does not exist.", },
"Could not pull trending pages.": "Could not pull trending pages.", "Premieres in `x`": "Premieres in `x`",
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.",
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field", "View YouTube comments": "View YouTube comments",
"Invalid challenge": "Invalid challenge", "View more comments on Reddit": "View more comments on Reddit",
"Invalid token": "Invalid token", "View `x` comments": "View `x` comments",
"Invalid user": "Invalid user", "View Reddit comments": "View Reddit comments",
"Token is expired, please try again": "Token is expired, please try again", "Hide replies": "Hide replies",
"English": "English", "Show replies": "Show replies",
"English (auto-generated)": "English (auto-generated)", "Incorrect password": "Incorrect password",
"Afrikaans": "Afrikaans", "Quota exceeded, try again in a few hours": "Quota exceeded, try again in a few hours",
"Albanian": "Albanian", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.",
"Amharic": "Amharic", "Invalid TFA code": "Invalid TFA code",
"Arabic": "Arabic", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Login failed. This may be because two-factor authentication is not turned on for your account.",
"Armenian": "Armenian", "Wrong answer": "Wrong answer",
"Azerbaijani": "Azerbaijani", "Erroneous CAPTCHA": "Erroneous CAPTCHA",
"Bangla": "Bangla", "CAPTCHA is a required field": "CAPTCHA is a required field",
"Basque": "Basque", "User ID is a required field": "User ID is a required field",
"Belarusian": "Belarusian", "Password is a required field": "Password is a required field",
"Bosnian": "Bosnian", "Wrong username or password": "Wrong username or password",
"Bulgarian": "Bulgarian", "Please sign in using 'Log in with Google'": "Please sign in using 'Log in with Google'",
"Burmese": "Burmese", "Password cannot be empty": "Password cannot be empty",
"Catalan": "Catalan", "Password cannot be longer than 55 characters": "Password cannot be longer than 55 characters",
"Cebuano": "Cebuano", "Please log in": "Please log in",
"Chinese (Simplified)": "Chinese (Simplified)", "Invidious Private Feed for `x`": "Invidious Private Feed for `x`",
"Chinese (Traditional)": "Chinese (Traditional)", "channel:`x`": "channel:`x`",
"Corsican": "Corsican", "Deleted or invalid channel": "Deleted or invalid channel",
"Croatian": "Croatian", "This channel does not exist.": "This channel does not exist.",
"Czech": "Czech", "Could not get channel info.": "Could not get channel info.",
"Danish": "Danish", "Could not fetch comments": "Could not fetch comments",
"Dutch": "Dutch", "View `x` replies": {
"Esperanto": "Esperanto", "(\\D|^)1(\\D|$)": "View `x` reply",
"Estonian": "Estonian", "": "View `x` replies"
"Filipino": "Filipino", },
"Finnish": "Finnish", "`x` ago": "`x` ago",
"French": "French", "Load more": "Load more",
"Galician": "Galician", "`x` points": {
"Georgian": "Georgian", "(\\D|^)1(\\D|$)": "`x` point",
"German": "German", "": "`x` points"
"Greek": "Greek", },
"Gujarati": "Gujarati", "Could not create mix.": "Could not create mix.",
"Haitian Creole": "Haitian Creole", "Empty playlist": "Empty playlist",
"Hausa": "Hausa", "Not a playlist.": "Not a playlist.",
"Hawaiian": "Hawaiian", "Playlist does not exist.": "Playlist does not exist.",
"Hebrew": "Hebrew", "Could not pull trending pages.": "Could not pull trending pages.",
"Hindi": "Hindi", "Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
"Hmong": "Hmong", "Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
"Hungarian": "Hungarian", "Erroneous challenge": "Erroneous challenge",
"Icelandic": "Icelandic", "Erroneous token": "Erroneous token",
"Igbo": "Igbo", "No such user": "No such user",
"Indonesian": "Indonesian", "Token is expired, please try again": "Token is expired, please try again",
"Irish": "Irish", "English": "English",
"Italian": "Italian", "English (auto-generated)": "English (auto-generated)",
"Japanese": "Japanese", "Afrikaans": "Afrikaans",
"Javanese": "Javanese", "Albanian": "Albanian",
"Kannada": "Kannada", "Amharic": "Amharic",
"Kazakh": "Kazakh", "Arabic": "Arabic",
"Khmer": "Khmer", "Armenian": "Armenian",
"Korean": "Korean", "Azerbaijani": "Azerbaijani",
"Kurdish": "Kurdish", "Bangla": "Bangla",
"Kyrgyz": "Kyrgyz", "Basque": "Basque",
"Lao": "Lao", "Belarusian": "Belarusian",
"Latin": "Latin", "Bosnian": "Bosnian",
"Latvian": "Latvian", "Bulgarian": "Bulgarian",
"Lithuanian": "Lithuanian", "Burmese": "Burmese",
"Luxembourgish": "Luxembourgish", "Catalan": "Catalan",
"Macedonian": "Macedonian", "Cebuano": "Cebuano",
"Malagasy": "Malagasy", "Chinese (Simplified)": "Chinese (Simplified)",
"Malay": "Malay", "Chinese (Traditional)": "Chinese (Traditional)",
"Malayalam": "Malayalam", "Corsican": "Corsican",
"Maltese": "Maltese", "Croatian": "Croatian",
"Maori": "Maori", "Czech": "Czech",
"Marathi": "Marathi", "Danish": "Danish",
"Mongolian": "Mongolian", "Dutch": "Dutch",
"Nepali": "Nepali", "Esperanto": "Esperanto",
"Norwegian": "Norwegian", "Estonian": "Estonian",
"Nyanja": "Nyanja", "Filipino": "Filipino",
"Pashto": "Pashto", "Finnish": "Finnish",
"Persian": "Persian", "French": "French",
"Polish": "Polish", "Galician": "Galician",
"Portuguese": "Portuguese", "Georgian": "Georgian",
"Punjabi": "Punjabi", "German": "German",
"Romanian": "Romanian", "Greek": "Greek",
"Russian": "Russian", "Gujarati": "Gujarati",
"Samoan": "Samoan", "Haitian Creole": "Haitian Creole",
"Scottish Gaelic": "Scottish Gaelic", "Hausa": "Hausa",
"Serbian": "Serbian", "Hawaiian": "Hawaiian",
"Shona": "Shona", "Hebrew": "Hebrew",
"Sindhi": "Sindhi", "Hindi": "Hindi",
"Sinhala": "Sinhala", "Hmong": "Hmong",
"Slovak": "Slovak", "Hungarian": "Hungarian",
"Slovenian": "Slovenian", "Icelandic": "Icelandic",
"Somali": "Somali", "Igbo": "Igbo",
"Southern Sotho": "Southern Sotho", "Indonesian": "Indonesian",
"Spanish": "Spanish", "Irish": "Irish",
"Spanish (Latin America)": "Spanish (Latin America)", "Italian": "Italian",
"Sundanese": "Sundanese", "Japanese": "Japanese",
"Swahili": "Swahili", "Javanese": "Javanese",
"Swedish": "Swedish", "Kannada": "Kannada",
"Tajik": "Tajik", "Kazakh": "Kazakh",
"Tamil": "Tamil", "Khmer": "Khmer",
"Telugu": "Telugu", "Korean": "Korean",
"Thai": "Thai", "Kurdish": "Kurdish",
"Turkish": "Turkish", "Kyrgyz": "Kyrgyz",
"Ukrainian": "Ukrainian", "Lao": "Lao",
"Urdu": "Urdu", "Latin": "Latin",
"Uzbek": "Uzbek", "Latvian": "Latvian",
"Vietnamese": "Vietnamese", "Lithuanian": "Lithuanian",
"Welsh": "Welsh", "Luxembourgish": "Luxembourgish",
"Western Frisian": "Western Frisian", "Macedonian": "Macedonian",
"Xhosa": "Xhosa", "Malagasy": "Malagasy",
"Yiddish": "Yiddish", "Malay": "Malay",
"Yoruba": "Yoruba", "Malayalam": "Malayalam",
"Zulu": "Zulu", "Maltese": "Maltese",
"`x` years": "`x` years", "Maori": "Maori",
"`x` months": "`x` months", "Marathi": "Marathi",
"`x` weeks": "`x` weeks", "Mongolian": "Mongolian",
"`x` days": "`x` days", "Nepali": "Nepali",
"`x` hours": "`x` hours", "Norwegian Bokmål": "Norwegian Bokmål",
"`x` minutes": "`x` minutes", "Nyanja": "Nyanja",
"`x` seconds": "`x` seconds", "Pashto": "Pashto",
"Fallback comments: ": "Fallback comments: ", "Persian": "Persian",
"Popular": "Popular", "Polish": "Polish",
"Top": "Top", "Portuguese": "Portuguese",
"About": "About", "Punjabi": "Punjabi",
"Rating: ": "Rating: ", "Romanian": "Romanian",
"Language: ": "Language: ", "Russian": "Russian",
"Default": "Default", "Samoan": "Samoan",
"Music": "Music", "Scottish Gaelic": "Scottish Gaelic",
"Gaming": "Gaming", "Serbian": "Serbian",
"News": "News", "Shona": "Shona",
"Movies": "Movies", "Sindhi": "Sindhi",
"Download": "Download", "Sinhala": "Sinhala",
"Download as: ": "Download as: ", "Slovak": "Slovak",
"%A %B %-d, %Y": "%A %B %-d, %Y", "Slovenian": "Slovenian",
"(edited)": "(edited)", "Somali": "Somali",
"Youtube permalink of the comment": "Youtube permalink of the comment", "Southern Sotho": "Southern Sotho",
"`x` marked it with a ❤": "`x` marked it with a ❤", "Spanish": "Spanish",
"Audio mode": "Audio mode", "Spanish (Latin America)": "Spanish (Latin America)",
"Video mode": "Video mode", "Sundanese": "Sundanese",
"Videos": "Videos", "Swahili": "Swahili",
"Playlists": "Playlists", "Swedish": "Swedish",
"Current version: ": "Current version: " "Tajik": "Tajik",
} "Tamil": "Tamil",
"Telugu": "Telugu",
"Thai": "Thai",
"Turkish": "Turkish",
"Ukrainian": "Ukrainian",
"Urdu": "Urdu",
"Uzbek": "Uzbek",
"Vietnamese": "Vietnamese",
"Welsh": "Welsh",
"Western Frisian": "Western Frisian",
"Xhosa": "Xhosa",
"Yiddish": "Yiddish",
"Yoruba": "Yoruba",
"Zulu": "Zulu",
"`x` years": {
"(\\D|^)1(\\D|$)": "`x` year",
"": "`x` years"
},
"`x` months": {
"(\\D|^)1(\\D|$)": "`x` month",
"": "`x` months"
},
"`x` weeks": {
"(\\D|^)1(\\D|$)": "`x` week",
"": "`x` weeks"
},
"`x` days": {
"(\\D|^)1(\\D|$)": "`x` day",
"": "`x` days"
},
"`x` hours": {
"(\\D|^)1(\\D|$)": "`x` hour",
"": "`x` hours"
},
"`x` minutes": {
"(\\D|^)1(\\D|$)": "`x` minute",
"": "`x` minutes"
},
"`x` seconds": {
"(\\D|^)1(\\D|$)": "`x` second",
"": "`x` seconds"
},
"Fallback comments: ": "Fallback comments: ",
"Popular": "Popular",
"Top": "Top",
"About": "About",
"Rating: ": "Rating: ",
"Language: ": "Language: ",
"View as playlist": "View as playlist",
"Default": "Default",
"Music": "Music",
"Gaming": "Gaming",
"News": "News",
"Movies": "Movies",
"Download": "Download",
"Download as: ": "Download as: ",
"%A %B %-d, %Y": "%A %B %-d, %Y",
"(edited)": "(edited)",
"YouTube comment permalink": "YouTube comment permalink",
"`x` marked it with a ❤": "`x` marked it with a ❤",
"Audio mode": "Audio mode",
"Video mode": "Video mode",
"Videos": "Videos",
"Playlists": "Playlists",
"Current version: ": "Current version: "
}

315
locales/eo.json Normal file
View File

@ -0,0 +1,315 @@
{
"`x` subscribers": "`x` abonantoj",
"`x` videos": "`x` videoj",
"LIVE": "NUNA",
"Shared `x` ago": "Konigita antaŭ `x`",
"Unsubscribe": "Malaboni",
"Subscribe": "Aboni",
"View channel on YouTube": "Vidi kanalon en YouTube",
"View playlist on YouTube": "",
"newest": "pli novaj",
"oldest": "pli malnovaj",
"popular": "popularaj",
"last": "lasta",
"Next page": "Sekva paĝo",
"Previous page": "Antaŭa paĝo",
"Clear watch history?": "Ĉu forigi vidohistorion?",
"New password": "Nova pasvorto",
"New passwords must match": "Novaj pasvortoj devas kongrui",
"Cannot change password for Google accounts": "Ne eblas ŝanĝi pasvorton por kontoj de Google",
"Authorize token?": "Ĉu rajtigi ĵetonon?",
"Authorize token for `x`?": "Ĉu rajtigi ĵetonon por `x`?",
"Yes": "Jes",
"No": "Ne",
"Import and Export Data": "Importi kaj Eksporti Datumojn",
"Import": "Importi",
"Import Invidious data": "Importi datumojn de Invidious",
"Import YouTube subscriptions": "Importi abonojn de YouTube",
"Import FreeTube subscriptions (.db)": "Importi abonojn de FreeTube (.db)",
"Import NewPipe subscriptions (.json)": "Importi abonojn de NewPipe (.json)",
"Import NewPipe data (.zip)": "Importi datumojn de NewPipe (.zip)",
"Export": "Eksporti",
"Export subscriptions as OPML": "Eksporti abonojn kiel OPML",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksporti abonojn kiel OPML (por NewPipe kaj FreeTube)",
"Export data as JSON": "Eksporti datumojn kiel JSON",
"Delete account?": "Ĉu forigi konton?",
"History": "Historio",
"An alternative front-end to YouTube": "Alternativa fasado al YouTube",
"JavaScript license information": "Ĝavoskripta licenca informo",
"source": "fonto",
"Log in": "Ensaluti",
"Log in/register": "Ensaluti/Registriĝi",
"Log in with Google": "Ensaluti al Google",
"User ID": "Uzula identigilo",
"Password": "Pasvorto",
"Time (h:mm:ss):": "Horo (h:mm:ss):",
"Text CAPTCHA": "Teksta CAPTCHA",
"Image CAPTCHA": "Bilda CAPTCHA",
"Sign In": "Ensaluti",
"Register": "Registriĝi",
"E-mail": "Retpoŝto",
"Google verification code": "Kontrolkodo de Google",
"Preferences": "Agordoj",
"Player preferences": "Spektilaj agordoj",
"Always loop: ": "Ĉiam ripeti: ",
"Autoplay: ": "Aŭtomate ludi: ",
"Play next by default: ": "Ludi sekvan defaŭlte: ",
"Autoplay next video: ": "Aŭtomate ludi sekvan videon: ",
"Listen by default: ": "Aŭskulti defaŭlte: ",
"Proxy videos? ": "Ĉu uzi prokuran servilon por videoj? ",
"Default speed: ": "Defaŭlta rapido: ",
"Preferred video quality: ": "Preferita videkvalito: ",
"Player volume: ": "Ludila sonforteco: ",
"Default comments: ": "Defaŭltaj komentoj: ",
"youtube": "youtube",
"reddit": "reddit",
"Default captions: ": "Defaŭltaj subtekstoj: ",
"Fallback captions: ": "Retrodefaŭltaj subtekstoj: ",
"Show related videos? ": "Ĉu montri rilatajn videojn? ",
"Show annotations by default? ": "Ĉu montri prinotojn defaŭlte? ",
"Visual preferences": "Vidaj preferoj",
"Dark mode: ": "Malhela reĝimo: ",
"Thin mode: ": "Maldika reĝimo: ",
"Subscription preferences": "Abonaj agordoj",
"Show annotations by default for subscribed channels? ": "Ĉu montri prinotojn defaŭlte por abonitaj kanaloj? ",
"Redirect homepage to feed: ": "Alidirekti hejmpâgon al fluo: ",
"Number of videos shown in feed: ": "Nombro da videoj montritaj en fluo: ",
"Sort videos by: ": "Ordi videojn laŭ: ",
"published": "publikigo",
"published - reverse": "publitigo - renverse",
"alphabetically": "alfabete",
"alphabetically - reverse": "alfabete - renverse",
"channel name": "kanala nombro",
"channel name - reverse": "kanala nombro - renverse",
"Only show latest video from channel: ": "Nur montri pli novan videon el kanalo: ",
"Only show latest unwatched video from channel: ": "Nur montri pli novan malviditan videon el kanalo: ",
"Only show unwatched: ": "Nur montri malviditajn: ",
"Only show notifications (if there are any): ": "Nur montri sciigojn (se estas): ",
"Data preferences": "Datumagordoj",
"Clear watch history": "Forigi vidohistorion",
"Import/export data": "Importi/Eksporti datumojn",
"Change password": "Ŝanĝi pasvorton",
"Manage subscriptions": "Administri abonojn",
"Manage tokens": "Administri ĵetonojn",
"Watch history": "Vidohistorio",
"Delete account": "Forigi konton",
"Administrator preferences": "Agordoj de administranto",
"Default homepage: ": "Defaŭlta hejmpaĝo: ",
"Feed menu: ": "Flua menuo: ",
"Top enabled? ": "Ĉu pli bonaj ŝaltitaj? ",
"CAPTCHA enabled? ": "Ĉu CAPTCHA ŝaltita? ",
"Login enabled? ": "Ĉu ensaluto aktivita? ",
"Registration enabled? ": "Ĉu registriĝo aktivita? ",
"Report statistics? ": "Ĉu raporti statistikojn? ",
"Save preferences": "Konservi agordojn",
"Subscription manager": "Administrilo de abonoj",
"Token manager": "Ĵetona administrilo",
"Token": "Ĵetono",
"`x` subscriptions": "`x` abonoj",
"`x` tokens": "`x` ĵetonoj",
"Import/export": "Importi/Eksporti",
"unsubscribe": "malaboni",
"revoke": "senvalidigi",
"Subscriptions": "Abonoj",
"`x` unseen notifications": "`x` neviditaj sciigoj",
"search": "serĉi",
"Log out": "Elsaluti",
"Released under the AGPLv3 by Omar Roth.": "Eldonita sub la AGPLv3 de Omar Roth.",
"Source available here.": "Fonto havebla ĉi tie.",
"View JavaScript license information.": "Vidi Ĝavoskriptan licencan informon.",
"View privacy policy.": "Vidi regularon pri privateco.",
"Trending": "Tendencoj",
"Unlisted": "Ne listigita",
"Watch on YouTube": "Vidi videon en Youtube",
"Hide annotations": "Kaŝi prinotojn",
"Show annotations": "Montri prinotojn",
"Genre: ": "Ĝenro: ",
"License: ": "Licenco: ",
"Family friendly? ": "Ĉu familie amika? ",
"Wilson score: ": "Poentaro de Wilson: ",
"Engagement: ": "Intereso: ",
"Whitelisted regions: ": "Regionoj listigitaj en blanka listo: ",
"Blacklisted regions: ": "Regionoj listigitaj en nigra listo: ",
"Shared `x`": "Konigita `x`",
"`x` views": "`x` spektaĵoj",
"Premieres in `x`": "Premieras en `x`",
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Saluton! Ŝajnas, ke vi havas Ĝavoskripton malebligitan. Klaku ĉi tie por vidi komentojn, memoru, ke la ŝargado povus daŭri iom pli.",
"View YouTube comments": "Vidi komentojn de YouTube",
"View more comments on Reddit": "Vidi pli komentoj en Reddit",
"View `x` comments": "Vidi `x` komentojn",
"View Reddit comments": "Vidi komentojn de Reddit",
"Hide replies": "Kaŝi respondojn",
"Show replies": "Montri respondojn",
"Incorrect password": "Malbona pasvorto",
"Quota exceeded, try again in a few hours": "Kvoto transpasita, provu denove post iuj horoj",
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Ne povas ensaluti, certigu, ke dufaktora aŭtentigo (Authenticator aŭ SMS) estas ebligita.",
"Invalid TFA code": "Nevalida TFA-kodo",
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Ensalutado fiaskis. Eble ĉar la dufaktora aŭtentigo estas malebligita en via konto.",
"Wrong answer": "Nevalida respondo",
"Erroneous CAPTCHA": "Nevalida CAPTCHA",
"CAPTCHA is a required field": "CAPTCHA estas deviga kampo",
"User ID is a required field": "Uzula identigilo estas deviga kampo",
"Password is a required field": "Pasvorto estas deviga kampo",
"Wrong username or password": "Nevalida uzantnomo aŭ pasvorto",
"Please sign in using 'Log in with Google'": "Bonvolu ensaluti per 'Ensaluti per Google'",
"Password cannot be empty": "Pasvorto ne povas esti malplena",
"Password cannot be longer than 55 characters": "Pasvorto ne povas esti pli longa ol 55 signoj",
"Please log in": "Bonvolu ensaluti",
"Invidious Private Feed for `x`": "Privata Fluo de Invidious por `x`",
"channel:`x`": "kanalo:`x`",
"Deleted or invalid channel": "Forigita aŭ nevalida kanalo",
"This channel does not exist.": "Ĉi tiu kanalo ne ekzistas.",
"Could not get channel info.": "Ne povis havigi kanalan informon.",
"Could not fetch comments": "Ne povis venigi komentojn",
"View `x` replies": "Vidi `x` respondojn",
"`x` ago": "antaŭ `x`",
"Load more": "Ŝarĝi pli",
"`x` points": "`x` poentoj",
"Could not create mix.": "Ne povis krei mikson.",
"Empty playlist": "Ludlisto estas malplena",
"Not a playlist.": "Nevalida ludlisto.",
"Playlist does not exist.": "Ludlisto ne ekzistas.",
"Could not pull trending pages.": "Ne povis venigi tendencajn paĝojn.",
"Hidden field \"challenge\" is a required field": "Kaŝita kampo \"challenge\" estas deviga kampo",
"Hidden field \"token\" is a required field": "Kaŝita kampo \"token\" estas deviga kampo",
"Erroneous challenge": "Nevalida defio",
"Erroneous token": "Nevalida ĵetono",
"No such user": "Nevalida uzanto",
"Token is expired, please try again": "Ĵetono senvalidiĝis, bonvolu provi denove",
"English": "Angla",
"English (auto-generated)": "Angla (aŭtomate generita)",
"Afrikaans": "Afrikansa",
"Albanian": "Albana",
"Amharic": "Amhara",
"Arabic": "Araba",
"Armenian": "Armena",
"Azerbaijani": "Azerbajĝana",
"Bangla": "Bengala",
"Basque": "Eŭska",
"Belarusian": "Belorusa",
"Bosnian": "Bosna",
"Bulgarian": "Bulgara",
"Burmese": "Birma",
"Catalan": "Kataluna",
"Cebuano": "Cebua",
"Chinese (Simplified)": "Ĉina (simpligita)",
"Chinese (Traditional)": "Ĉina (tradicia)",
"Corsican": "Korsika",
"Croatian": "Kroata",
"Czech": "Ĉeĥa",
"Danish": "Dana",
"Dutch": "Nederlanda",
"Esperanto": "Esperanto",
"Estonian": "Estona",
"Filipino": "Filipina",
"Finnish": "Finna",
"French": "Franca",
"Galician": "Galega",
"Georgian": "Kartvela",
"German": "Germana",
"Greek": "Greka",
"Gujarati": "Guĝarata",
"Haitian Creole": "Haitia kreola",
"Hausa": "Haŭsa",
"Hawaiian": "Havaja",
"Hebrew": "Hebrea",
"Hindi": "Hindia",
"Hmong": "Miaa",
"Hungarian": "Hungara",
"Icelandic": "Islanda",
"Igbo": "Igba",
"Indonesian": "Indonezia",
"Irish": "Irlanda",
"Italian": "Itala",
"Japanese": "Japana",
"Javanese": "Java",
"Kannada": "Kanara",
"Kazakh": "Kazaĥa",
"Khmer": "Kmera",
"Korean": "Korea",
"Kurdish": "Kurda",
"Kyrgyz": "Kirgiza",
"Lao": "Laosa",
"Latin": "Latina",
"Latvian": "Latva",
"Lithuanian": "Litova",
"Luxembourgish": "Luksemburga",
"Macedonian": "Makedona",
"Malagasy": "Malagasa",
"Malay": "Malaja",
"Malayalam": "Malajala",
"Maltese": "Malta",
"Maori": "Maoria",
"Marathi": "Marata",
"Mongolian": "Mongola",
"Nepali": "Nepala",
"Norwegian Bokmål": "Norvega",
"Nyanja": "Njanĝa",
"Pashto": "Paŝtuna",
"Persian": "Persa",
"Polish": "Pola",
"Portuguese": "Portugala",
"Punjabi": "Panĝaba",
"Romanian": "Rumana",
"Russian": "Rusa",
"Samoan": "Samoa",
"Scottish Gaelic": "Skotgaela",
"Serbian": "Serba",
"Shona": "Ŝona",
"Sindhi": "Sinda",
"Sinhala": "Sinhala",
"Slovak": "Slovaka",
"Slovenian": "Slovena",
"Somali": "Somala",
"Southern Sotho": "Sota",
"Spanish": "Hispana",
"Spanish (Latin America)": "Hispana (Latinameriko)",
"Sundanese": "Sunda",
"Swahili": "Svahila",
"Swedish": "Sveda",
"Tajik": "Taĝika",
"Tamil": "Tamila",
"Telugu": "Telugua",
"Thai": "Taja",
"Turkish": "Turka",
"Ukrainian": "Ukraina",
"Urdu": "Urduo",
"Uzbek": "Uzbeka",
"Vietnamese": "Vjetnama",
"Welsh": "Kimra",
"Western Frisian": "Okcidentfrisa",
"Xhosa": "Kosa",
"Yiddish": "Jida",
"Yoruba": "Joruba",
"Zulu": "Zulua",
"`x` years": "`x` jaroj",
"`x` months": "`x` monatoj",
"`x` weeks": "`x` semajnoj",
"`x` days": "`x` tagoj",
"`x` hours": "`x` horoj",
"`x` minutes": "`x` minutoj",
"`x` seconds": "`x` sekundoj",
"Fallback comments: ": "Retrodefaŭltaj komentoj: ",
"Popular": "Popularaj",
"Top": "Supraj",
"About": "Pri",
"Rating: ": "Takso: ",
"Language: ": "Lingvo: ",
"View as playlist": "Vidi kiel ludlisto",
"Default": "Defaŭlte",
"Music": "Musiko",
"Gaming": "Komputiloludoj",
"News": "Novaĵoj",
"Movies": "Filmoj",
"Download": "Elŝuti",
"Download as: ": "Elŝuti kiel: ",
"%A %B %-d, %Y": "%A %-d de %B %Y",
"(edited)": "(redaktita)",
"YouTube comment permalink": "Fiksligilo de la komento en YouTube",
"`x` marked it with a ❤": "`x` markis ĝin per ❤",
"Audio mode": "Aŭda reĝimo",
"Video mode": "Videa reĝimo",
"Videos": "Videoj",
"Playlists": "Ludlistoj",
"Current version: ": "Nuna versio: "
}

315
locales/es.json Normal file
View File

@ -0,0 +1,315 @@
{
"`x` subscribers": "`x` suscriptores",
"`x` videos": "`x` vídeos",
"LIVE": "DIRECTO",
"Shared `x` ago": "Compartido hace `x`",
"Unsubscribe": "Desuscribirse",
"Subscribe": "Suscribirse",
"View channel on YouTube": "Ver el canal en YouTube",
"View playlist on YouTube": "",
"newest": "más nuevos",
"oldest": "más viejos",
"popular": "populares",
"last": "último",
"Next page": "Página siguiente",
"Previous page": "Página anterior",
"Clear watch history?": "¿Quiere borrar el historial de reproducción?",
"New password": "Nueva contraseña",
"New passwords must match": "Las nuevas contraseñas deben coincidir",
"Cannot change password for Google accounts": "No se puede cambiar la contraseña de la cuenta de Google",
"Authorize token?": "¿Autorizar el token?",
"Authorize token for `x`?": "¿Autorizar el token para `x`?",
"Yes": "Sí",
"No": "No",
"Import and Export Data": "Importación y exportación de datos",
"Import": "Importar",
"Import Invidious data": "Importar datos de Invidious",
"Import YouTube subscriptions": "Importar suscripciones de YouTube",
"Import FreeTube subscriptions (.db)": "Importar suscripciones de FreeTube (.db)",
"Import NewPipe subscriptions (.json)": "Importar suscripciones de NewPipe (.json)",
"Import NewPipe data (.zip)": "Importar datos de NewPipe (.zip)",
"Export": "Exportar",
"Export subscriptions as OPML": "Exportar suscripciones como OPML",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exportar suscripciones como OPML (para NewPipe y FreeTube)",
"Export data as JSON": "Exportar datos como JSON",
"Delete account?": "¿Quiere borrar la cuenta?",
"History": "Historial",
"An alternative front-end to YouTube": "Una interfaz alternativa para YouTube",
"JavaScript license information": "Información de licencia de JavaScript",
"source": "código fuente",
"Log in": "Iniciar sesión",
"Log in/register": "Iniciar sesión/Registrarse",
"Log in with Google": "Iniciar sesión en Google",
"User ID": "Nombre",
"Password": "Contraseña",
"Time (h:mm:ss):": "Hora (h:mm:ss):",
"Text CAPTCHA": "CAPTCHA en texto",
"Image CAPTCHA": "CAPTCHA en imagen",
"Sign In": "Iniciar sesión",
"Register": "Registrarse",
"E-mail": "Correo",
"Google verification code": "Código de verificación de Google",
"Preferences": "Preferencias",
"Player preferences": "Preferencias del reproductor",
"Always loop: ": "Repetir siempre: ",
"Autoplay: ": "Reproducción automática: ",
"Play next by default: ": "Reproducir siguiente por defecto: ",
"Autoplay next video: ": "Reproducir automáticamente el vídeo siguiente: ",
"Listen by default: ": "Activar el sonido por defecto: ",
"Proxy videos? ": "¿Usar un proxy para los vídeos? ",
"Default speed: ": "Velocidad por defecto: ",
"Preferred video quality: ": "Calidad de vídeo preferida: ",
"Player volume: ": "Volumen del reproductor: ",
"Default comments: ": "Comentarios por defecto: ",
"youtube": "YouTube",
"reddit": "Reddit",
"Default captions: ": "Subtítulos por defecto: ",
"Fallback captions: ": "Subtítulos alternativos: ",
"Show related videos? ": "¿Mostrar vídeos relacionados? ",
"Show annotations by default? ": "¿Mostrar anotaciones por defecto? ",
"Visual preferences": "Preferencias visuales",
"Dark mode: ": "Modo oscuro: ",
"Thin mode: ": "Modo compacto: ",
"Subscription preferences": "Preferencias de la suscripción",
"Show annotations by default for subscribed channels? ": "¿Mostrar anotaciones por defecto para los canales suscritos? ",
"Redirect homepage to feed: ": "Redirigir la página de inicio a la fuente: ",
"Number of videos shown in feed: ": "Número de vídeos mostrados en la fuente: ",
"Sort videos by: ": "Ordenar los vídeos por: ",
"published": "fecha de publicación",
"published - reverse": "fecha de publicación: orden inverso",
"alphabetically": "alfabéticamente",
"alphabetically - reverse": "alfabéticamente: orden inverso",
"channel name": "nombre del canal",
"channel name - reverse": "nombre del canal: orden inverso",
"Only show latest video from channel: ": "Mostrar solo el último vídeo del canal: ",
"Only show latest unwatched video from channel: ": "Mostrar solo el último vídeo sin ver del canal: ",
"Only show unwatched: ": "Mostrar solo los no vistos: ",
"Only show notifications (if there are any): ": "Mostrar solo notificaciones (si hay alguna): ",
"Data preferences": "Preferencias de los datos",
"Clear watch history": "Borrar el historial de reproducción",
"Import/export data": "Importar/Exportar datos",
"Change password": "Cambiar contraseña",
"Manage subscriptions": "Gestionar las suscripciones",
"Manage tokens": "Gestionar tokens",
"Watch history": "Historial de reproducción",
"Delete account": "Borrar cuenta",
"Administrator preferences": "Preferencias de administrador",
"Default homepage: ": "Página de inicio por defecto: ",
"Feed menu: ": "Menú de fuentes: ",
"Top enabled? ": "¿Habilitar los destacados? ",
"CAPTCHA enabled? ": "¿Habilitar los CAPTCHA? ",
"Login enabled? ": "¿Habilitar el inicio de sesión? ",
"Registration enabled? ": "¿Habilitar el registro? ",
"Report statistics? ": "¿Enviar estadísticas? ",
"Save preferences": "Guardar las preferencias",
"Subscription manager": "Gestor de suscripciones",
"Token manager": "Gestor de tokens",
"Token": "Token",
"`x` subscriptions": "`x` suscripciones",
"`x` tokens": "`x` tokens",
"Import/export": "Importar/Exportar",
"unsubscribe": "Desuscribirse",
"revoke": "revocar",
"Subscriptions": "Suscripciones",
"`x` unseen notifications": "`x` notificaciones sin ver",
"search": "buscar",
"Log out": "Cerrar la sesión",
"Released under the AGPLv3 by Omar Roth.": "Publicado bajo licencia AGPLv3 por Omar Roth.",
"Source available here.": "Código fuente disponible aquí.",
"View JavaScript license information.": "Ver información de licencia de JavaScript.",
"View privacy policy.": "Ver la política de privacidad.",
"Trending": "Tendencias",
"Unlisted": "No listado",
"Watch on YouTube": "Ver el vídeo en Youtube",
"Hide annotations": "Ocultar anotaciones",
"Show annotations": "Mostrar anotaciones",
"Genre: ": "Género: ",
"License: ": "Licencia: ",
"Family friendly? ": "¿Filtrar contenidos? ",
"Wilson score: ": "Puntuación Wilson: ",
"Engagement: ": "Compromiso: ",
"Whitelisted regions: ": "Regiones permitidas: ",
"Blacklisted regions: ": "Regiones bloqueadas: ",
"Shared `x`": "Compartido `x`",
"`x` views": "`x` visualizaciones",
"Premieres in `x`": "Se estrena en `x`",
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "¡Hola! Parece que tiene JavaScript desactivado. Haga clic aquí para ver los comentarios, pero tenga en cuenta que pueden tardar un poco más en cargarse.",
"View YouTube comments": "Ver los comentarios de YouTube",
"View more comments on Reddit": "Ver más comentarios en Reddit",
"View `x` comments": "Ver `x` comentarios",
"View Reddit comments": "Ver los comentarios de Reddit",
"Hide replies": "Ocultar las respuestas",
"Show replies": "Mostrar las respuestas",
"Incorrect password": "Contraseña incorrecta",
"Quota exceeded, try again in a few hours": "Cuota excedida, pruebe otra vez en unas horas",
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "No se puede iniciar sesión, asegúrese de que la autentificación de dos factores (autentificador o SMS) esté habilitada.",
"Invalid TFA code": "Código TFA no válido",
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Error de inicio de sesion. Puede deberse a que la autentificación de dos factores no está habilitada en su cuenta.",
"Wrong answer": "Respuesta no válida",
"Erroneous CAPTCHA": "CAPTCHA no válido",
"CAPTCHA is a required field": "El CAPTCHA es un campo obligatorio",
"User ID is a required field": "El nombre es un campo obligatorio",
"Password is a required field": "La contraseña es un campo obligatorio",
"Wrong username or password": "Nombre o contraseña incorrecto",
"Please sign in using 'Log in with Google'": "Inicie sesión con «Iniciar sesión con Google»",
"Password cannot be empty": "La contraseña no puede estar en blanco",
"Password cannot be longer than 55 characters": "La contraseña no puede tener más de 55 caracteres",
"Please log in": "Inicie sesión, por favor",
"Invidious Private Feed for `x`": "Fuente privada de Invidious para `x`",
"channel:`x`": "canal: `x`",
"Deleted or invalid channel": "El canal no es válido o ha sido borrado",
"This channel does not exist.": "El canal no existe.",
"Could not get channel info.": "No se ha podido obtener información del canal.",
"Could not fetch comments": "No se han podido recuperar los comentarios",
"View `x` replies": "Ver `x` respuestas",
"`x` ago": "hace `x`",
"Load more": "Cargar más",
"`x` points": "`x` puntos",
"Could not create mix.": "No se ha podido crear la mezcla.",
"Empty playlist": "La lista de reproducción está vacía",
"Not a playlist.": "Lista de reproducción no válida.",
"Playlist does not exist.": "La lista de reproducción no existe.",
"Could not pull trending pages.": "No se han podido obtener las páginas de tendencias.",
"Hidden field \"challenge\" is a required field": "El campo oculto «desafío» es un campo obligatorio",
"Hidden field \"token\" is a required field": "El campo oculto «símbolo» es un campo obligatorio",
"Erroneous challenge": "Desafío no válido",
"Erroneous token": "Símbolo no válido",
"No such user": "Usuario no válido",
"Token is expired, please try again": "El símbolo ha caducado, inténtelo de nuevo",
"English": "Inglés",
"English (auto-generated)": "Inglés (autogenerado)",
"Afrikaans": "Afrikáans",
"Albanian": "Albanés",
"Amharic": "Amárico",
"Arabic": "Árabe",
"Armenian": "Armenio",
"Azerbaijani": "Azerbaiyano",
"Bangla": "Bengalí",
"Basque": "Euskera",
"Belarusian": "Bielorruso",
"Bosnian": "Bosnio",
"Bulgarian": "Búlgaro",
"Burmese": "Birmano",
"Catalan": "Catalán",
"Cebuano": "Cebuano",
"Chinese (Simplified)": "Chino (simplificado)",
"Chinese (Traditional)": "Chino (tradicional)",
"Corsican": "Corso",
"Croatian": "Croata",
"Czech": "Checo",
"Danish": "Danés",
"Dutch": "Holandés",
"Esperanto": "Esperanto",
"Estonian": "Estonio",
"Filipino": "Filipino",
"Finnish": "Finés",
"French": "Francés",
"Galician": "Gallego",
"Georgian": "Georgiano",
"German": "Alemán",
"Greek": "Griego",
"Gujarati": "Guyaratí",
"Haitian Creole": "Criollo haitiano",
"Hausa": "Hausa",
"Hawaiian": "Hawaiano",
"Hebrew": "Hebreo",
"Hindi": "Hindi",
"Hmong": "Hmong",
"Hungarian": "Húngaro",
"Icelandic": "Islandés",
"Igbo": "Igbo",
"Indonesian": "Indonesio",
"Irish": "Irlandés",
"Italian": "Italiano",
"Japanese": "Japonés",
"Javanese": "Javanés",
"Kannada": "Canarés",
"Kazakh": "Kazajo",
"Khmer": "Camboyano",
"Korean": "Coreano",
"Kurdish": "Kurdo",
"Kyrgyz": "Kirguís",
"Lao": "Laosiano",
"Latin": "Latín",
"Latvian": "Letón",
"Lithuanian": "Lituano",
"Luxembourgish": "Luxemburgués",
"Macedonian": "Macedonio",
"Malagasy": "Malgache",
"Malay": "Malayo",
"Malayalam": "Malabar",
"Maltese": "Maltés",
"Maori": "Maorí",
"Marathi": "Maratí",
"Mongolian": "Mongol",
"Nepali": "Nepalí",
"Norwegian Bokmål": "Noruego",
"Nyanja": "Chichewa",
"Pashto": "Pastún",
"Persian": "Persa",
"Polish": "Polaco",
"Portuguese": "Portugués",
"Punjabi": "Panyabí",
"Romanian": "Rumano",
"Russian": "Ruso",
"Samoan": "Samoano",
"Scottish Gaelic": "Gaélico escocés",
"Serbian": "Serbio",
"Shona": "Shona",
"Sindhi": "Sindi",
"Sinhala": "Cingalés",
"Slovak": "Eslovaco",
"Slovenian": "Esloveno",
"Somali": "Somalí",
"Southern Sotho": "Sesoto",
"Spanish": "Español",
"Spanish (Latin America)": "Español (Hispanoamérica)",
"Sundanese": "Sondanés",
"Swahili": "Suajili",
"Swedish": "Sueco",
"Tajik": "Tayiko",
"Tamil": "Tamil",
"Telugu": "Telugu",
"Thai": "Tailandés",
"Turkish": "Turco",
"Ukrainian": "Ucraniano",
"Urdu": "Urdu",
"Uzbek": "Uzbeko",
"Vietnamese": "Vietnamita",
"Welsh": "Galés",
"Western Frisian": "Frisón",
"Xhosa": "Xhosa",
"Yiddish": "Yidis",
"Yoruba": "Yoruba",
"Zulu": "Zulú",
"`x` years": "`x` años",
"`x` months": "`x` meses",
"`x` weeks": "`x` semanas",
"`x` days": "`x` días",
"`x` hours": "`x` horas",
"`x` minutes": "`x` minutos",
"`x` seconds": "`x` segundos",
"Fallback comments: ": "Comentarios alternativos: ",
"Popular": "Populares",
"Top": "Destacados",
"About": "Acerca de",
"Rating: ": "Valoración: ",
"Language: ": "Idioma: ",
"View as playlist": "Ver como lista de reproducción",
"Default": "Por defecto",
"Music": "Música",
"Gaming": "Videojuegos",
"News": "Noticias",
"Movies": "Películas",
"Download": "Descargar",
"Download as: ": "Descargar como: ",
"%A %B %-d, %Y": "%A %B %-d, %Y",
"(edited)": "(editado)",
"YouTube comment permalink": "Enlace permanente de YouTube del comentario",
"`x` marked it with a ❤": "`x` lo ha marcado con un ❤",
"Audio mode": "Modo de audio",
"Video mode": "Modo de vídeo",
"Videos": "Vídeos",
"Playlists": "Listas de reproducción",
"Current version: ": "Versión actual: "
}

View File

@ -1,295 +1,313 @@
{ {
"`x` subscribers": "`x` harpidedun", "`x` subscribers": "`x` harpidedun",
"`x` videos": "`x` bideo", "`x` videos": "`x` bideo",
"LIVE": "ZUZENEAN", "LIVE": "ZUZENEAN",
"Shared `x` ago": "Duela `x` partekatua", "Shared `x` ago": "Duela `x` partekatua",
"Unsubscribe": "Harpidetza kendu", "Unsubscribe": "Harpidetza kendu",
"Subscribe": "Harpidetu", "Subscribe": "Harpidetu",
"Login to subscribe to `x`": "Saioa hasi `x`(e)ra harpidetzeko", "View channel on YouTube": "Ikusi kanala YouTuben",
"View channel on YouTube": "Ikusi kanala YouTuben", "View playlist on YouTube": "",
"newest": "berrienak", "newest": "berrienak",
"oldest": "zaharrenak", "oldest": "zaharrenak",
"popular": "ospetsuenak", "popular": "ospetsuenak",
"last": "", "last": "azkena",
"Next page": "Hurrengo orria", "Next page": "Hurrengo orria",
"Previous page": "Aurreko orria", "Previous page": "Aurreko orria",
"Clear watch history?": "Garbitu ikusitakoen historia?", "Clear watch history?": "Garbitu ikusitakoen historia?",
"Yes": "Bai", "New password": "Pasahitz berria",
"No": "Ez", "New passwords must match": "",
"Import and Export Data": "Datuak inportatu eta esportatu", "Cannot change password for Google accounts": "",
"Import": "Inportatu", "Authorize token?": "",
"Import Invidious data": "Invidiouseko datuak inportatu", "Authorize token for `x`?": "",
"Import YouTube subscriptions": "YouTubeko harpidetzak inportatu", "Yes": "Bai",
"Import FreeTube subscriptions (.db)": "FreeTubeko harpidetzak inportatu (.db)", "No": "Ez",
"Import NewPipe subscriptions (.json)": "NewPipeko harpidetzak inportatu (.json)", "Import and Export Data": "Datuak inportatu eta esportatu",
"Import NewPipe data (.zip)": "NewPipeko datuak inportatu (.zip)", "Import": "Inportatu",
"Export": "Esportatu", "Import Invidious data": "Invidiouseko datuak inportatu",
"Export subscriptions as OPML": "Esportatu harpidetzak OPML bezala", "Import YouTube subscriptions": "YouTubeko harpidetzak inportatu",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Harpidetzak OPML bezala esportatu (NewPipe eta FreeTuberako)", "Import FreeTube subscriptions (.db)": "FreeTubeko harpidetzak inportatu (.db)",
"Export data as JSON": "Datuak JSON bezala esportatu", "Import NewPipe subscriptions (.json)": "NewPipeko harpidetzak inportatu (.json)",
"Delete account?": "Kontua ezabatu?", "Import NewPipe data (.zip)": "NewPipeko datuak inportatu (.zip)",
"History": "Historia", "Export": "Esportatu",
"An alternative front-end to YouTube": "YouTuberako interfaze alternatibo bat", "Export subscriptions as OPML": "Esportatu harpidetzak OPML bezala",
"JavaScript license information": "JavaScript lizentzia informazioa", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Harpidetzak OPML bezala esportatu (NewPipe eta FreeTuberako)",
"source": "iturburua", "Export data as JSON": "Datuak JSON bezala esportatu",
"Login": "Saioa hasi", "Delete account?": "Kontua ezabatu?",
"Login/Register": "Saioa hasi/Izena eman", "History": "Historia",
"Login to Google": "Googlekin hasi saioa", "An alternative front-end to YouTube": "YouTuberako interfaze alternatibo bat",
"User ID:": "Erabiltzaile IDa:", "JavaScript license information": "JavaScript lizentzia informazioa",
"Password:": "Pasahitza:", "source": "iturburua",
"Time (h:mm:ss):": "Denbora (o:mm:ss):", "Log in": "Saioa hasi",
"Text CAPTCHA": "Testu CAPTCHA", "Log in/register": "Saioa hasi/Izena eman",
"Image CAPTCHA": "Irudi CAPTCHA", "Log in with Google": "Googlekin hasi saioa",
"Sign In": "", "User ID": "Erabiltzaile IDa",
"Register": "", "Password": "Pasahitza",
"Email:": "", "Time (h:mm:ss):": "Denbora (o:mm:ss):",
"Google verification code:": "", "Text CAPTCHA": "Testu CAPTCHA",
"Preferences": "", "Image CAPTCHA": "Irudi CAPTCHA",
"Player preferences": "", "Sign In": "",
"Always loop: ": "", "Register": "",
"Autoplay: ": "", "E-mail": "",
"Autoplay next video: ": "", "Google verification code": "",
"Listen by default: ": "", "Preferences": "",
"Proxy videos? ": "", "Player preferences": "",
"Default speed: ": "", "Always loop: ": "",
"Preferred video quality: ": "", "Autoplay: ": "",
"Player volume: ": "", "Play next by default: ": "",
"Default comments: ": "", "Autoplay next video: ": "",
"Default captions: ": "", "Listen by default: ": "",
"Fallback captions: ": "", "Proxy videos? ": "",
"Show related videos? ": "", "Default speed: ": "",
"Visual preferences": "", "Preferred video quality: ": "",
"Dark mode: ": "", "Player volume: ": "",
"Thin mode: ": "", "Default comments: ": "",
"Subscription preferences": "", "youtube": "",
"Redirect homepage to feed: ": "", "reddit": "",
"Number of videos shown in feed: ": "", "Default captions: ": "",
"Sort videos by: ": "", "Fallback captions: ": "",
"published": "", "Show related videos? ": "",
"published - reverse": "", "Show annotations by default? ": "",
"alphabetically": "", "Visual preferences": "",
"alphabetically - reverse": "", "Dark mode: ": "",
"channel name": "", "Thin mode: ": "",
"channel name - reverse": "", "Subscription preferences": "",
"Only show latest video from channel: ": "", "Show annotations by default for subscribed channels? ": "",
"Only show latest unwatched video from channel: ": "", "Redirect homepage to feed: ": "",
"Only show unwatched: ": "", "Number of videos shown in feed: ": "",
"Only show notifications (if there are any): ": "", "Sort videos by: ": "",
"Data preferences": "", "published": "",
"Clear watch history": "", "published - reverse": "",
"Import/Export data": "", "alphabetically": "",
"Manage subscriptions": "", "alphabetically - reverse": "",
"Watch history": "", "channel name": "",
"Delete account": "", "channel name - reverse": "",
"Administrator preferences": "", "Only show latest video from channel: ": "",
"Default homepage: ": "", "Only show latest unwatched video from channel: ": "",
"Feed menu: ": "", "Only show unwatched: ": "",
"Top enabled? ": "", "Only show notifications (if there are any): ": "",
"CAPTCHA enabled? ": "", "Data preferences": "",
"Login enabled? ": "", "Clear watch history": "",
"Registration enabled? ": "", "Import/export data": "",
"Report statistics? ": "", "Change password": "",
"Save preferences": "", "Manage subscriptions": "",
"Subscription manager": "", "Manage tokens": "",
"`x` subscriptions": "", "Watch history": "",
"Import/Export": "", "Delete account": "",
"unsubscribe": "", "Administrator preferences": "",
"Subscriptions": "", "Default homepage: ": "",
"`x` unseen notifications": "", "Feed menu: ": "",
"search": "", "Top enabled? ": "",
"Sign out": "", "CAPTCHA enabled? ": "",
"Released under the AGPLv3 by Omar Roth.": "", "Login enabled? ": "",
"Source available here.": "", "Registration enabled? ": "",
"View JavaScript license information.": "", "Report statistics? ": "",
"View privacy policy.": "", "Save preferences": "",
"Unlisted": "", "Subscription manager": "",
"Trending": "", "Token manager": "",
"Watch video on Youtube": "", "Token": "",
"Genre: ": "", "`x` subscriptions": "",
"License: ": "", "`x` tokens": "",
"Family friendly? ": "", "Import/export": "",
"Wilson score: ": "", "unsubscribe": "",
"Engagement: ": "", "revoke": "",
"Whitelisted regions: ": "", "Subscriptions": "",
"Blacklisted regions: ": "", "`x` unseen notifications": "",
"Shared `x`": "", "search": "",
"Premieres in `x`": "", "Log out": "",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "", "Released under the AGPLv3 by Omar Roth.": "",
"View YouTube comments": "", "Source available here.": "",
"View more comments on Reddit": "", "View JavaScript license information.": "",
"View `x` comments": "", "View privacy policy.": "",
"View Reddit comments": "", "Trending": "",
"Hide replies": "", "Unlisted": "",
"Show replies": "", "Watch on YouTube": "",
"Incorrect password": "", "Hide annotations": "",
"Quota exceeded, try again in a few hours": "", "Show annotations": "",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "", "Genre: ": "",
"Invalid TFA code": "", "License: ": "",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "", "Family friendly? ": "",
"Invalid answer": "", "Wilson score: ": "",
"Invalid CAPTCHA": "", "Engagement: ": "",
"CAPTCHA is a required field": "", "Whitelisted regions: ": "",
"User ID is a required field": "", "Blacklisted regions: ": "",
"Password is a required field": "", "Shared `x`": "",
"Invalid username or password": "", "`x` views": "",
"Please sign in using 'Sign in with Google'": "", "Premieres in `x`": "",
"Password cannot be empty": "", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "",
"Password cannot be longer than 55 characters": "", "View YouTube comments": "",
"Please sign in": "", "View more comments on Reddit": "",
"Invidious Private Feed for `x`": "", "View `x` comments": "",
"channel:`x`": "", "View Reddit comments": "",
"Deleted or invalid channel": "", "Hide replies": "",
"This channel does not exist.": "", "Show replies": "",
"Could not get channel info.": "", "Incorrect password": "",
"Could not fetch comments": "", "Quota exceeded, try again in a few hours": "",
"View `x` replies": "", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "",
"`x` ago": "", "Invalid TFA code": "",
"Load more": "", "Login failed. This may be because two-factor authentication is not turned on for your account.": "",
"`x` points": "", "Wrong answer": "",
"Could not create mix.": "", "Erroneous CAPTCHA": "",
"Playlist is empty": "", "CAPTCHA is a required field": "",
"Invalid playlist.": "", "User ID is a required field": "",
"Playlist does not exist.": "", "Password is a required field": "",
"Could not pull trending pages.": "", "Wrong username or password": "",
"Hidden field \"challenge\" is a required field": "", "Please sign in using 'Log in with Google'": "",
"Hidden field \"token\" is a required field": "", "Password cannot be empty": "",
"Invalid challenge": "", "Password cannot be longer than 55 characters": "",
"Invalid token": "", "Please log in": "",
"Invalid user": "", "Invidious Private Feed for `x`": "",
"Token is expired, please try again": "", "channel:`x`": "",
"English": "", "Deleted or invalid channel": "",
"English (auto-generated)": "", "This channel does not exist.": "",
"Afrikaans": "", "Could not get channel info.": "",
"Albanian": "", "Could not fetch comments": "",
"Amharic": "", "View `x` replies": "",
"Arabic": "", "`x` ago": "",
"Armenian": "", "Load more": "",
"Azerbaijani": "", "`x` points": "",
"Bangla": "", "Could not create mix.": "",
"Basque": "", "Empty playlist": "",
"Belarusian": "", "Not a playlist.": "",
"Bosnian": "", "Playlist does not exist.": "",
"Bulgarian": "", "Could not pull trending pages.": "",
"Burmese": "", "Hidden field \"challenge\" is a required field": "",
"Catalan": "", "Hidden field \"token\" is a required field": "",
"Cebuano": "", "Erroneous challenge": "",
"Chinese (Simplified)": "", "Erroneous token": "",
"Chinese (Traditional)": "", "No such user": "",
"Corsican": "", "Token is expired, please try again": "",
"Croatian": "", "English": "",
"Czech": "", "English (auto-generated)": "",
"Danish": "", "Afrikaans": "",
"Dutch": "", "Albanian": "",
"Esperanto": "", "Amharic": "",
"Estonian": "", "Arabic": "",
"Filipino": "", "Armenian": "",
"Finnish": "", "Azerbaijani": "",
"French": "", "Bangla": "",
"Galician": "", "Basque": "",
"Georgian": "", "Belarusian": "",
"German": "", "Bosnian": "",
"Greek": "", "Bulgarian": "",
"Gujarati": "", "Burmese": "",
"Haitian Creole": "", "Catalan": "",
"Hausa": "", "Cebuano": "",
"Hawaiian": "", "Chinese (Simplified)": "",
"Hebrew": "", "Chinese (Traditional)": "",
"Hindi": "", "Corsican": "",
"Hmong": "", "Croatian": "",
"Hungarian": "", "Czech": "",
"Icelandic": "", "Danish": "",
"Igbo": "", "Dutch": "",
"Indonesian": "", "Esperanto": "",
"Irish": "", "Estonian": "",
"Italian": "", "Filipino": "",
"Japanese": "", "Finnish": "",
"Javanese": "", "French": "",
"Kannada": "", "Galician": "",
"Kazakh": "", "Georgian": "",
"Khmer": "", "German": "",
"Korean": "", "Greek": "",
"Kurdish": "", "Gujarati": "",
"Kyrgyz": "", "Haitian Creole": "",
"Lao": "", "Hausa": "",
"Latin": "", "Hawaiian": "",
"Latvian": "", "Hebrew": "",
"Lithuanian": "", "Hindi": "",
"Luxembourgish": "", "Hmong": "",
"Macedonian": "", "Hungarian": "",
"Malagasy": "", "Icelandic": "",
"Malay": "", "Igbo": "",
"Malayalam": "", "Indonesian": "",
"Maltese": "", "Irish": "",
"Maori": "", "Italian": "",
"Marathi": "", "Japanese": "",
"Mongolian": "", "Javanese": "",
"Nepali": "", "Kannada": "",
"Norwegian": "", "Kazakh": "",
"Nyanja": "", "Khmer": "",
"Pashto": "", "Korean": "",
"Persian": "", "Kurdish": "",
"Polish": "", "Kyrgyz": "",
"Portuguese": "", "Lao": "",
"Punjabi": "", "Latin": "",
"Romanian": "", "Latvian": "",
"Russian": "", "Lithuanian": "",
"Samoan": "", "Luxembourgish": "",
"Scottish Gaelic": "", "Macedonian": "",
"Serbian": "", "Malagasy": "",
"Shona": "", "Malay": "",
"Sindhi": "", "Malayalam": "",
"Sinhala": "", "Maltese": "",
"Slovak": "", "Maori": "",
"Slovenian": "", "Marathi": "",
"Somali": "", "Mongolian": "",
"Southern Sotho": "", "Nepali": "",
"Spanish": "", "Norwegian Bokmål": "",
"Spanish (Latin America)": "", "Nyanja": "",
"Sundanese": "", "Pashto": "",
"Swahili": "", "Persian": "",
"Swedish": "", "Polish": "",
"Tajik": "", "Portuguese": "",
"Tamil": "", "Punjabi": "",
"Telugu": "", "Romanian": "",
"Thai": "", "Russian": "",
"Turkish": "", "Samoan": "",
"Ukrainian": "", "Scottish Gaelic": "",
"Urdu": "", "Serbian": "",
"Uzbek": "", "Shona": "",
"Vietnamese": "", "Sindhi": "",
"Welsh": "", "Sinhala": "",
"Western Frisian": "", "Slovak": "",
"Xhosa": "", "Slovenian": "",
"Yiddish": "", "Somali": "",
"Yoruba": "", "Southern Sotho": "",
"Zulu": "", "Spanish": "",
"`x` years": "", "Spanish (Latin America)": "",
"`x` months": "", "Sundanese": "",
"`x` weeks": "", "Swahili": "",
"`x` days": "", "Swedish": "",
"`x` hours": "", "Tajik": "",
"`x` minutes": "", "Tamil": "",
"`x` seconds": "", "Telugu": "",
"Fallback comments: ": "", "Thai": "",
"Popular": "", "Turkish": "",
"Top": "", "Ukrainian": "",
"About": "", "Urdu": "",
"Rating: ": "", "Uzbek": "",
"Language: ": "", "Vietnamese": "",
"Default": "", "Welsh": "",
"Music": "", "Western Frisian": "",
"Gaming": "", "Xhosa": "",
"News": "", "Yiddish": "",
"Movies": "", "Yoruba": "",
"Download": "", "Zulu": "",
"Download as: ": "", "`x` years": "",
"%A %B %-d, %Y": "", "`x` months": "",
"(edited)": "", "`x` weeks": "",
"Youtube permalink of the comment": "", "`x` days": "",
"`x` marked it with a ❤": "", "`x` hours": "",
"Audio mode": "", "`x` minutes": "",
"Video mode": "", "`x` seconds": "",
"Videos": "", "Fallback comments: ": "",
"Playlists": "", "Popular": "",
"Current version: ": "" "Top": "",
} "About": "",
"Rating: ": "",
"Language: ": "",
"View as playlist": "",
"Default": "",
"Music": "",
"Gaming": "",
"News": "",
"Movies": "",
"Download": "",
"Download as: ": "",
"%A %B %-d, %Y": "",
"(edited)": "",
"YouTube comment permalink": "",
"`x` marked it with a ❤": "",
"Audio mode": "",
"Video mode": "",
"Videos": ""
}

View File

@ -1,295 +1,315 @@
{ {
"`x` subscribers": "`x` abonnés", "`x` subscribers": "`x` abonnés",
"`x` videos": "`x` vidéos", "`x` videos": "`x` vidéos",
"LIVE": "EN DIRECT", "LIVE": "EN DIRECT",
"Shared `x` ago": "Ajoutée il y a `x`", "Shared `x` ago": "Ajoutée il y a `x`",
"Unsubscribe": "Se désabonner", "Unsubscribe": "Se désabonner",
"Subscribe": "S'abonner", "Subscribe": "S'abonner",
"Login to subscribe to `x`": "Vous devez vous connecter pour vous abonner à `x`", "View channel on YouTube": "Voir la chaîne sur YouTube",
"View channel on YouTube": "Voir la chaîne sur YouTube", "View playlist on YouTube": "",
"newest": "Date d'ajout (la plus récente)", "newest": "Date d'ajout (la plus récente)",
"oldest": "Date d'ajout (la plus ancienne)", "oldest": "Date d'ajout (la plus ancienne)",
"popular": "Les plus populaires", "popular": "Les plus populaires",
"last": "Dernières", "last": "Dernières",
"Next page": "Page suivante", "Next page": "Page suivante",
"Previous page": "Page précédente", "Previous page": "Page précédente",
"Clear watch history?": "Êtes-vous sûr de vouloir supprimer l'historique des vidéos regardées ?", "Clear watch history?": "Êtes-vous sûr de vouloir supprimer l'historique des vidéos regardées ?",
"Yes": "Oui", "New password": "Nouveau mot de passe",
"No": "Non", "New passwords must match": "Les nouveaux mots de passe doivent être identiques",
"Import and Export Data": "Importer et exporter des données", "Cannot change password for Google accounts": "Le mot de passe d'un compte Google ne peut pas être changé",
"Import": "Importer", "Authorize token?": "Autoriser le token ?",
"Import Invidious data": "Importer des données Invidious", "Authorize token for `x`?": "Autoriser le token pour `x` ?",
"Import YouTube subscriptions": "Importer des abonnements YouTube", "Yes": "Oui",
"Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)", "No": "Non",
"Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)", "Import and Export Data": "Importer et exporter des données",
"Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)", "Import": "Importer",
"Export": "Exporter", "Import Invidious data": "Importer des données Invidious",
"Export subscriptions as OPML": "Exporter les abonnements en OPML", "Import YouTube subscriptions": "Importer des abonnements YouTube",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporter les abonnements en OPML (pour NewPipe & FreeTube)", "Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)",
"Export data as JSON": "Exporter les données au format JSON", "Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)",
"Delete account?": "Êtes-vous sûr de vouloir supprimer votre compte ?", "Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)",
"History": "Historique", "Export": "Exporter",
"An alternative front-end to YouTube": "Un front-end alternatif à YouTube", "Export subscriptions as OPML": "Exporter les abonnements en OPML",
"JavaScript license information": "Informations sur les licences JavaScript", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporter les abonnements en OPML (pour NewPipe & FreeTube)",
"source": "source", "Export data as JSON": "Exporter les données au format JSON",
"Login": "Se connecter", "Delete account?": "Êtes-vous sûr de vouloir supprimer votre compte ?",
"Login/Register": "Se connecter/Créer un compte", "History": "Historique",
"Login to Google": "Se connecter avec Google", "An alternative front-end to YouTube": "Un front-end alternatif à YouTube",
"User ID:": "Identifiant utilisateur :", "JavaScript license information": "Informations sur les licences JavaScript",
"Password:": "Mot de passe :", "source": "source",
"Time (h:mm:ss):": "Heure (h:mm:ss) :", "Log in": "Se connecter",
"Text CAPTCHA": "CAPTCHA Texte", "Log in/register": "Se connecter/Créer un compte",
"Image CAPTCHA": "CAPTCHA Image", "Log in with Google": "Se connecter avec Google",
"Sign In": "Se connecter", "User ID": "Identifiant utilisateur",
"Register": "S'inscrire", "Password": "Mot de passe",
"Email:": "E-mail :", "Time (h:mm:ss):": "Heure (h:mm:ss) :",
"Google verification code:": "Code de vérification Google :", "Text CAPTCHA": "CAPTCHA Texte",
"Preferences": "Préférences", "Image CAPTCHA": "CAPTCHA Image",
"Player preferences": "Préférences du lecteur", "Sign In": "Se connecter",
"Always loop: ": "Lire en boucle : ", "Register": "S'inscrire",
"Autoplay: ": "Lire automatiquement : ", "E-mail": "E-mail",
"Autoplay next video: ": "Lire automatiquement la vidéo suivante : ", "Google verification code": "Code de vérification Google",
"Listen by default: ": "Audio uniquement : ", "Preferences": "Préférences",
"Proxy videos? ": "Charger les vidéos à travers un proxy ? ", "Player preferences": "Préférences du lecteur",
"Default speed: ": "Vitesse par défaut : ", "Always loop: ": "Lire en boucle : ",
"Preferred video quality: ": "Qualité vidéo souhaitée : ", "Autoplay: ": "Lire automatiquement : ",
"Player volume: ": "Volume du lecteur : ", "Play next by default: ": "Jouer suirvante par défaut : ",
"Default comments: ": "Source des commentaires : ", "Autoplay next video: ": "Lire automatiquement la vidéo suivante : ",
"Default captions: ": "Sous-titres par défaut : ", "Listen by default: ": "Audio uniquement : ",
"Fallback captions: ": "Fallback captions: ", "Proxy videos? ": "Charger les vidéos à travers un proxy ? ",
"Show related videos? ": "Voir les vidéos liées ? ", "Default speed: ": "Vitesse par défaut : ",
"Visual preferences": "Préférences du site", "Preferred video quality: ": "Qualité vidéo souhaitée : ",
"Dark mode: ": "Mode Sombre : ", "Player volume: ": "Volume du lecteur : ",
"Thin mode: ": "Mode Simplifié : ", "Default comments: ": "Source des commentaires : ",
"Subscription preferences": "Préférences de la page d'abonnements", "youtube": "YouTube",
"Redirect homepage to feed: ": "Rediriger la page d'accueil vers la page d'abonnements : ", "reddit": "Reddit",
"Number of videos shown in feed: ": "Nombre de vidéos montrées dans la page d'abonnements : ", "Default captions: ": "Sous-titres par défaut : ",
"Sort videos by: ": "Trier les vidéos par : ", "Fallback captions: ": "Sous-titres de repli : ",
"published": "publication", "Show related videos? ": "Voir les vidéos liées ? ",
"published - reverse": "publication - inversé", "Show annotations by default? ": "Voir les annotations par défaut ? ",
"alphabetically": "alphabétiquement", "Visual preferences": "Préférences du site",
"alphabetically - reverse": "alphabétiquement - inversé", "Dark mode: ": "Mode Sombre : ",
"channel name": "nom de la chaîne", "Thin mode: ": "Mode Simplifié : ",
"channel name - reverse": "nom de la chaîne - inversé", "Subscription preferences": "Préférences de la page d'abonnements",
"Only show latest video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne : ", "Show annotations by default for subscribed channels? ": "Voir les annotations par défaut sur les chaînes suivies ? ",
"Only show latest unwatched video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne non regardée : ", "Redirect homepage to feed: ": "Rediriger la page d'accueil vers la page d'abonnements : ",
"Only show unwatched: ": "Afficher uniquement les vidéos non regardées : ", "Number of videos shown in feed: ": "Nombre de vidéos montrées dans la page d'abonnements : ",
"Only show notifications (if there are any): ": "Afficher uniquement les notifications (s'il y en a) : ", "Sort videos by: ": "Trier les vidéos par : ",
"Data preferences": "Préférences liées aux données", "published": "date de publication",
"Clear watch history": "Supprimer l'historique des vidéos regardées", "published - reverse": "date de publication - inversé",
"Import/Export data": "Importer/exporter les données", "alphabetically": "alphabétiquement",
"Manage subscriptions": "Gérer les abonnements", "alphabetically - reverse": "alphabétiquement - inversé",
"Watch history": "Historique de visionnage", "channel name": "nom de la chaîne",
"Delete account": "Supprimer votre compte", "channel name - reverse": "nom de la chaîne - inversé",
"Administrator preferences": "Préferences d'Administrateur", "Only show latest video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne : ",
"Default homepage: ": "Page d'accueil par défaut : ", "Only show latest unwatched video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne non regardée : ",
"Feed menu: ": "Menu des Flux : ", "Only show unwatched: ": "Afficher uniquement les vidéos non regardées : ",
"Top enabled? ": "Top activé ? ", "Only show notifications (if there are any): ": "Afficher uniquement les notifications (s'il y en a) : ",
"CAPTCHA enabled? ": "CAPTCHA activé ? ", "Data preferences": "Préférences liées aux données",
"Login enabled? ": "Connexion activé ? ", "Clear watch history": "Supprimer l'historique des vidéos regardées",
"Registration enabled? ": "Inscription activée ? ", "Import/export data": "Importer/exporter les données",
"Report statistics? ": "Télémétrie activé ? ", "Change password": "Modifier le mot de passe",
"Save preferences": "Enregistrer les préférences", "Manage subscriptions": "Gérer les abonnements",
"Subscription manager": "Gestionnaire d'abonnement", "Manage tokens": "Gérer les tokens",
"`x` subscriptions": "`x` abonnements", "Watch history": "Historique de visionnage",
"Import/Export": "Importer/Exporter", "Delete account": "Supprimer votre compte",
"unsubscribe": "se désabonner", "Administrator preferences": "Préferences d'Administrateur",
"Subscriptions": "Abonnements", "Default homepage: ": "Page d'accueil par défaut : ",
"`x` unseen notifications": "`x` notifications non vues", "Feed menu: ": "Menu des Flux : ",
"search": "Rechercher", "Top enabled? ": "Top activé ? ",
"Sign out": "Déconnexion", "CAPTCHA enabled? ": "CAPTCHA activé ? ",
"Released under the AGPLv3 by Omar Roth.": "Publié sous licence AGPLv3 par Omar Roth.", "Login enabled? ": "Connexion activé ? ",
"Source available here.": "Code Source.", "Registration enabled? ": "Inscription activée ? ",
"View JavaScript license information.": "Voir les informations des licences JavaScript.", "Report statistics? ": "Télémétrie activé ? ",
"View privacy policy.": "Politique de confidentialité", "Save preferences": "Enregistrer les préférences",
"Trending": "Tendances", "Subscription manager": "Gestionnaire d'abonnement",
"Unlisted": "Non répertoriée", "Token manager": "Gestionnaire de tokens",
"Watch video on Youtube": "Voir la vidéo sur Youtube", "Token": "Token",
"Genre: ": "Genre : ", "`x` subscriptions": "`x` abonnements",
"License: ": "Licence : ", "`x` tokens": "`x` tokens",
"Family friendly? ": "Tout Public ? ", "Import/export": "Importer/Exporter",
"Wilson score: ": "Score de Wilson : ", "unsubscribe": "se désabonner",
"Engagement: ": "Poucentage de spectateur aillant aimé Like ou Dislike la vidéo : ", "revoke": "annuler",
"Whitelisted regions: ": "Régions en liste blanche : ", "Subscriptions": "Abonnements",
"Blacklisted regions: ": "Régions sur liste noire : ", "`x` unseen notifications": "`x` notifications non vues",
"Shared `x`": "Ajoutée le `x`", "search": "Rechercher",
"Premieres in `x`": "Première dans `x`", "Log out": "Déconnexion",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Il semblerait que JavaScript soit désactivé. Cliquez ici pour voir les commentaires sans. Gardez à l'esprit que le chargement peut prendre plus de temps.", "Released under the AGPLv3 by Omar Roth.": "Publié sous licence AGPLv3 par Omar Roth.",
"View YouTube comments": "Voir les commentaires YouTube", "Source available here.": "Code Source.",
"View more comments on Reddit": "Voir plus de commentaires sur Reddit", "View JavaScript license information.": "Voir les informations des licences JavaScript.",
"View `x` comments": "Voir `x` commentaires", "View privacy policy.": "Voir la politique de confidentialité.",
"View Reddit comments": "Voir les commentaires Reddit", "Trending": "Tendances",
"Hide replies": "Masquer les réponses", "Unlisted": "Non répertoriée",
"Show replies": "Afficher les réponses", "Watch on YouTube": "Voir la vidéo sur Youtube",
"Incorrect password": "Mot de passe incorrect", "Hide annotations": "Masquer les annotations",
"Quota exceeded, try again in a few hours": "Nombre de tentative de connexion dépassé, réessayez dans quelques heures", "Show annotations": "Afficher les annotations",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Si vous ne parvenez pas à vous connecter, assurez-vous que l'authentification à deux facteurs (Authenticator ou SMS) est activée.", "Genre: ": "Genre : ",
"Invalid TFA code": "Code d'authentification à deux facteurs invalide", "License: ": "Licence : ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "La connexion a échoué. Cela peut être dû au fait que l'authentification à deux facteurs n'est pas activée sur votre compte.", "Family friendly? ": "Tout Public ? ",
"Invalid answer": "Réponse invalide", "Wilson score: ": "Score de Wilson : ",
"Invalid CAPTCHA": "CAPTCHA invalide", "Engagement: ": "Poucentage de spectateur aillant aimé Like ou Dislike la vidéo : ",
"CAPTCHA is a required field": "Veuillez entrer un CAPTCHA", "Whitelisted regions: ": "Régions en liste blanche : ",
"User ID is a required field": "Veuillez entrer un Identifiant Utilisateur", "Blacklisted regions: ": "Régions sur liste noire : ",
"Password is a required field": "Veuillez entrer un Mot de passe", "Shared `x`": "Ajoutée le `x`",
"Invalid username or password": "Nom d'utilisateur ou mot de passe invalide", "`x` views": "`x` vues",
"Please sign in using 'Sign in with Google'": "Veuillez vous connecter en utilisant \"Se connecter avec Google\"", "Premieres in `x`": "Première dans `x`",
"Password cannot be empty": "Le mot de passe ne peut pas être vide", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Il semblerait que JavaScript soit désactivé. Cliquez ici pour voir les commentaires sans. Gardez à l'esprit que le chargement peut prendre plus de temps.",
"Password cannot be longer than 55 characters": "Le mot de passe ne doit pas comporter plus de 55 caractères", "View YouTube comments": "Voir les commentaires YouTube",
"Please sign in": "Veuillez vous connecter", "View more comments on Reddit": "Voir plus de commentaires sur Reddit",
"Invidious Private Feed for `x`": "Flux RSS privé pour `x`", "View `x` comments": "Voir `x` commentaires",
"channel:`x`": "chaîne:`x`", "View Reddit comments": "Voir les commentaires Reddit",
"Deleted or invalid channel": "Chaîne supprimée ou invalide", "Hide replies": "Masquer les réponses",
"This channel does not exist.": "Cette chaine n'existe pas.", "Show replies": "Afficher les réponses",
"Could not get channel info.": "Impossible de charger les informations de cette chaîne.", "Incorrect password": "Mot de passe incorrect",
"Could not fetch comments": "Impossible de charger les commentaires", "Quota exceeded, try again in a few hours": "Nombre de tentative de connexion dépassé, réessayez dans quelques heures",
"View `x` replies": "Voir `x` réponses", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Si vous ne parvenez pas à vous connecter, assurez-vous que l'authentification à deux facteurs (Authenticator ou SMS) est activée.",
"`x` ago": "il y a `x`", "Invalid TFA code": "Code d'authentification à deux facteurs invalide",
"Load more": "Charger plus", "Login failed. This may be because two-factor authentication is not turned on for your account.": "La connexion a échoué. Cela peut être dû au fait que l'authentification à deux facteurs n'est pas activée sur votre compte.",
"`x` points": "`x` points", "Wrong answer": "Réponse invalide",
"Could not create mix.": "Impossible de charger cette liste de lecture.", "Erroneous CAPTCHA": "CAPTCHA invalide",
"Playlist is empty": "La liste de lecture est vide", "CAPTCHA is a required field": "Veuillez entrer un CAPTCHA",
"Invalid playlist.": "Liste de lecture invalide.", "User ID is a required field": "Veuillez entrer un Identifiant Utilisateur",
"Playlist does not exist.": "La liste de lecture n'existe pas.", "Password is a required field": "Veuillez entrer un Mot de passe",
"Could not pull trending pages.": "Impossible de charger les pages de tendances.", "Wrong username or password": "Nom d'utilisateur ou mot de passe invalide",
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field", "Please sign in using 'Log in with Google'": "Veuillez vous connecter en utilisant \"Se connecter avec Google\"",
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field", "Password cannot be empty": "Le mot de passe ne peut pas être vide",
"Invalid challenge": "Invalid challenge", "Password cannot be longer than 55 characters": "Le mot de passe ne doit pas comporter plus de 55 caractères",
"Invalid token": "Invalid token", "Please log in": "Veuillez vous connecter",
"Invalid user": "Invalid user", "Invidious Private Feed for `x`": "Flux RSS privé pour `x`",
"Token is expired, please try again": "Token is expired, please try again", "channel:`x`": "chaîne:`x`",
"English": "Anglais", "Deleted or invalid channel": "Chaîne supprimée ou invalide",
"English (auto-generated)": "Anglais (générés automatiquement)", "This channel does not exist.": "Cette chaine n'existe pas.",
"Afrikaans": "Afrikaans", "Could not get channel info.": "Impossible de charger les informations de cette chaîne.",
"Albanian": "Albanais", "Could not fetch comments": "Impossible de charger les commentaires",
"Amharic": "Amharique", "View `x` replies": "Voir `x` réponses",
"Arabic": "Arabe", "`x` ago": "il y a `x`",
"Armenian": "Arménien", "Load more": "Charger plus",
"Azerbaijani": "Azerbaïdjanais", "`x` points": "`x` points",
"Bangla": "Bangla", "Could not create mix.": "Impossible de charger cette liste de lecture.",
"Basque": "Basque", "Empty playlist": "La liste de lecture est vide",
"Belarusian": "Belarusian", "Not a playlist.": "Liste de lecture invalide.",
"Bosnian": "Bosnian", "Playlist does not exist.": "La liste de lecture n'existe pas.",
"Bulgarian": "Bulgarian", "Could not pull trending pages.": "Impossible de charger les pages de tendances.",
"Burmese": "Birman", "Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
"Catalan": "Catalan", "Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
"Cebuano": "Cebuano", "Erroneous challenge": "Erroneous challenge",
"Chinese (Simplified)": "Chinois (Simplifié)", "Erroneous token": "Erroneous token",
"Chinese (Traditional)": "Chinois (Traditionnel)", "No such user": "No such user",
"Corsican": "Corse", "Token is expired, please try again": "Token is expired, please try again",
"Croatian": "Croate", "English": "Anglais",
"Czech": "Tchèque", "English (auto-generated)": "Anglais (générés automatiquement)",
"Danish": "Danois", "Afrikaans": "Afrikaans",
"Dutch": "Hollandais", "Albanian": "Albanais",
"Esperanto": "Espéranto", "Amharic": "Amharique",
"Estonian": "Estonien", "Arabic": "Arabe",
"Filipino": "Philippin", "Armenian": "Arménien",
"Finnish": "Finlandais", "Azerbaijani": "Azerbaïdjanais",
"French": "Français", "Bangla": "Bangla",
"Galician": "Galicien", "Basque": "Basque",
"Georgian": "Géorgien", "Belarusian": "Belarusian",
"German": "Allemand", "Bosnian": "Bosnian",
"Greek": "Grec", "Bulgarian": "Bulgarian",
"Gujarati": "Gujarati", "Burmese": "Birman",
"Haitian Creole": "Créole Haïtien", "Catalan": "Catalan",
"Hausa": "Haoussa", "Cebuano": "Cebuano",
"Hawaiian": "Hawaïen", "Chinese (Simplified)": "Chinois (Simplifié)",
"Hebrew": "Hébraïque", "Chinese (Traditional)": "Chinois (Traditionnel)",
"Hindi": "Hindi", "Corsican": "Corse",
"Hmong": "Hmong", "Croatian": "Croate",
"Hungarian": "Hongrois", "Czech": "Tchèque",
"Icelandic": "Islandais", "Danish": "Danois",
"Igbo": "Igbo", "Dutch": "Hollandais",
"Indonesian": "Indonésien", "Esperanto": "Espéranto",
"Irish": "Irlandais", "Estonian": "Estonien",
"Italian": "Italien", "Filipino": "Philippin",
"Japanese": "Japonais", "Finnish": "Finlandais",
"Javanese": "Javanais", "French": "Français",
"Kannada": "Kannada", "Galician": "Galicien",
"Kazakh": "Kazakh", "Georgian": "Géorgien",
"Khmer": "Khmer", "German": "Allemand",
"Korean": "Coréen", "Greek": "Grec",
"Kurdish": "Kurde", "Gujarati": "Gujarati",
"Kyrgyz": "Kirghize", "Haitian Creole": "Créole Haïtien",
"Lao": "Lao", "Hausa": "Haoussa",
"Latin": "Latin", "Hawaiian": "Hawaïen",
"Latvian": "Letton", "Hebrew": "Hébraïque",
"Lithuanian": "Lituanien", "Hindi": "Hindi",
"Luxembourgish": "Luxembourgeois", "Hmong": "Hmong",
"Macedonian": "Macédonien", "Hungarian": "Hongrois",
"Malagasy": "Malgache", "Icelandic": "Islandais",
"Malay": "Malais", "Igbo": "Igbo",
"Malayalam": "Malayalam", "Indonesian": "Indonésien",
"Maltese": "Maltais", "Irish": "Irlandais",
"Maori": "Maori", "Italian": "Italien",
"Marathi": "Marathi", "Japanese": "Japonais",
"Mongolian": "Mongol", "Javanese": "Javanais",
"Nepali": "Népalais", "Kannada": "Kannada",
"Norwegian": "Norvégien", "Kazakh": "Kazakh",
"Nyanja": "Nyanja", "Khmer": "Khmer",
"Pashto": "Pachtou", "Korean": "Coréen",
"Persian": "Persan", "Kurdish": "Kurde",
"Polish": "Polonais", "Kyrgyz": "Kirghize",
"Portuguese": "Portugais", "Lao": "Lao",
"Punjabi": "Punjabi", "Latin": "Latin",
"Romanian": "Roumain", "Latvian": "Letton",
"Russian": "Russe", "Lithuanian": "Lituanien",
"Samoan": "Samoan", "Luxembourgish": "Luxembourgeois",
"Scottish Gaelic": "Eaélique Ècossais", "Macedonian": "Macédonien",
"Serbian": "Serbe", "Malagasy": "Malgache",
"Shona": "Shona", "Malay": "Malais",
"Sindhi": "Sindhi", "Malayalam": "Malayalam",
"Sinhala": "Cinghalais", "Maltese": "Maltais",
"Slovak": "Slovaque", "Maori": "Maori",
"Slovenian": "Slovène", "Marathi": "Marathi",
"Somali": "Somalien", "Mongolian": "Mongol",
"Southern Sotho": "Sotho du Sud", "Nepali": "Népalais",
"Spanish": "Espagnol", "Norwegian Bokmål": "Norvégien",
"Spanish (Latin America)": "Espagnol (Amérique latine)", "Nyanja": "Nyanja",
"Sundanese": "Sundanais", "Pashto": "Pachtou",
"Swahili": "Swahili", "Persian": "Persan",
"Swedish": "Suédois", "Polish": "Polonais",
"Tajik": "Tajik", "Portuguese": "Portugais",
"Tamil": "Tamil", "Punjabi": "Punjabi",
"Telugu": "Telugu", "Romanian": "Roumain",
"Thai": "Thaï", "Russian": "Russe",
"Turkish": "Turc", "Samoan": "Samoan",
"Ukrainian": "Ukrainien", "Scottish Gaelic": "Eaélique Ècossais",
"Urdu": "Ourdou", "Serbian": "Serbe",
"Uzbek": "Ouzbek", "Shona": "Shona",
"Vietnamese": "Vietnamien", "Sindhi": "Sindhi",
"Welsh": "Gallois", "Sinhala": "Cinghalais",
"Western Frisian": "Frison occidental", "Slovak": "Slovaque",
"Xhosa": "Xhosa", "Slovenian": "Slovène",
"Yiddish": "Yiddish", "Somali": "Somalien",
"Yoruba": "Yoruba", "Southern Sotho": "Sotho du Sud",
"Zulu": "Zoulou", "Spanish": "Espagnol",
"`x` years": "`x` ans", "Spanish (Latin America)": "Espagnol (Amérique latine)",
"`x` months": "`x` mois", "Sundanese": "Sundanais",
"`x` weeks": "`x` semaines", "Swahili": "Swahili",
"`x` days": "`x` jours", "Swedish": "Suédois",
"`x` hours": "`x` heures", "Tajik": "Tajik",
"`x` minutes": "`x` minutes", "Tamil": "Tamil",
"`x` seconds": "`x` secondes", "Telugu": "Telugu",
"Fallback comments: ": "Fallback comments: ", "Thai": "Thaï",
"Popular": "Populaire", "Turkish": "Turc",
"Top": "Top", "Ukrainian": "Ukrainien",
"About": "A Propos", "Urdu": "Ourdou",
"Rating: ": "Évaluation : ", "Uzbek": "Ouzbek",
"Language: ": "Langue : ", "Vietnamese": "Vietnamien",
"Default": "Défaut", "Welsh": "Gallois",
"Music": "Musique", "Western Frisian": "Frison occidental",
"Gaming": "Jeux Vidéo", "Xhosa": "Xhosa",
"News": "Actualités", "Yiddish": "Yiddish",
"Movies": "Films", "Yoruba": "Yoruba",
"Download": "Télécharger", "Zulu": "Zoulou",
"Download as: ": "Télécharger en : ", "`x` years": "`x` ans",
"%A %B %-d, %Y": "%A %-d %B %Y", "`x` months": "`x` mois",
"(edited)": "(modifié)", "`x` weeks": "`x` semaines",
"Youtube permalink of the comment": "Lien YouTube permanent vers le commentaire", "`x` days": "`x` jours",
"`x` marked it with a ❤": "`x` l'a marqué d'un ❤", "`x` hours": "`x` heures",
"Audio mode": "Mode Audio", "`x` minutes": "`x` minutes",
"Video mode": "Mode Vidéo", "`x` seconds": "`x` secondes",
"Videos": "Vidéos", "Fallback comments: ": "Fallback comments: ",
"Playlists": "Liste de lecture", "Popular": "Populaire",
"Current version: ": "Version :" "Top": "Top",
} "About": "À propos",
"Rating: ": "Évaluation : ",
"Language: ": "Langue : ",
"View as playlist": "Voir en tant que liste de lecture",
"Default": "Défaut",
"Music": "Musique",
"Gaming": "Jeux Vidéo",
"News": "Actualités",
"Movies": "Films",
"Download": "Télécharger",
"Download as: ": "Télécharger en : ",
"%A %B %-d, %Y": "%A %-d %B %Y",
"(edited)": "(modifié)",
"YouTube comment permalink": "Lien YouTube permanent vers le commentaire",
"`x` marked it with a ❤": "`x` l'a marqué d'un ❤",
"Audio mode": "Mode Audio",
"Video mode": "Mode Vidéo",
"Videos": "Vidéos",
"Playlists": "Liste de lecture",
"Current version: ": "Version actuelle : "
}

View File

@ -1,295 +1,315 @@
{ {
"`x` subscribers": "`x` iscritti", "`x` subscribers": "`x` iscritti",
"`x` videos": "`x` video", "`x` videos": "`x` video",
"LIVE": "IN DIRETTA", "LIVE": "IN DIRETTA",
"Shared `x` ago": "Condiviso `x` fa", "Shared `x` ago": "Condiviso `x` fa",
"Unsubscribe": "Disiscriviti", "Unsubscribe": "Disiscriviti",
"Subscribe": "Iscriviti", "Subscribe": "Iscriviti",
"Login to subscribe to `x`": "Accedi per iscriverti a `x`", "View channel on YouTube": "Vedi canale su YouTube",
"View channel on YouTube": "Vedi canale su YouTube", "View playlist on YouTube": "",
"newest": "Data di aggiunta (più recente)", "newest": "Data di aggiunta (più recente)",
"oldest": "Data di aggiunta (più vecchia)", "oldest": "Data di aggiunta (più vecchia)",
"popular": "Tendenze", "popular": "Tendenze",
"last": "", "last": "durare",
"Next page": "Pagina successiva", "Next page": "Pagina successiva",
"Previous page": "Pagina precedente", "Previous page": "Pagina precedente",
"Clear watch history?": "Sei sicuro di voler cancellare la cronologia dei video guardati?", "Clear watch history?": "Sei sicuro di voler cancellare la cronologia dei video guardati?",
"Yes": "Si", "New password": "Nuova password",
"No": "No", "New passwords must match": "Le nuove password devono corrispondere",
"Import and Export Data": "Importazione ed esportazione dati", "Cannot change password for Google accounts": "Non è possibile modificare la password per gli account Google",
"Import": "Importa", "Authorize token?": "Autorizzare gettone?",
"Import Invidious data": "Importa dati Invidious", "Authorize token for `x`?": "",
"Import YouTube subscriptions": "Importa le iscrizioni da YouTube", "Yes": "Si",
"Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)", "No": "No",
"Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)", "Import and Export Data": "Importazione ed esportazione dati",
"Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)", "Import": "Importa",
"Export": "Esporta", "Import Invidious data": "Importa dati Invidious",
"Export subscriptions as OPML": "Esporta gli abbonamenti come OPML", "Import YouTube subscriptions": "Importa le iscrizioni da YouTube",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Esporta gli abbonamenti come OPML (per NewPipe e FreeTube)", "Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)",
"Export data as JSON": "Esporta i dati in formato JSON", "Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)",
"Delete account?": "Sei sicuro di voler cancellare l'account?", "Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)",
"History": "Cronologia", "Export": "Esporta",
"An alternative front-end to YouTube": "Un'interfaccia alternativa per YouTube", "Export subscriptions as OPML": "Esporta gli abbonamenti come OPML",
"JavaScript license information": "Info licenze JavaScript", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Esporta gli abbonamenti come OPML (per NewPipe e FreeTube)",
"source": "sorgente", "Export data as JSON": "Esporta i dati in formato JSON",
"Login": "Entra", "Delete account?": "Sei sicuro di voler cancellare l'account?",
"Login/Register": "Entra/Registrati", "History": "Cronologia",
"Login to Google": "Entra con Google", "An alternative front-end to YouTube": "Un'interfaccia alternativa per YouTube",
"User ID:": "ID utente:", "JavaScript license information": "Info licenze JavaScript",
"Password:": "Password:", "source": "sorgente",
"Time (h:mm:ss):": "Orario (h:mm:ss):", "Log in": "Entra",
"Text CAPTCHA": "Testo del CAPTCHA", "Log in/register": "Entra/Registrati",
"Image CAPTCHA": "Immagine CAPTCHA", "Log in with Google": "Entra con Google",
"Sign In": "Entra", "User ID": "ID utente",
"Register": "Registrati", "Password": "Password",
"Email:": "Email:", "Time (h:mm:ss):": "Orario (h:mm:ss):",
"Google verification code:": "Codice di verifica Google:", "Text CAPTCHA": "Testo del CAPTCHA",
"Preferences": "Preferenze", "Image CAPTCHA": "Immagine CAPTCHA",
"Player preferences": "Preferenze del riproduttore", "Sign In": "Entra",
"Always loop: ": "Ripeti sempre: ", "Register": "Registrati",
"Autoplay: ": "Riproduzione automatica: ", "E-mail": "Email",
"Autoplay next video: ": "Riproduci automaticamente il prossimo video: ", "Google verification code": "Codice di verifica Google",
"Listen by default: ": "Modalità solo audio come predefinita: ", "Preferences": "Preferenze",
"Proxy videos? ": "", "Player preferences": "Preferenze del riproduttore",
"Default speed: ": "Velocità di riproduzione predefinita: ", "Always loop: ": "Ripeti sempre: ",
"Preferred video quality: ": "Preferenza sulla qualità video: ", "Autoplay: ": "Riproduzione automatica: ",
"Player volume: ": "Volume di riproduzione: ", "Play next by default: ": "Riproduzione successiva per impostazione predefinita: ",
"Default comments: ": "Origine dei commenti: ", "Autoplay next video: ": "Riproduci automaticamente il prossimo video: ",
"Default captions: ": "Sottotitoli predefiniti: ", "Listen by default: ": "Modalità solo audio come predefinita: ",
"Fallback captions: ": "Sottotitoli alternativi: ", "Proxy videos? ": "",
"Show related videos? ": "Mostra video correlati? ", "Default speed: ": "Velocità di riproduzione predefinita: ",
"Visual preferences": "Preferenze grafiche", "Preferred video quality: ": "Preferenza sulla qualità video: ",
"Dark mode: ": "Tema scuro: ", "Player volume: ": "Volume di riproduzione: ",
"Thin mode: ": "Modalità per connessioni lente: ", "Default comments: ": "Origine dei commenti: ",
"Subscription preferences": "Preferenze iscrizioni", "youtube": "",
"Redirect homepage to feed: ": "Reindirizza la pagina principale a quella delle iscrizioni: ", "reddit": "",
"Number of videos shown in feed: ": "Numero di video da mostrare nelle iscrizioni: ", "Default captions: ": "Sottotitoli predefiniti: ",
"Sort videos by: ": "Ordinare i video per: ", "Fallback captions: ": "Sottotitoli alternativi: ",
"published": "data di pubblicazione", "Show related videos? ": "Mostra video correlati? ",
"published - reverse": "data di pubblicazione - decrescente", "Show annotations by default? ": "Mostra le annotazioni per impostazione predefinita? ",
"alphabetically": "ordine alfabetico", "Visual preferences": "Preferenze grafiche",
"alphabetically - reverse": "ordine alfabetico - decrescente", "Dark mode: ": "Tema scuro: ",
"channel name": "nome del canale", "Thin mode: ": "Modalità per connessioni lente: ",
"channel name - reverse": "nome del canale - decrescente", "Subscription preferences": "Preferenze iscrizioni",
"Only show latest video from channel: ": "Mostra solo il video più recente del canale: ", "Show annotations by default for subscribed channels? ": "",
"Only show latest unwatched video from channel: ": "Mostra solo il video più recente non guardato del canale: ", "Redirect homepage to feed: ": "Reindirizza la pagina principale a quella delle iscrizioni: ",
"Only show unwatched: ": "Mostra solo i video non guardati: ", "Number of videos shown in feed: ": "Numero di video da mostrare nelle iscrizioni: ",
"Only show notifications (if there are any): ": "Mostra solo le notifiche (se presenti): ", "Sort videos by: ": "Ordinare i video per: ",
"Data preferences": "Preferenze dati", "published": "data di pubblicazione",
"Clear watch history": "Cancella la cronologia dei video guardati", "published - reverse": "data di pubblicazione - decrescente",
"Import/Export data": "Importazione/esportazione dati", "alphabetically": "ordine alfabetico",
"Manage subscriptions": "Gestisci le iscrizioni", "alphabetically - reverse": "ordine alfabetico - decrescente",
"Watch history": "Cronologia dei video", "channel name": "nome del canale",
"Delete account": "Elimina l'account", "channel name - reverse": "nome del canale - decrescente",
"Administrator preferences": "", "Only show latest video from channel: ": "Mostra solo il video più recente del canale: ",
"Default homepage: ": "", "Only show latest unwatched video from channel: ": "Mostra solo il video più recente non guardato del canale: ",
"Feed menu: ": "", "Only show unwatched: ": "Mostra solo i video non guardati: ",
"Top enabled? ": "", "Only show notifications (if there are any): ": "Mostra solo le notifiche (se presenti): ",
"CAPTCHA enabled? ": "", "Data preferences": "Preferenze dati",
"Login enabled? ": "", "Clear watch history": "Cancella la cronologia dei video guardati",
"Registration enabled? ": "", "Import/export data": "Importazione/esportazione dati",
"Report statistics? ": "", "Change password": "",
"Save preferences": "Salva le preferenze", "Manage subscriptions": "Gestisci le iscrizioni",
"Subscription manager": "Gestisci le iscrizioni", "Manage tokens": "",
"`x` subscriptions": "`x` iscrizioni", "Watch history": "Cronologia dei video",
"Import/Export": "Importa/esporta", "Delete account": "Elimina l'account",
"unsubscribe": "disiscriviti", "Administrator preferences": "",
"Subscriptions": "Iscrizioni", "Default homepage: ": "",
"`x` unseen notifications": "`x` notifiche non visualizzate", "Feed menu: ": "",
"search": "Cerca", "Top enabled? ": "",
"Sign out": "Esci", "CAPTCHA enabled? ": "",
"Released under the AGPLv3 by Omar Roth.": "Pubblicato con licenza AGPLv3 da Omar Roth.", "Login enabled? ": "",
"Source available here.": "Codice sorgente.", "Registration enabled? ": "",
"View JavaScript license information.": "Guarda le informazioni di licenza del codice JavaScript.", "Report statistics? ": "",
"View privacy policy.": "", "Save preferences": "Salva le preferenze",
"Trending": "Tendenze", "Subscription manager": "Gestisci le iscrizioni",
"Unlisted": "", "Token manager": "",
"Watch video on Youtube": "Guarda il video su YouTube", "Token": "",
"Genre: ": "Genere: ", "`x` subscriptions": "`x` iscrizioni",
"License: ": "Licenza: ", "`x` tokens": "",
"Family friendly? ": "Per tutti? ", "Import/export": "Importa/esporta",
"Wilson score: ": "Punteggio di Wilson: ", "unsubscribe": "disiscriviti",
"Engagement: ": "Tasso di coinvolgimento: ", "revoke": "",
"Whitelisted regions: ": "Regioni nella lista bianca: ", "Subscriptions": "Iscrizioni",
"Blacklisted regions: ": "Regioni nella lista nera: ", "`x` unseen notifications": "`x` notifiche non visualizzate",
"Shared `x`": "Condiviso `x`", "search": "Cerca",
"Premieres in `x`": "", "Log out": "Esci",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Ciao! Sembra che tu abbia disattivato JavaScript. Clicca qui per visualizzare i commenti. Considera che potrebbe volerci più tempo.", "Released under the AGPLv3 by Omar Roth.": "Pubblicato con licenza AGPLv3 da Omar Roth.",
"View YouTube comments": "Visualizza i commenti da YouTube", "Source available here.": "Codice sorgente.",
"View more comments on Reddit": "Visualizza più commenti su Reddit", "View JavaScript license information.": "Guarda le informazioni di licenza del codice JavaScript.",
"View `x` comments": "Visualizza `x` commenti", "View privacy policy.": "",
"View Reddit comments": "Visualizza i commenti da Reddit", "Trending": "Tendenze",
"Hide replies": "Nascondi le risposte", "Unlisted": "",
"Show replies": "Mostra le risposte", "Watch on YouTube": "Guarda il video su YouTube",
"Incorrect password": "Password sbagliata", "Hide annotations": "",
"Quota exceeded, try again in a few hours": "Limite superato, prova di nuovo fra qualche ora", "Show annotations": "",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Impossibile autenticarsi, controlla che l'autenticazione in due passaggi (Authenticator o SMS) sia attiva.", "Genre: ": "Genere: ",
"Invalid TFA code": "Codice di autenticazione a due fattori non valido", "License: ": "Licenza: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login fallito. L'errore potrebbe essere causato dal fatto che la verifica in due passaggi non è attiva sul tuo account.", "Family friendly? ": "Per tutti? ",
"Invalid answer": "Risposta errata", "Wilson score: ": "Punteggio di Wilson: ",
"Invalid CAPTCHA": "CAPTCHA errato", "Engagement: ": "Tasso di coinvolgimento: ",
"CAPTCHA is a required field": "Il CAPTCHA è un campo obbligatorio", "Whitelisted regions: ": "Regioni nella lista bianca: ",
"User ID is a required field": "L'ID utente è obbligatorio", "Blacklisted regions: ": "Regioni nella lista nera: ",
"Password is a required field": "La password è un campo obbligatorio", "Shared `x`": "Condiviso `x`",
"Invalid username or password": "Nome utente o password errati", "`x` views": "",
"Please sign in using 'Sign in with Google'": "Per favore accedi con \"Entra con Google\"", "Premieres in `x`": "",
"Password cannot be empty": "La password non può essere vuota", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Ciao! Sembra che tu abbia disattivato JavaScript. Clicca qui per visualizzare i commenti. Considera che potrebbe volerci più tempo.",
"Password cannot be longer than 55 characters": "La password non può contenere più di 55 caratteri", "View YouTube comments": "Visualizza i commenti da YouTube",
"Please sign in": "Per favore, entra", "View more comments on Reddit": "Visualizza più commenti su Reddit",
"Invidious Private Feed for `x`": "Feed privato Invidious per `x`", "View `x` comments": "Visualizza `x` commenti",
"channel:`x`": "canale:`x`", "View Reddit comments": "Visualizza i commenti da Reddit",
"Deleted or invalid channel": "Canale cancellato o invalido", "Hide replies": "Nascondi le risposte",
"This channel does not exist.": "Canale inesistente.", "Show replies": "Mostra le risposte",
"Could not get channel info.": "Impossibile ottenere le informazioni del canale.", "Incorrect password": "Password sbagliata",
"Could not fetch comments": "Impossibile recuperare i commenti", "Quota exceeded, try again in a few hours": "Limite superato, prova di nuovo fra qualche ora",
"View `x` replies": "Visualizza `x` risposte", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Impossibile autenticarsi, controlla che l'autenticazione in due passaggi (Authenticator o SMS) sia attiva.",
"`x` ago": "`x` fa", "Invalid TFA code": "Codice di autenticazione a due fattori non valido",
"Load more": "Carica altro", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Login fallito. L'errore potrebbe essere causato dal fatto che la verifica in due passaggi non è attiva sul tuo account.",
"`x` points": "`x` punti", "Wrong answer": "Risposta errata",
"Could not create mix.": "Impossibile creare il mix.", "Erroneous CAPTCHA": "CAPTCHA errato",
"Playlist is empty": "Playlist vuota", "CAPTCHA is a required field": "Il CAPTCHA è un campo obbligatorio",
"Invalid playlist.": "Playlist invalida.", "User ID is a required field": "L'ID utente è obbligatorio",
"Playlist does not exist.": "Playlist inesistente.", "Password is a required field": "La password è un campo obbligatorio",
"Could not pull trending pages.": "Impossibile recuperare le tendenze.", "Wrong username or password": "Nome utente o password errati",
"Hidden field \"challenge\" is a required field": "Il campo nascosto \"challenge\" è obbligatorio", "Please sign in using 'Log in with Google'": "Per favore accedi con \"Entra con Google\"",
"Hidden field \"token\" is a required field": "Il campo nascosto \"token\" è obbligatorio", "Password cannot be empty": "La password non può essere vuota",
"Invalid challenge": "Campo \"challenge\" invalido", "Password cannot be longer than 55 characters": "La password non può contenere più di 55 caratteri",
"Invalid token": "Campo \"token\" invalido", "Please log in": "Per favore, entra",
"Invalid user": "Utente invalido", "Invidious Private Feed for `x`": "Feed privato Invidious per `x`",
"Token is expired, please try again": "Token scaduto, riprova", "channel:`x`": "canale:`x`",
"English": "Inglese", "Deleted or invalid channel": "Canale cancellato o invalido",
"English (auto-generated)": "Inglese (generati automaticamente)", "This channel does not exist.": "Canale inesistente.",
"Afrikaans": "Afrikaans", "Could not get channel info.": "Impossibile ottenere le informazioni del canale.",
"Albanian": "Albanese", "Could not fetch comments": "Impossibile recuperare i commenti",
"Amharic": "Amarico", "View `x` replies": "Visualizza `x` risposte",
"Arabic": "Arabo", "`x` ago": "`x` fa",
"Armenian": "Armeno", "Load more": "Carica altro",
"Azerbaijani": "Azero", "`x` points": "`x` punti",
"Bangla": "Bengalese", "Could not create mix.": "Impossibile creare il mix.",
"Basque": "Basco", "Empty playlist": "Playlist vuota",
"Belarusian": "Biellorusso", "Not a playlist.": "Playlist invalida.",
"Bosnian": "Bosniaco", "Playlist does not exist.": "Playlist inesistente.",
"Bulgarian": "Bulgaro", "Could not pull trending pages.": "Impossibile recuperare le tendenze.",
"Burmese": "Birmano", "Hidden field \"challenge\" is a required field": "Il campo nascosto \"challenge\" è obbligatorio",
"Catalan": "Catalano", "Hidden field \"token\" is a required field": "Il campo nascosto \"token\" è obbligatorio",
"Cebuano": "Sugbuanon", "Erroneous challenge": "Campo \"challenge\" invalido",
"Chinese (Simplified)": "Cinese semplifiato", "Erroneous token": "Campo \"token\" invalido",
"Chinese (Traditional)": "Cinese tradizionale", "No such user": "Utente invalido",
"Corsican": "Corso", "Token is expired, please try again": "Token scaduto, riprova",
"Croatian": "Croato", "English": "Inglese",
"Czech": "Ceco", "English (auto-generated)": "Inglese (generati automaticamente)",
"Danish": "Danese", "Afrikaans": "Afrikaans",
"Dutch": "Olandese", "Albanian": "Albanese",
"Esperanto": "Esperanto", "Amharic": "Amarico",
"Estonian": "Estone", "Arabic": "Arabo",
"Filipino": "Filippino", "Armenian": "Armeno",
"Finnish": "Finlandese", "Azerbaijani": "Azero",
"French": "Francese", "Bangla": "Bengalese",
"Galician": "Galiziano", "Basque": "Basco",
"Georgian": "Georgiano", "Belarusian": "Biellorusso",
"German": "Tedesco", "Bosnian": "Bosniaco",
"Greek": "Greco", "Bulgarian": "Bulgaro",
"Gujarati": "Gujarati", "Burmese": "Birmano",
"Haitian Creole": "Creolo haitiano", "Catalan": "Catalano",
"Hausa": "Lingua hausa", "Cebuano": "Sugbuanon",
"Hawaiian": "Hawaiano", "Chinese (Simplified)": "Cinese semplifiato",
"Hebrew": "Ebreo", "Chinese (Traditional)": "Cinese tradizionale",
"Hindi": "Hindi", "Corsican": "Corso",
"Hmong": "Hmong", "Croatian": "Croato",
"Hungarian": "Ungarese", "Czech": "Ceco",
"Icelandic": "Islandese", "Danish": "Danese",
"Igbo": "Igbo", "Dutch": "Olandese",
"Indonesian": "Indonesiano", "Esperanto": "Esperanto",
"Irish": "Irlandese", "Estonian": "Estone",
"Italian": "Italiano", "Filipino": "Filippino",
"Japanese": "Giapponese", "Finnish": "Finlandese",
"Javanese": "Giavanese", "French": "Francese",
"Kannada": "Kannada", "Galician": "Galiziano",
"Kazakh": "Kazaco", "Georgian": "Georgiano",
"Khmer": "Khmer", "German": "Tedesco",
"Korean": "Coreano", "Greek": "Greco",
"Kurdish": "Curdo", "Gujarati": "Gujarati",
"Kyrgyz": "Kirghize", "Haitian Creole": "Creolo haitiano",
"Lao": "Lao", "Hausa": "Lingua hausa",
"Latin": "Latino", "Hawaiian": "Hawaiano",
"Latvian": "Lettone", "Hebrew": "Ebreo",
"Lithuanian": "Lituano", "Hindi": "Hindi",
"Luxembourgish": "Lussemburghese", "Hmong": "Hmong",
"Macedonian": "Macedone", "Hungarian": "Ungarese",
"Malagasy": "Malgascio", "Icelandic": "Islandese",
"Malay": "Malese", "Igbo": "Igbo",
"Malayalam": "Lingua malayalam", "Indonesian": "Indonesiano",
"Maltese": "Maltese", "Irish": "Irlandese",
"Maori": "Maori", "Italian": "Italiano",
"Marathi": "Marathi", "Japanese": "Giapponese",
"Mongolian": "Mongolo", "Javanese": "Giavanese",
"Nepali": "Nepalese", "Kannada": "Kannada",
"Norwegian": "Norvegese", "Kazakh": "Kazaco",
"Nyanja": "Nyanja", "Khmer": "Khmer",
"Pashto": "Lingua pashtu", "Korean": "Coreano",
"Persian": "Persiano", "Kurdish": "Curdo",
"Polish": "Polacco", "Kyrgyz": "Kirghize",
"Portuguese": "Portoghese", "Lao": "Lao",
"Punjabi": "Punjabi", "Latin": "Latino",
"Romanian": "Rumeno", "Latvian": "Lettone",
"Russian": "Russo", "Lithuanian": "Lituano",
"Samoan": "Samoan", "Luxembourgish": "Lussemburghese",
"Scottish Gaelic": "Gaelico scozzese", "Macedonian": "Macedone",
"Serbian": "Serbo", "Malagasy": "Malgascio",
"Shona": "Shona", "Malay": "Malese",
"Sindhi": "Sindhi", "Malayalam": "Lingua malayalam",
"Sinhala": "Cingalese", "Maltese": "Maltese",
"Slovak": "Slovacco", "Maori": "Maori",
"Slovenian": "Sloveno", "Marathi": "Marathi",
"Somali": "Somalo", "Mongolian": "Mongolo",
"Southern Sotho": "Sotho del Sud", "Nepali": "Nepalese",
"Spanish": "Spagnolo", "Norwegian Bokmål": "Norvegese",
"Spanish (Latin America)": "Spagnolo (America latina)", "Nyanja": "Nyanja",
"Sundanese": "Sudanese", "Pashto": "Lingua pashtu",
"Swahili": "Swahili", "Persian": "Persiano",
"Swedish": "Svedese", "Polish": "Polacco",
"Tajik": "Tajik", "Portuguese": "Portoghese",
"Tamil": "Tamil", "Punjabi": "Punjabi",
"Telugu": "Telugu", "Romanian": "Rumeno",
"Thai": "Thaï", "Russian": "Russo",
"Turkish": "Turco", "Samoan": "Samoan",
"Ukrainian": "Ucraino", "Scottish Gaelic": "Gaelico scozzese",
"Urdu": "Urdu", "Serbian": "Serbo",
"Uzbek": "Uzbeco", "Shona": "Shona",
"Vietnamese": "Vietnamese", "Sindhi": "Sindhi",
"Welsh": "Gallese", "Sinhala": "Cingalese",
"Western Frisian": "Frisone occidentale", "Slovak": "Slovacco",
"Xhosa": "Xhosa", "Slovenian": "Sloveno",
"Yiddish": "Yiddish", "Somali": "Somalo",
"Yoruba": "Yoruba", "Southern Sotho": "Sotho del Sud",
"Zulu": "Zulu", "Spanish": "Spagnolo",
"`x` years": "`x` anni", "Spanish (Latin America)": "Spagnolo (America latina)",
"`x` months": "`x` mesi", "Sundanese": "Sudanese",
"`x` weeks": "`x` settimane", "Swahili": "Swahili",
"`x` days": "`x` giorni", "Swedish": "Svedese",
"`x` hours": "`x` ore", "Tajik": "Tajik",
"`x` minutes": "`x` minuti", "Tamil": "Tamil",
"`x` seconds": "`x` secondi", "Telugu": "Telugu",
"Fallback comments: ": "Commenti alternativi: ", "Thai": "Thaï",
"Popular": "Popolare", "Turkish": "Turco",
"Top": "Top", "Ukrainian": "Ucraino",
"About": "A proposito", "Urdu": "Urdu",
"Rating: ": "Punteggio: ", "Uzbek": "Uzbeco",
"Language: ": "Lingua: ", "Vietnamese": "Vietnamese",
"Default": "Predefinito", "Welsh": "Gallese",
"Music": "Musica", "Western Frisian": "Frisone occidentale",
"Gaming": "Videogiochi", "Xhosa": "Xhosa",
"News": "Notizie", "Yiddish": "Yiddish",
"Movies": "Film", "Yoruba": "Yoruba",
"Download": "Scarica", "Zulu": "Zulu",
"Download as: ": "Scarica come: ", "`x` years": "`x` anni",
"%A %B %-d, %Y": "%A %-d %B %Y", "`x` months": "`x` mesi",
"(edited)": "(modificato)", "`x` weeks": "`x` settimane",
"Youtube permalink of the comment": "Link permanente al commento di YouTube", "`x` days": "`x` giorni",
"`x` marked it with a ❤": "`x` l'ha contrassegnato con un ❤", "`x` hours": "`x` ore",
"Audio mode": "Modalità audio", "`x` minutes": "`x` minuti",
"Video mode": "Modalità video", "`x` seconds": "`x` secondi",
"Videos": "", "Fallback comments: ": "Commenti alternativi: ",
"Playlists": "", "Popular": "Popolare",
"Current version: ": "" "Top": "Top",
} "About": "A proposito",
"Rating: ": "Punteggio: ",
"Language: ": "Lingua: ",
"View as playlist": "",
"Default": "Predefinito",
"Music": "Musica",
"Gaming": "Videogiochi",
"News": "Notizie",
"Movies": "Film",
"Download": "Scarica",
"Download as: ": "Scarica come: ",
"%A %B %-d, %Y": "%A %-d %B %Y",
"(edited)": "(modificato)",
"YouTube comment permalink": "Link permanente al commento di YouTube",
"`x` marked it with a ❤": "`x` l'ha contrassegnato con un ❤",
"Audio mode": "Modalità audio",
"Video mode": "Modalità video",
"Videos": "",
"Playlists": "",
"Current version: ": ""
}

View File

@ -1,295 +1,315 @@
{ {
"`x` subscribers": "`x` abonnenter", "`x` subscribers": "`x` abonnenter",
"`x` videos": "`x` videoer", "`x` videos": "`x` videoer",
"LIVE": "SANNTIDSVISNING", "LIVE": "SANNTIDSVISNING",
"Shared `x` ago": "Delt for `x` siden", "Shared `x` ago": "Delt for `x` siden",
"Unsubscribe": "Opphev abonnement", "Unsubscribe": "Opphev abonnement",
"Subscribe": "Abonner", "Subscribe": "Abonner",
"Login to subscribe to `x`": "Logg inn for å abonnere på `x`", "View channel on YouTube": "Vis kanal på YouTube",
"View channel on YouTube": "Vis kanal på YouTube", "View playlist on YouTube": "",
"newest": "nyeste", "newest": "nyeste",
"oldest": "eldste", "oldest": "eldste",
"popular": "populært", "popular": "populært",
"last": "siste", "last": "siste",
"Next page": "Neste side", "Next page": "Neste side",
"Previous page": "Forrige side", "Previous page": "Forrige side",
"Clear watch history?": "Tøm visningshistorikk?", "Clear watch history?": "Tøm visningshistorikk?",
"Yes": "Ja", "New password": "Nytt passord",
"No": "Nei", "New passwords must match": "Nye passordfelter må stemme overens",
"Import and Export Data": "Importer- og eksporter data", "Cannot change password for Google accounts": "Kan ikke endre passord for Google-kontoer",
"Import": "Importer", "Authorize token?": "Identitetsbekreft symbol?",
"Import Invidious data": "Importer Invidious-data", "Authorize token for `x`?": "Identitetsbekreft symbol for `x`?",
"Import YouTube subscriptions": "Importer YouTube-abonnenter", "Yes": "Ja",
"Import FreeTube subscriptions (.db)": "Importer FreeTube-abonnenter (.db)", "No": "Nei",
"Import NewPipe subscriptions (.json)": "Importer NewPipe-abonnenter (.json)", "Import and Export Data": "Importer- og eksporter data",
"Import NewPipe data (.zip)": "Importer NewPipe-data (.zip)", "Import": "Importer",
"Export": "Eksporter", "Import Invidious data": "Importer Invidious-data",
"Export subscriptions as OPML": "Eksporter abonnenter som OPML", "Import YouTube subscriptions": "Importer YouTube-abonnenter",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksporter abonnenter som OPML (for NewPipe og FreeTube)", "Import FreeTube subscriptions (.db)": "Importer FreeTube-abonnenter (.db)",
"Export data as JSON": "Eksporter data som JSON", "Import NewPipe subscriptions (.json)": "Importer NewPipe-abonnenter (.json)",
"Delete account?": "Slett konto?", "Import NewPipe data (.zip)": "Importer NewPipe-data (.zip)",
"History": "Historikk", "Export": "Eksporter",
"An alternative front-end to YouTube": "En alternativ grenseflate for YouTube", "Export subscriptions as OPML": "Eksporter abonnenter som OPML",
"JavaScript license information": "JavaScript-lisensinformasjon", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksporter abonnenter som OPML (for NewPipe og FreeTube)",
"source": "kilde", "Export data as JSON": "Eksporter data som JSON",
"Login": "Logg inn", "Delete account?": "Slett konto?",
"Login/Register": "Logg inn/registrer", "History": "Historikk",
"Login to Google": "Logg inn med Google", "An alternative front-end to YouTube": "En alternativ grenseflate for YouTube",
"User ID:": "Bruker-ID:", "JavaScript license information": "JavaScript-lisensinformasjon",
"Password:": "Passord:", "source": "kilde",
"Time (h:mm:ss):": "Tid (h:mm:ss):", "Log in": "Logg inn",
"Text CAPTCHA": "Tekst-CAPTCHA", "Log in/register": "Logg inn/registrer",
"Image CAPTCHA": "Bilde-CAPTCHA", "Log in with Google": "Logg inn med Google",
"Sign In": "Innlogging", "User ID": "Bruker-ID",
"Register": "Registrer", "Password": "Passord",
"Email:": "E-post:", "Time (h:mm:ss):": "Tid (h:mm:ss):",
"Google verification code:": "Google-bekreftelseskode:", "Text CAPTCHA": "Tekst-CAPTCHA",
"Preferences": "Innstillinger", "Image CAPTCHA": "Bilde-CAPTCHA",
"Player preferences": "Avspillerinnstillinger", "Sign In": "Innlogging",
"Always loop: ": "Alltid gjenta: ", "Register": "Registrer",
"Autoplay: ": "Autoavspilling: ", "E-mail": "E-post",
"Autoplay next video: ": "Autospill neste video: ", "Google verification code": "Google-bekreftelseskode",
"Listen by default: ": "Lytt som forvalg: ", "Preferences": "Innstillinger",
"Proxy videos? ": "", "Player preferences": "Avspillerinnstillinger",
"Default speed: ": "Forvalgt hastighet: ", "Always loop: ": "Alltid gjenta: ",
"Preferred video quality: ": "Foretrukket videokvalitet: ", "Autoplay: ": "Autoavspilling: ",
"Player volume: ": "Avspillerlydstyrke: ", "Play next by default: ": "Spill neste som forvalg: ",
"Default comments: ": "Forvalgte kommentarer: ", "Autoplay next video: ": "Autospill neste video: ",
"Default captions: ": "Forvalgte undertitler: ", "Listen by default: ": "Lytt som forvalg: ",
"Fallback captions: ": "Tilbakefallsundertitler: ", "Proxy videos? ": "Mellomtjen videoer? ",
"Show related videos? ": "Vis relaterte videoer? ", "Default speed: ": "Forvalgt hastighet: ",
"Visual preferences": "Visuelle innstillinger", "Preferred video quality: ": "Foretrukket videokvalitet: ",
"Dark mode: ": "Mørk drakt: ", "Player volume: ": "Avspillerlydstyrke: ",
"Thin mode: ": "Tynt modus: ", "Default comments: ": "Forvalgte kommentarer: ",
"Subscription preferences": "Abonnementsinnstillinger", "youtube": "YouTube",
"Redirect homepage to feed: ": "Videresend hjemmeside til flyt: ", "reddit": "Reddit",
"Number of videos shown in feed: ": "Antall videoer å vise i flyt: ", "Default captions: ": "Forvalgte undertitler: ",
"Sort videos by: ": "Sorter videoer etter: ", "Fallback captions: ": "Tilbakefallsundertitler: ",
"published": "publisert", "Show related videos? ": "Vis relaterte videoer? ",
"published - reverse": "publisert - motsatt", "Show annotations by default? ": "Vis merknader som forvalg? ",
"alphabetically": "alfabetisk", "Visual preferences": "Visuelle innstillinger",
"alphabetically - reverse": "alfabetisk - motsatt", "Dark mode: ": "Mørk drakt: ",
"channel name": "kanalnavn", "Thin mode: ": "Tynt modus: ",
"channel name - reverse": "kanalnavn - motsatt", "Subscription preferences": "Abonnementsinnstillinger",
"Only show latest video from channel: ": "Kun vis siste video fra kanal: ", "Show annotations by default for subscribed channels? ": "Vis merknader som forvalg for kanaler det abonneres på? ",
"Only show latest unwatched video from channel: ": "Kun vis siste usette video fra kanal: ", "Redirect homepage to feed: ": "Videresend hjemmeside til flyt: ",
"Only show unwatched: ": "Kun vis usette: ", "Number of videos shown in feed: ": "Antall videoer å vise i flyt: ",
"Only show notifications (if there are any): ": "Kun vis merknader (hvis det er noen): ", "Sort videos by: ": "Sorter videoer etter: ",
"Data preferences": "Datainnstillinger", "published": "publisert",
"Clear watch history": "Tøm visningshistorikk", "published - reverse": "publisert - motsatt",
"Import/Export data": "Importer/eksporter data", "alphabetically": "alfabetisk",
"Manage subscriptions": "Behandle abonnementer", "alphabetically - reverse": "alfabetisk - motsatt",
"Watch history": "Visningshistorikk", "channel name": "kanalnavn",
"Delete account": "Slett konto", "channel name - reverse": "kanalnavn - motsatt",
"Administrator preferences": "Administratorinnstillinger", "Only show latest video from channel: ": "Kun vis siste video fra kanal: ",
"Default homepage: ": "Forvalgt hjemmeside: ", "Only show latest unwatched video from channel: ": "Kun vis siste usette video fra kanal: ",
"Feed menu: ": "Flyt-meny: ", "Only show unwatched: ": "Kun vis usette: ",
"Top enabled? ": "Topp påskrudd? ", "Only show notifications (if there are any): ": "Kun vis merknader (hvis det er noen): ",
"CAPTCHA enabled? ": "CAPTCHA påskrudd? ", "Data preferences": "Datainnstillinger",
"Login enabled? ": "Innlogging påskrudd? ", "Clear watch history": "Tøm visningshistorikk",
"Registration enabled? ": "Registrering påskrudd? ", "Import/export data": "Importer/eksporter data",
"Report statistics? ": "Innrapporter statistikk? ", "Change password": "Endre passord",
"Save preferences": "Lagre innstillinger", "Manage subscriptions": "Behandle abonnementer",
"Subscription manager": "Abonnementsbehandler", "Manage tokens": "Behandle symboler",
"`x` subscriptions": "`x` abonnementer", "Watch history": "Visningshistorikk",
"Import/Export": "Importer/eksporter", "Delete account": "Slett konto",
"unsubscribe": "opphev abonnement", "Administrator preferences": "Administratorinnstillinger",
"Subscriptions": "Abonnement", "Default homepage: ": "Forvalgt hjemmeside: ",
"`x` unseen notifications": "`x` usette merknader", "Feed menu: ": "Flyt-meny: ",
"search": "søk", "Top enabled? ": "Topp påskrudd? ",
"Sign out": "Logg ut", "CAPTCHA enabled? ": "CAPTCHA påskrudd? ",
"Released under the AGPLv3 by Omar Roth.": "Utgitt med AGPLv3+lisens av Omar Roth.", "Login enabled? ": "Innlogging påskrudd? ",
"Source available here.": "Kildekode tilgjengelig her.", "Registration enabled? ": "Registrering påskrudd? ",
"View JavaScript license information.": "Vis JavaScript-lisensinfo.", "Report statistics? ": "Innrapporter statistikk? ",
"View privacy policy.": "", "Save preferences": "Lagre innstillinger",
"Trending": "Trendsettende", "Subscription manager": "Abonnementsbehandler",
"Unlisted": "", "Token manager": "Symbolbehandler",
"Watch video on Youtube": "Vis video på YouTube", "Token": "Symbol",
"Genre: ": "Sjanger: ", "`x` subscriptions": "`x` abonnementer",
"License: ": "Lisens: ", "`x` tokens": "`x` symboler",
"Family friendly? ": "Familievennlig? ", "Import/export": "Importer/eksporter",
"Wilson score: ": "Wilson-poengsum: ", "unsubscribe": "opphev abonnement",
"Engagement: ": "Engasjement: ", "revoke": "tilbakekall",
"Whitelisted regions: ": "Hvitlistede regioner: ", "Subscriptions": "Abonnement",
"Blacklisted regions: ": "Svartelistede regioner: ", "`x` unseen notifications": "`x` usette merknader",
"Shared `x`": "Delt `x`", "search": "søk",
"Premieres in `x`": "", "Log out": "Logg ut",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hei. Det ser ut til at du har JavaScript avslått. Klikk her for å vise kommentarer, ha i minnet at innlasting tar lengre tid.", "Released under the AGPLv3 by Omar Roth.": "Utgitt med AGPLv3+lisens av Omar Roth.",
"View YouTube comments": "Vis YouTube-kommentarer", "Source available here.": "Kildekode tilgjengelig her.",
"View more comments on Reddit": "Vis flere kommenterer på Reddit", "View JavaScript license information.": "Vis JavaScript-lisensinfo.",
"View `x` comments": "Vis `x` kommentarer", "View privacy policy.": "Vis personvernspraksis.",
"View Reddit comments": "Vis Reddit-kommentarer", "Trending": "Trendsettende",
"Hide replies": "Skjul svar", "Unlisted": "Ulistet",
"Show replies": "Vis svar", "Watch on YouTube": "Vis video på YouTube",
"Incorrect password": "Feil passord", "Hide annotations": "Skjul merknader",
"Quota exceeded, try again in a few hours": "Kvote overskredet, prøv igjen om et par timer", "Show annotations": "Vis merknader",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Kunne ikke logge inn, forsikre deg om at tofaktor-identitetsbekreftelse (Authenticator eller SMS) er skrudd på.", "Genre: ": "Sjanger: ",
"Invalid TFA code": "Ugyldig tofaktorkode", "License: ": "Lisens: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Innlogging mislyktes. Dette kan være fordi tofaktor-identitetsbekreftelse er skrudd av på kontoen din.", "Family friendly? ": "Familievennlig? ",
"Invalid answer": "Ugyldig svar", "Wilson score: ": "Wilson-poengsum: ",
"Invalid CAPTCHA": "Ugyldig CAPTCHA", "Engagement: ": "Engasjement: ",
"CAPTCHA is a required field": "CAPTCHA er et påkrevd felt", "Whitelisted regions: ": "Hvitlistede regioner: ",
"User ID is a required field": "Bruker-ID er et påkrevd felt", "Blacklisted regions: ": "Svartelistede regioner: ",
"Password is a required field": "Passord er et påkrevd felt", "Shared `x`": "Delt `x`",
"Invalid username or password": "Ugyldig brukernavn eller passord", "`x` views": "`x` visninger",
"Please sign in using 'Sign in with Google'": "Logg inn ved bruk av \"Google-innlogging\"", "Premieres in `x`": "Premiere om `x`",
"Password cannot be empty": "Passordet kan ikke være tomt", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hei. Det ser ut til at du har JavaScript avslått. Klikk her for å vise kommentarer, ha i minnet at innlasting tar lengre tid.",
"Password cannot be longer than 55 characters": "Passordet kan ikke være lengre enn 55 tegn", "View YouTube comments": "Vis YouTube-kommentarer",
"Please sign in": "Logg inn", "View more comments on Reddit": "Vis flere kommenterer på Reddit",
"Invidious Private Feed for `x`": "Ugyldig privat flyt for `x`", "View `x` comments": "Vis `x` kommentarer",
"channel:`x`": "kanal `x`", "View Reddit comments": "Vis Reddit-kommentarer",
"Deleted or invalid channel": "Slettet eller ugyldig kanal", "Hide replies": "Skjul svar",
"This channel does not exist.": "Denne kanalen finnes ikke.", "Show replies": "Vis svar",
"Could not get channel info.": "Kunne ikke innhente kanalinfo.", "Incorrect password": "Feil passord",
"Could not fetch comments": "Kunne ikke hente kommentarer", "Quota exceeded, try again in a few hours": "Kvote overskredet, prøv igjen om et par timer",
"View `x` replies": "Vis `x` svar", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Kunne ikke logge inn, forsikre deg om at tofaktor-identitetsbekreftelse (Authenticator eller SMS) er skrudd på.",
"`x` ago": "`x` siden", "Invalid TFA code": "Ugyldig tofaktorkode",
"Load more": "Last inn flere", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Innlogging mislyktes. Dette kan være fordi tofaktor-identitetsbekreftelse er skrudd av på kontoen din.",
"`x` points": "`x` poeng", "Wrong answer": "Ugyldig svar",
"Could not create mix.": "Kunne ikke opprette miks.", "Erroneous CAPTCHA": "Ugyldig CAPTCHA",
"Playlist is empty": "Spillelisten er tom", "CAPTCHA is a required field": "CAPTCHA er et påkrevd felt",
"Invalid playlist.": "Ugyldig spilleliste.", "User ID is a required field": "Bruker-ID er et påkrevd felt",
"Playlist does not exist.": "Spillelisten finnes ikke.", "Password is a required field": "Passord er et påkrevd felt",
"Could not pull trending pages.": "Kunne ikke hente trendsettende sider.", "Wrong username or password": "Ugyldig brukernavn eller passord",
"Hidden field \"challenge\" is a required field": "Skjult felt \"utfordring\" er et påkrevd felt", "Please sign in using 'Log in with Google'": "Logg inn ved bruk av \"Google-innlogging\"",
"Hidden field \"token\" is a required field": "Skjult felt \"symbol\" er et påkrevd felt", "Password cannot be empty": "Passordet kan ikke være tomt",
"Invalid challenge": "Ugyldig utfordring", "Password cannot be longer than 55 characters": "Passordet kan ikke være lengre enn 55 tegn",
"Invalid token": "Ugyldig symbol", "Please log in": "Logg inn",
"Invalid user": "Ugyldig bruker", "Invidious Private Feed for `x`": "Ugyldig privat flyt for `x`",
"Token is expired, please try again": "Symbol utløpt, prøv igjen", "channel:`x`": "kanal `x`",
"English": "Engelsk", "Deleted or invalid channel": "Slettet eller ugyldig kanal",
"English (auto-generated)": "Engelsk (auto-generert)", "This channel does not exist.": "Denne kanalen finnes ikke.",
"Afrikaans": "", "Could not get channel info.": "Kunne ikke innhente kanalinfo.",
"Albanian": "Albansk", "Could not fetch comments": "Kunne ikke hente kommentarer",
"Amharic": "", "View `x` replies": "Vis `x` svar",
"Arabic": "Arabisk", "`x` ago": "`x` siden",
"Armenian": "Armensk", "Load more": "Last inn flere",
"Azerbaijani": "", "`x` points": "`x` poeng",
"Bangla": "", "Could not create mix.": "Kunne ikke opprette miks.",
"Basque": "", "Empty playlist": "Spillelisten er tom",
"Belarusian": "Hviterussisk", "Not a playlist.": "Ugyldig spilleliste.",
"Bosnian": "Bosnisk", "Playlist does not exist.": "Spillelisten finnes ikke.",
"Bulgarian": "Bulgarsk", "Could not pull trending pages.": "Kunne ikke hente trendsettende sider.",
"Burmese": "Burmesisk", "Hidden field \"challenge\" is a required field": "Skjult felt \"utfordring\" er et påkrevd felt",
"Catalan": "Katalansk", "Hidden field \"token\" is a required field": "Skjult felt \"symbol\" er et påkrevd felt",
"Cebuano": "", "Erroneous challenge": "Ugyldig utfordring",
"Chinese (Simplified)": "", "Erroneous token": "Ugyldig symbol",
"Chinese (Traditional)": "", "No such user": "Ugyldig bruker",
"Corsican": "", "Token is expired, please try again": "Symbol utløpt, prøv igjen",
"Croatian": "", "English": "Engelsk",
"Czech": "Tsjekkisk", "English (auto-generated)": "Engelsk (auto-generert)",
"Danish": "Dansk", "Afrikaans": "",
"Dutch": "", "Albanian": "Albansk",
"Esperanto": "Esperanto", "Amharic": "",
"Estonian": "", "Arabic": "Arabisk",
"Filipino": "", "Armenian": "Armensk",
"Finnish": "Finsk", "Azerbaijani": "",
"French": "Fransk", "Bangla": "",
"Galician": "", "Basque": "",
"Georgian": "", "Belarusian": "Hviterussisk",
"German": "", "Bosnian": "Bosnisk",
"Greek": "", "Bulgarian": "Bulgarsk",
"Gujarati": "", "Burmese": "Burmesisk",
"Haitian Creole": "", "Catalan": "Katalansk",
"Hausa": "", "Cebuano": "",
"Hawaiian": "", "Chinese (Simplified)": "",
"Hebrew": "", "Chinese (Traditional)": "",
"Hindi": "", "Corsican": "",
"Hmong": "", "Croatian": "",
"Hungarian": "Ungarsk", "Czech": "Tsjekkisk",
"Icelandic": "Islandsk", "Danish": "Dansk",
"Igbo": "", "Dutch": "",
"Indonesian": "Indonesisk", "Esperanto": "Esperanto",
"Irish": "Irsk", "Estonian": "",
"Italian": "Italiensk", "Filipino": "",
"Japanese": "Japansk", "Finnish": "Finsk",
"Javanese": "", "French": "Fransk",
"Kannada": "", "Galician": "",
"Kazakh": "", "Georgian": "",
"Khmer": "", "German": "",
"Korean": "", "Greek": "",
"Kurdish": "", "Gujarati": "",
"Kyrgyz": "", "Haitian Creole": "",
"Lao": "", "Hausa": "",
"Latin": "", "Hawaiian": "",
"Latvian": "", "Hebrew": "",
"Lithuanian": "", "Hindi": "",
"Luxembourgish": "", "Hmong": "",
"Macedonian": "", "Hungarian": "Ungarsk",
"Malagasy": "", "Icelandic": "Islandsk",
"Malay": "", "Igbo": "",
"Malayalam": "", "Indonesian": "Indonesisk",
"Maltese": "", "Irish": "Irsk",
"Maori": "", "Italian": "Italiensk",
"Marathi": "", "Japanese": "Japansk",
"Mongolian": "", "Javanese": "",
"Nepali": "", "Kannada": "",
"Norwegian": "Norsk bokmål", "Kazakh": "",
"Nyanja": "", "Khmer": "",
"Pashto": "", "Korean": "",
"Persian": "", "Kurdish": "",
"Polish": "", "Kyrgyz": "",
"Portuguese": "", "Lao": "",
"Punjabi": "", "Latin": "",
"Romanian": "", "Latvian": "",
"Russian": "Russisk", "Lithuanian": "",
"Samoan": "", "Luxembourgish": "",
"Scottish Gaelic": "", "Macedonian": "",
"Serbian": "Serbisk", "Malagasy": "",
"Shona": "", "Malay": "",
"Sindhi": "", "Malayalam": "",
"Sinhala": "", "Maltese": "",
"Slovak": "Slovakisk", "Maori": "",
"Slovenian": "Slovensk", "Marathi": "",
"Somali": "Somali", "Mongolian": "",
"Southern Sotho": "", "Nepali": "",
"Spanish": "Spansk", "Norwegian Bokmål": "Norsk bokmål",
"Spanish (Latin America)": "", "Nyanja": "",
"Sundanese": "", "Pashto": "",
"Swahili": "", "Persian": "",
"Swedish": "Svensk", "Polish": "",
"Tajik": "", "Portuguese": "",
"Tamil": "", "Punjabi": "",
"Telugu": "", "Romanian": "",
"Thai": "", "Russian": "Russisk",
"Turkish": "Tyrkisk", "Samoan": "",
"Ukrainian": "Ukrainsk", "Scottish Gaelic": "",
"Urdu": "", "Serbian": "Serbisk",
"Uzbek": "", "Shona": "",
"Vietnamese": "Vietnamesisk", "Sindhi": "",
"Welsh": "", "Sinhala": "",
"Western Frisian": "", "Slovak": "Slovakisk",
"Xhosa": "", "Slovenian": "Slovensk",
"Yiddish": "", "Somali": "Somali",
"Yoruba": "", "Southern Sotho": "",
"Zulu": "", "Spanish": "Spansk",
"`x` years": "`x` år", "Spanish (Latin America)": "",
"`x` months": "`x` måneder", "Sundanese": "",
"`x` weeks": "`x` uker", "Swahili": "",
"`x` days": "`x` dager", "Swedish": "Svensk",
"`x` hours": "`x` timer", "Tajik": "",
"`x` minutes": "`x` minutter", "Tamil": "",
"`x` seconds": "`x` sekunder", "Telugu": "",
"Fallback comments: ": "Tilbakefallskommentarer: ", "Thai": "",
"Popular": "Pupulært", "Turkish": "Tyrkisk",
"Top": "Topp", "Ukrainian": "Ukrainsk",
"About": "Om", "Urdu": "",
"Rating: ": "Vurdering: ", "Uzbek": "",
"Language: ": "Språk: ", "Vietnamese": "Vietnamesisk",
"Default": "Forvalg", "Welsh": "",
"Music": "Musikk", "Western Frisian": "",
"Gaming": "Spill", "Xhosa": "",
"News": "Nyheter", "Yiddish": "",
"Movies": "Filmer", "Yoruba": "",
"Download": "Last ned", "Zulu": "",
"Download as: ": "Last ned som: ", "`x` years": "`x` år",
"%A %B %-d, %Y": "", "`x` months": "`x` måneder",
"(edited)": "(redigert)", "`x` weeks": "`x` uker",
"Youtube permalink of the comment": "Permanent YouTube-lenke til innholdet", "`x` days": "`x` dager",
"`x` marked it with a ❤": "`x` levnet et ❤", "`x` hours": "`x` timer",
"Audio mode": "Lydmodus", "`x` minutes": "`x` minutter",
"Video mode": "Video-modus", "`x` seconds": "`x` sekunder",
"Videos": "Videoer", "Fallback comments: ": "Tilbakefallskommentarer: ",
"Playlists": "Spillelister", "Popular": "Pupulært",
"Current version: ": "Nåværende versjon: " "Top": "Topp",
} "About": "Om",
"Rating: ": "Vurdering: ",
"Language: ": "Språk: ",
"View as playlist": "Vis som spilleliste",
"Default": "Forvalg",
"Music": "Musikk",
"Gaming": "Spill",
"News": "Nyheter",
"Movies": "Filmer",
"Download": "Last ned",
"Download as: ": "Last ned som: ",
"%A %B %-d, %Y": "",
"(edited)": "(redigert)",
"YouTube comment permalink": "Permanent YouTube-lenke til innholdet",
"`x` marked it with a ❤": "`x` levnet et ❤",
"Audio mode": "Lydmodus",
"Video mode": "Video-modus",
"Videos": "Videoer",
"Playlists": "Spillelister",
"Current version: ": "Nåværende versjon: "
}

View File

@ -1,295 +1,315 @@
{ {
"`x` subscribers": "`x` abonnees", "`x` subscribers": "`x` abonnees",
"`x` videos": "`x` videos", "`x` videos": "`x` video's",
"LIVE": "LIVE", "LIVE": "LIVE",
"Shared `x` ago": "Gedeeld `x` geleden", "Shared `x` ago": "Gedeeld: `x` geleden",
"Unsubscribe": "Abonnement opzeggen", "Unsubscribe": "Deabonneren",
"Subscribe": "Abonneren", "Subscribe": "Abonneren",
"Login to subscribe to `x`": "Log in om te abonneren op `x`", "View channel on YouTube": "Bekijk kanaal op YouTube",
"View channel on YouTube": "Bekijk kanaal op Youtube", "View playlist on YouTube": "Bekijk afspeellijst op YouTube",
"newest": "nieuwste", "newest": "nieuwste",
"oldest": "oudste", "oldest": "oudste",
"popular": "populair", "popular": "populair",
"last": "", "last": "laatste",
"Next page": "Volgende pagina", "Next page": "Volgende pagina",
"Previous page": "Vorige pagina", "Previous page": "Vorige pagina",
"Clear watch history?": "Kijk geschiedenis wissen?", "Clear watch history?": "Wil je de kijkgeschiedenis wissen?",
"Yes": "Ja", "New password": "Nieuw wachtwoord",
"No": "Nee", "New passwords must match": "De nieuwe wachtwoorden moeten overeenkomen",
"Import and Export Data": "Importeer en Exporteer Gegevens", "Cannot change password for Google accounts": "Kan het wachtwoord van Google-accounts niet wijzigen",
"Import": "Importeren", "Authorize token?": "Wil je de toegangssleutel machtigen?",
"Import Invidious data": "Importeer Invidious gegevens", "Authorize token for `x`?": "Wil je de toegangssleutel machtigen voor `x`?",
"Import YouTube subscriptions": "Importeer Youtube abonnees", "Yes": "Ja",
"Import FreeTube subscriptions (.db)": "Importeer FreeTube abonnees (.db)", "No": "Nee",
"Import NewPipe subscriptions (.json)": "Importeer NewPipe abonnees (.json)", "Import and Export Data": "Gegevens im- en exporteren",
"Import NewPipe data (.zip)": "Importeer NewPipe gegevens (.zip)", "Import": "Importeren",
"Export": "Exporteren", "Import Invidious data": "Invidious-gegevens importeren",
"Export subscriptions as OPML": "Exporteer abonnees als OPML", "Import YouTube subscriptions": "YouTube-abonnementen importeren",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporteer abonnees als OPML (voor NewPipe & FreeTube)", "Import FreeTube subscriptions (.db)": "FreeTube-abonnementen importeren (.db)",
"Export data as JSON": "Exporteer gegevens als JSON", "Import NewPipe subscriptions (.json)": "NewPipe-abonnementen importeren (.json)",
"Delete account?": "Verwijder account?", "Import NewPipe data (.zip)": "NewPipe-gegevens importeren (.zip)",
"History": "Geschiedenis", "Export": "Exporteren",
"An alternative front-end to YouTube": "Een alternatieve front-end voor YouTube", "Export subscriptions as OPML": "Abonnementen exporteren als OPML",
"JavaScript license information": "JavaScript licentie informatie", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnementen exporteren als OPML (voor NewPipe en FreeTube)",
"source": "bron", "Export data as JSON": "Gegevens exporteren als JSON",
"Login": "Inloggen", "Delete account?": "Wil je je account verwijderen?",
"Login/Register": "Inloggen/Registreren", "History": "Geschiedenis",
"Login to Google": "Inloggen op Google", "An alternative front-end to YouTube": "Een alternatief front-end voor YouTube",
"User ID:": "Gebruiker ID:", "JavaScript license information": "JavaScript-licentieinformatie",
"Password:": "Wachtwoord:", "source": "bron",
"Time (h:mm:ss):": "Tijd (h:mm:ss):", "Log in": "Inloggen",
"Text CAPTCHA": "Tekst CAPTCHA", "Log in/register": "Inloggen/Registreren",
"Image CAPTCHA": "Afbeelding CAPTCHA", "Log in with Google": "Inloggen met Google",
"Sign In": "Aanmelden", "User ID": "Gebruikers-id",
"Register": "Registreren", "Password": "Wachtwoord",
"Email:": "Email:", "Time (h:mm:ss):": "Tijd (h:mm:ss):",
"Google verification code:": "Google verificatie code:", "Text CAPTCHA": "Tekst-CAPTCHA",
"Preferences": "Voorkeuren", "Image CAPTCHA": "Afbeelding-CAPTCHA",
"Player preferences": "Afspeler voorkeuren", "Sign In": "Inloggen",
"Always loop: ": "Altijd herhalen: ", "Register": "Registreren",
"Autoplay: ": "Automatisch afspelen: ", "E-mail": "E-mailadres",
"Autoplay next video: ": "Automatisch volgende video afspelen: ", "Google verification code": "Google-verificatiecode",
"Listen by default: ": "Standaard luisteren: ", "Preferences": "Instellingen",
"Proxy videos? ": "", "Player preferences": "Spelerinstellingen",
"Default speed: ": "Standaard snelheid: ", "Always loop: ": "Altijd herhalen: ",
"Preferred video quality: ": "Video kwaliteit voorkeur: ", "Autoplay: ": "Automatisch afspelen: ",
"Player volume: ": "Afspeler volume: ", "Play next by default: ": "Standaard volgende video afspelen: ",
"Default comments: ": "Standaard reacties: ", "Autoplay next video: ": "Volgende video automatisch afspelen: ",
"Default captions: ": "Standaard ondertitels: ", "Listen by default: ": "Standaard luisteren: ",
"Fallback captions: ": "Alternatieve ondertitels: ", "Proxy videos? ": "Video's afspelen via proxy? ",
"Show related videos? ": "Laat gerelateerde videos zien? ", "Default speed: ": "Standaard afspeelsnelheid: ",
"Visual preferences": "Visuele voorkeuren", "Preferred video quality: ": "Voorkeurskwaliteit: ",
"Dark mode: ": "Donkere modus: ", "Player volume: ": "Spelervolume: ",
"Thin mode: ": "Smalle modus: ", "Default comments: ": "Reacties tonen van: ",
"Subscription preferences": "Abonnement voorkeuren", "youtube": "YouTube",
"Redirect homepage to feed: ": "Startpagina omleiden naar feed: ", "reddit": "Reddit",
"Number of videos shown in feed: ": "Aantal videos te zien in feed: ", "Default captions: ": "Standaard ondertiteling: ",
"Sort videos by: ": "Sorteer videos op: ", "Fallback captions: ": "Alternatieve ondertiteling: ",
"published": "gepubliceerd", "Show related videos? ": "Gerelateerde video's tonen? ",
"published - reverse": "gepubliceerd - omgekeerd", "Show annotations by default? ": "Standaard annotaties tonen? ",
"alphabetically": "alfabetische volgorde", "Visual preferences": "Visuele instellingen",
"alphabetically - reverse": "alfabetisch - omgekeerd", "Dark mode: ": "Donkere modus: ",
"channel name": "kanaal naam", "Thin mode: ": "Smalle modus: ",
"channel name - reverse": "kanaal naam - omgekeerd", "Subscription preferences": "Abonnementsinstellingen",
"Only show latest video from channel: ": "Laat alleen laatste video van kanaal zien: ", "Show annotations by default for subscribed channels? ": "Standaard annotaties tonen voor geabonneerde kanalen? ",
"Only show latest unwatched video from channel: ": "Laat alleen de laatste onbekeken video zien van kanaal: ", "Redirect homepage to feed: ": "Startpagina omleiden naar feed: ",
"Only show unwatched: ": "Laat alleen onbekeken videos zien: ", "Number of videos shown in feed: ": "Aantal te tonen video's in feed: ",
"Only show notifications (if there are any): ": "Laat alleen notificaties zien (als die er zijn): ", "Sort videos by: ": "Video's sorteren op: ",
"Data preferences": "Gegevens voorkeuren", "published": "publicatiedatum",
"Clear watch history": "Kijkgeschiedenis wissen", "published - reverse": "publicatiedatum - omgekeerd",
"Import/Export data": "Importeer/Exporteer gegevens", "alphabetically": "alfabetische volgorde",
"Manage subscriptions": "Abonnees beheren", "alphabetically - reverse": "alfabetische volgorde - omgekeerd",
"Watch history": "Kijkgeschiedenis", "channel name": "kanaalnaam",
"Delete account": "Account verwijderen", "channel name - reverse": "kanaalnaam - omgekeerd",
"Administrator preferences": "", "Only show latest video from channel: ": "Alleen nieuwste video van kanaal tonen: ",
"Default homepage: ": "", "Only show latest unwatched video from channel: ": "Alleen nieuwste niet-bekeken video van kanaal tonen: ",
"Feed menu: ": "", "Only show unwatched: ": "Alleen niet-bekeken videos tonen: ",
"Top enabled? ": "", "Only show notifications (if there are any): ": "Alleen meldingen tonen (als die er zijn): ",
"CAPTCHA enabled? ": "", "Data preferences": "Gegevensinstellingen",
"Login enabled? ": "", "Clear watch history": "Kijkgeschiedenis wissen",
"Registration enabled? ": "", "Import/export data": "Gegevens im-/exporteren",
"Report statistics? ": "", "Change password": "Wachtwoord wijzigen",
"Save preferences": "Opslaan voorkeuren", "Manage subscriptions": "Abonnementen beheren",
"Subscription manager": "Abonnees beheerder", "Manage tokens": "Toegangssleutels beheren",
"`x` subscriptions": "`x` abonnees", "Watch history": "Kijkgeschiedenis",
"Import/Export": "Importeer/Exporteer", "Delete account": "Account verwijderen",
"unsubscribe": "abonnement opzeggen", "Administrator preferences": "Beheerdersinstellingen",
"Subscriptions": "Abonnees", "Default homepage: ": "Standaard startpagina: ",
"`x` unseen notifications": "`x` onbekeken notificaties", "Feed menu: ": "Feedmenu:",
"search": "zoeken", "Top enabled? ": "Bovenkant inschakelen? ",
"Sign out": "Afmelden", "CAPTCHA enabled? ": "CAPTCHA gebruiken? ",
"Released under the AGPLv3 by Omar Roth.": "Uitgegeven onder AGPLv3 door Omar Roth.", "Login enabled? ": "Inloggen toestaan? ",
"Source available here.": "Bron beschikbaar hier.", "Registration enabled? ": "Registratie toestaan? ",
"View JavaScript license information.": "Bekijk JavaScript licentie informatie.", "Report statistics? ": "Statistieken bijhouden? ",
"View privacy policy.": "", "Save preferences": "Instellingen opslaan",
"Trending": "Trending", "Subscription manager": "Abonnementen beheren",
"Unlisted": "", "Token manager": "Toegangssleutels beheren",
"Watch video on Youtube": "Bekijk video op Youtube", "Token": "Toegangssleutel",
"Genre: ": "Genre: ", "`x` subscriptions": "`x` abonnementen",
"License: ": "Licentie: ", "`x` tokens": "`x` toegangssleutels",
"Family friendly? ": "Gezinsvriendelijk? ", "Import/export": "Importeren/Exporteren",
"Wilson score: ": "Wilson score: ", "unsubscribe": "Deabonneren",
"Engagement: ": "Betrokkenheid: ", "revoke": "Intrekken",
"Whitelisted regions: ": "Toegestane regio's: ", "Subscriptions": "Abonnementen",
"Blacklisted regions: ": "Geblokkeerde regio's: ", "`x` unseen notifications": "`x` ongelezen meldingen",
"Shared `x`": "`x` gedeeld", "search": "zoeken",
"Premieres in `x`": "", "Log out": "Uitloggen",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hoi! Het lijkt erop dat je JavaScript uit hebt staan. Klik hier om de reacties te bekijken, hou er rekening mee dat het wat langer duurt om te laden.", "Released under the AGPLv3 by Omar Roth.": "Uitgegeven onder de AGPLv3-licentie door Omar Roth.",
"View YouTube comments": "Bekijk YouTube reacties", "Source available here.": "De broncode is hier beschikbaar.",
"View more comments on Reddit": "Bekijk meer reacties op Reddit", "View JavaScript license information.": "JavaScript-licentieinformatie tonen.",
"View `x` comments": "`x` reacties zien", "View privacy policy.": "Privacybeleid tonen",
"View Reddit comments": "Bekijk Reddit reacties", "Trending": "Uitgelicht",
"Hide replies": "Verberg antwoorden", "Unlisted": "Verborgen",
"Show replies": "Laat antwoorden zien", "Watch on YouTube": "Bekijk video op YouTube",
"Incorrect password": "Onjuist wachtwoord", "Hide annotations": "Annotaties verbergen",
"Quota exceeded, try again in a few hours": "Quota overschreden, probeer het over een paar uur opnieuw", "Show annotations": "Annotaties tonen",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Niet in staat om in te loggen, zorg ervoor dat two-factor authentication (Authenticator of SMS) is ingeschakeld.", "Genre: ": "Genre: ",
"Invalid TFA code": "Onjuiste TFA code", "License: ": "Licentie: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Aanmelden mislukt. Dit kan zijn omdat two-factor authentication niet is ingeschakeld voor uw account.", "Family friendly? ": "Gezinsvriendelijk? ",
"Invalid answer": "Onjuist antwoord", "Wilson score: ": "Wilson-score: ",
"Invalid CAPTCHA": "Onjuiste CAPTCHA", "Engagement: ": "Betrokkenheid: ",
"CAPTCHA is a required field": "CAPTCHA is een vereist veld", "Whitelisted regions: ": "Toegestane regio's: ",
"User ID is a required field": "Gebruiker ID is een vereist veld", "Blacklisted regions: ": "Geblokkeerde regio's: ",
"Password is a required field": "Wachtwoord is een vereist veld", "Shared `x`": "`x` gedeeld",
"Invalid username or password": "Ongeldige gebruikersnaam of wachtwoord", "`x` views": "`x` weergaven",
"Please sign in using 'Sign in with Google'": "Meld u aan met 'Aanmelden met Google'", "Premieres in `x`": "Verschijnt over `x`",
"Password cannot be empty": "Wachtwoord mag niet leeg zijn", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hoi! Het lijkt erop dat je JavaScript hebt uitgeschakeld. Klik hier om de reacties te bekijken. Let op: het laden duurt wat langer.",
"Password cannot be longer than 55 characters": "Wachtwoord mag niet langer dan 55 tekens zijn", "View YouTube comments": "YouTube-reacties tonen",
"Please sign in": "Meld u aan", "View more comments on Reddit": "Meer reacties bekijken op Reddit",
"Invidious Private Feed for `x`": "Invidious Privé Feed voor `x`", "View `x` comments": "`x` reacties tonen",
"channel:`x`": "kanaal:`x`", "View Reddit comments": "Reddit-reacties tonen",
"Deleted or invalid channel": "Verwijderd of ongeldig kanaal", "Hide replies": "Antwoorden verbergen",
"This channel does not exist.": "Dit kanaal bestaat niet.", "Show replies": "Antwoorden tonen",
"Could not get channel info.": "Kan kanaal informatie niet verkrijgen.", "Incorrect password": "Wachtwoord is onjuist",
"Could not fetch comments": "Kan reacties niet verkrijgen", "Quota exceeded, try again in a few hours": "Quota overschreden; probeer het over een paar uur opnieuw",
"View `x` replies": "`x` antwoorden zien", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Kan niet inloggen. Zorg ervoor dat authenticatie in twee stappen (Authenticator of sms) is ingeschakeld.",
"`x` ago": "`x` geleden", "Invalid TFA code": "Onjuiste TFA-code",
"Load more": "Meer laden", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Inloggen mislukt. Wellicht is authenticatie in twee stappen niet ingeschakeld op je account.",
"`x` points": "`x` punten", "Wrong answer": "Onjuist antwoord",
"Could not create mix.": "Kon mix niet maken.", "Erroneous CAPTCHA": "Onjuiste CAPTCHA",
"Playlist is empty": "Afspeellijst is leeg", "CAPTCHA is a required field": "CAPTCHA is vereist",
"Invalid playlist.": "Ongeldige afspeellijst.", "User ID is a required field": "Gebruikers-id is vereist",
"Playlist does not exist.": "Afspeellijst bestaat niet.", "Password is a required field": "Wachtwoord is vereist",
"Could not pull trending pages.": "Kon trending paginas niet verkrijgen.", "Wrong username or password": "Onjuiste gebruikersnaam of wachtwoord",
"Hidden field \"challenge\" is a required field": "Verborgen veld \"uitdaging\" is een vereist veld", "Please sign in using 'Log in with Google'": "Log in via 'Inloggen met Google'",
"Hidden field \"token\" is a required field": "Verborgen veld \"token\" is een vereist veld", "Password cannot be empty": "Het wachtwoordveld mag niet leeg zijn",
"Invalid challenge": "Ongeldige uitdaging", "Password cannot be longer than 55 characters": "Het wachtwoord mag niet langer dan 55 tekens zijn",
"Invalid token": "Ongeldige token", "Please log in": "Log in",
"Invalid user": "Ongeldige gebruiker", "Invidious Private Feed for `x`": "Invidious-privéfeed van `x`",
"Token is expired, please try again": "Token is verlopen, probeer het opnieuw", "channel:`x`": "kanaal:`x`",
"English": "", "Deleted or invalid channel": "Verwijderd of niet-bestaand kanaal",
"English (auto-generated)": "", "This channel does not exist.": "Dit kanaal bestaat niet.",
"Afrikaans": "", "Could not get channel info.": "Kan geen kanaalinformatie ophalen.",
"Albanian": "", "Could not fetch comments": "Kan reacties niet ophalen",
"Amharic": "", "View `x` replies": "`x` antwoorden tonen",
"Arabic": "", "`x` ago": "`x` geleden",
"Armenian": "", "Load more": "Meer laden",
"Azerbaijani": "", "`x` points": "`x` punten",
"Bangla": "", "Could not create mix.": "Kan geen mix maken.",
"Basque": "", "Empty playlist": "Lege afspeellijst",
"Belarusian": "", "Not a playlist.": "Ongeldige afspeellijst.",
"Bosnian": "", "Playlist does not exist.": "Afspeellijst bestaat niet.",
"Bulgarian": "", "Could not pull trending pages.": "Kan uitgelichte pagina's niet ophalen.",
"Burmese": "", "Hidden field \"challenge\" is a required field": "Verborgen veld \"uitdaging\" is vereist",
"Catalan": "", "Hidden field \"token\" is a required field": "Verborgen veld \"toegangssleutel\" is vereist",
"Cebuano": "", "Erroneous challenge": "Ongeldige uitdaging",
"Chinese (Simplified)": "", "Erroneous token": "Ongeldige toegangssleutel",
"Chinese (Traditional)": "", "No such user": "Gebruiker bestaat niet",
"Corsican": "", "Token is expired, please try again": "Toegangssleutel verlopen; probeer het opnieuw",
"Croatian": "", "English": "Engels",
"Czech": "", "English (auto-generated)": "Engels (automatisch gegenereerd)",
"Danish": "", "Afrikaans": "Afrikaans",
"Dutch": "", "Albanian": "Albanees",
"Esperanto": "", "Amharic": "Amhaars",
"Estonian": "", "Arabic": "Arabisch",
"Filipino": "", "Armenian": "Armeens",
"Finnish": "", "Azerbaijani": "Azerbeidzjaans",
"French": "", "Bangla": "Bangla",
"Galician": "", "Basque": "Baskisch",
"Georgian": "", "Belarusian": "Wit-Rrussisch",
"German": "", "Bosnian": "Bosnisch",
"Greek": "", "Bulgarian": "Bulgaars",
"Gujarati": "", "Burmese": "Birmaans",
"Haitian Creole": "", "Catalan": "Catalaans",
"Hausa": "", "Cebuano": "Cebuano",
"Hawaiian": "", "Chinese (Simplified)": "Chinees (Veereenvoudigd)",
"Hebrew": "", "Chinese (Traditional)": "Chinees (Traditioneel)",
"Hindi": "", "Corsican": "Corsicaans",
"Hmong": "", "Croatian": "Kroatisch",
"Hungarian": "", "Czech": "Tsjechisch",
"Icelandic": "", "Danish": "Deens",
"Igbo": "", "Dutch": "Nederlands",
"Indonesian": "", "Esperanto": "Esperanto",
"Irish": "", "Estonian": "Ests",
"Italian": "", "Filipino": "Filipijns",
"Japanese": "", "Finnish": "Fins",
"Javanese": "", "French": "Frans",
"Kannada": "", "Galician": "Galicisch",
"Kazakh": "", "Georgian": "Georgisch",
"Khmer": "", "German": "Duits",
"Korean": "", "Greek": "Grieks",
"Kurdish": "", "Gujarati": "Gujarati",
"Kyrgyz": "", "Haitian Creole": "Creools",
"Lao": "", "Hausa": "Hausa",
"Latin": "", "Hawaiian": "Hawaïaans",
"Latvian": "", "Hebrew": "Heebreeuws",
"Lithuanian": "", "Hindi": "Hindi",
"Luxembourgish": "", "Hmong": "Hmong",
"Macedonian": "", "Hungarian": "Hongaars",
"Malagasy": "", "Icelandic": "IJslands",
"Malay": "", "Igbo": "Igbo",
"Malayalam": "", "Indonesian": "Indonesisch",
"Maltese": "", "Irish": "Iers",
"Maori": "", "Italian": "Italiaans",
"Marathi": "", "Japanese": "Japans",
"Mongolian": "", "Javanese": "Javaans",
"Nepali": "", "Kannada": "Kannada",
"Norwegian": "", "Kazakh": "Kazachs",
"Nyanja": "", "Khmer": "Khmer",
"Pashto": "", "Korean": "Koreaans",
"Persian": "", "Kurdish": "Koerdisch",
"Polish": "", "Kyrgyz": "Kirgizisch",
"Portuguese": "", "Lao": "Laotiaans",
"Punjabi": "", "Latin": "Latijns",
"Romanian": "", "Latvian": "Lets",
"Russian": "", "Lithuanian": "Litouws",
"Samoan": "", "Luxembourgish": "Luxemburgs",
"Scottish Gaelic": "", "Macedonian": "Macedonisch",
"Serbian": "", "Malagasy": "Malagassisch",
"Shona": "", "Malay": "Maleisisch",
"Sindhi": "", "Malayalam": "Malayalam",
"Sinhala": "", "Maltese": "Maltees",
"Slovak": "", "Maori": "Maorisch",
"Slovenian": "", "Marathi": "Marathi",
"Somali": "", "Mongolian": "Mongools",
"Southern Sotho": "", "Nepali": "Nepalees",
"Spanish": "", "Norwegian Bokmål": "Noors (Bokmål)",
"Spanish (Latin America)": "", "Nyanja": "Nyanja",
"Sundanese": "", "Pashto": "Pashto",
"Swahili": "", "Persian": "Perzisch",
"Swedish": "", "Polish": "Pools",
"Tajik": "", "Portuguese": "Portugees",
"Tamil": "", "Punjabi": "Punjabi",
"Telugu": "", "Romanian": "Roemeens",
"Thai": "", "Russian": "Russisch",
"Turkish": "", "Samoan": "Samoaans",
"Ukrainian": "", "Scottish Gaelic": "Schots-Gaelisch",
"Urdu": "", "Serbian": "Servisch",
"Uzbek": "", "Shona": "Shona",
"Vietnamese": "", "Sindhi": "Sindhi",
"Welsh": "", "Sinhala": "Sinhala",
"Western Frisian": "", "Slovak": "Slowaaks",
"Xhosa": "", "Slovenian": "Sloveens",
"Yiddish": "", "Somali": "Somalisch",
"Yoruba": "", "Southern Sotho": "Zuid-Sotho",
"Zulu": "", "Spanish": "Spaans",
"`x` years": "`x` jaar", "Spanish (Latin America)": "Spaans (Latijns-Amerika)",
"`x` months": "`x` maanden", "Sundanese": "Soedanees",
"`x` weeks": "`x` weken", "Swahili": "Swahili",
"`x` days": "`x` dagen", "Swedish": "Zweeds",
"`x` hours": "`x` uur", "Tajik": "Tajik",
"`x` minutes": "`x` minuten", "Tamil": "Tamil",
"`x` seconds": "`x` seconden", "Telugu": "Telugu",
"Fallback comments: ": "", "Thai": "Thaïs",
"Popular": "", "Turkish": "Turks",
"Top": "", "Ukrainian": "Oekraïens",
"About": "", "Urdu": "Urdu",
"Rating: ": "", "Uzbek": "Oezbeeks",
"Language: ": "", "Vietnamese": "Vietnamees",
"Default": "", "Welsh": "Welsh",
"Music": "", "Western Frisian": "Fries",
"Gaming": "", "Xhosa": "Xhosa",
"News": "", "Yiddish": "Joods",
"Movies": "", "Yoruba": "Yoruba",
"Download": "", "Zulu": "Zulu",
"Download as: ": "", "`x` years": "`x` jaar",
"%A %B %-d, %Y": "", "`x` months": "`x` maanden",
"(edited)": "", "`x` weeks": "`x` weken",
"Youtube permalink of the comment": "", "`x` days": "`x` dagen",
"`x` marked it with a ❤": "", "`x` hours": "`x` uur",
"Audio mode": "", "`x` minutes": "`x` minuten",
"Video mode": "", "`x` seconds": "`x` seconden",
"Videos": "", "Fallback comments: ": "Terugvallen op",
"Playlists": "", "Popular": "Populair",
"Current version: ": "" "Top": "Top",
"About": "Over",
"Rating: ": "Waardering",
"Language: ": "Taal",
"View as playlist": "Tonen als afspeellijst",
"Default": "Standaard",
"Music": "Muziek",
"Gaming": "Gaming",
"News": "Nieuws",
"Movies": "Films",
"Download": "Downloaden",
"Download as: ": "Downloaden als: ",
"%A %B %-d, %Y": "%A %B %-d, %Y",
"(edited)": "(bewerkt)",
"YouTube comment permalink": "Link naar YouTube-reactie",
"`x` marked it with a ❤": "`x` heeft dit gemarkeerd met ❤",
"Audio mode": "Audiomodus",
"Video mode": "Videomodus",
"Videos": "Video's",
"Playlists": "Afspeellijsten",
"Current version: ": "Huidige versie: "
} }

View File

@ -1,295 +1,315 @@
{ {
"`x` subscribers": "`x` subskrybcji", "`x` subscribers": "`x` subskrybcji",
"`x` videos": "`x` filmów", "`x` videos": "`x` filmów",
"LIVE": "NA ŻYWO", "LIVE": "NA ŻYWO",
"Shared `x` ago": "Udostępniono `x` temu", "Shared `x` ago": "Udostępniono `x` temu",
"Unsubscribe": "Odsubskrybuj", "Unsubscribe": "Odsubskrybuj",
"Subscribe": "Subskrybuj", "Subscribe": "Subskrybuj",
"Login to subscribe to `x`": "Zaloguj się, aby subskrybować `x`", "View channel on YouTube": "Wyświetl kanał na YouTube",
"View channel on YouTube": "Wyświetl kanał na YouTube", "View playlist on YouTube": "",
"newest": "najnowsze", "newest": "najnowsze",
"oldest": "najstarsze", "oldest": "najstarsze",
"popular": "popularne", "popular": "popularne",
"last": "ostatnie", "last": "ostatnie",
"Next page": "Następna strona", "Next page": "Następna strona",
"Previous page": "Poprzednia strona", "Previous page": "Poprzednia strona",
"Clear watch history?": "Wyczyścić historię?", "Clear watch history?": "Wyczyścić historię?",
"Yes": "Tak", "New password": "",
"No": "Nie", "New passwords must match": "",
"Import and Export Data": "Import i eksport danych", "Cannot change password for Google accounts": "",
"Import": "Import", "Authorize token?": "",
"Import Invidious data": "Importuj dane Invidious", "Authorize token for `x`?": "",
"Import YouTube subscriptions": "Importuj subskrybcje z YouTube", "Yes": "Tak",
"Import FreeTube subscriptions (.db)": "Importuj subskrybcje z FreeTube (.db)", "No": "Nie",
"Import NewPipe subscriptions (.json)": "Importuj subskrybcje z NewPipe (.json)", "Import and Export Data": "Import i eksport danych",
"Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)", "Import": "Import",
"Export": "Eksport", "Import Invidious data": "Importuj dane Invidious",
"Export subscriptions as OPML": "Eksportuj subskrybcje jako OPML", "Import YouTube subscriptions": "Importuj subskrybcje z YouTube",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrybcje jako OPML (dla NewPipe i FreeTube)", "Import FreeTube subscriptions (.db)": "Importuj subskrybcje z FreeTube (.db)",
"Export data as JSON": "Eksportuj dane jako JSON", "Import NewPipe subscriptions (.json)": "Importuj subskrybcje z NewPipe (.json)",
"Delete account?": "Usunąć konto?", "Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)",
"History": "Historia", "Export": "Eksport",
"An alternative front-end to YouTube": "Alternatywny front-end dla YouTube", "Export subscriptions as OPML": "Eksportuj subskrybcje jako OPML",
"JavaScript license information": "Informacja o licencji JavaScript", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrybcje jako OPML (dla NewPipe i FreeTube)",
"source": "źródło", "Export data as JSON": "Eksportuj dane jako JSON",
"Login": "Zaloguj", "Delete account?": "Usunąć konto?",
"Login/Register": "Zaloguj/Zarejestruj", "History": "Historia",
"Login to Google": "Zaloguj do Google", "An alternative front-end to YouTube": "Alternatywny front-end dla YouTube",
"User ID:": "ID użytkownika:", "JavaScript license information": "Informacja o licencji JavaScript",
"Password:": "Hasło:", "source": "źródło",
"Time (h:mm:ss):": "Godzina (h:mm:ss):", "Log in": "Zaloguj",
"Text CAPTCHA": "Tekst CAPTCHA", "Log in/register": "Zaloguj/Zarejestruj",
"Image CAPTCHA": "Obraz CAPTCHA", "Log in with Google": "Zaloguj do Google",
"Sign In": "Zaloguj się", "User ID": "ID użytkownika",
"Register": "Zarejestruj się", "Password": "Hasło",
"Email:": "Email:", "Time (h:mm:ss):": "Godzina (h:mm:ss):",
"Google verification code:": "Kod weryfikacyjny Google:", "Text CAPTCHA": "Tekst CAPTCHA",
"Preferences": "Preferencje", "Image CAPTCHA": "Obraz CAPTCHA",
"Player preferences": "Ustawienia odtwarzacza", "Sign In": "Zaloguj się",
"Always loop: ": "Zawsze zapętlaj: ", "Register": "Zarejestruj się",
"Autoplay: ": "Autoodtwarzanie: ", "E-mail": "Email",
"Autoplay next video: ": "Odtwórz następny film: ", "Google verification code": "Kod weryfikacyjny Google",
"Listen by default: ": "Tryb dźwiękowy: ", "Preferences": "Preferencje",
"Proxy videos? ": "Filmy przez proxy? ", "Player preferences": "Ustawienia odtwarzacza",
"Default speed: ": "Domyślna prędkość: ", "Always loop: ": "Zawsze zapętlaj: ",
"Preferred video quality: ": "Preferowana jakość filmów: ", "Autoplay: ": "Autoodtwarzanie: ",
"Player volume: ": "Głośność odtwarzacza: ", "Play next by default: ": "",
"Default comments: ": "Domyślne komentarze: ", "Autoplay next video: ": "Odtwórz następny film: ",
"Default captions: ": "Domyślne napisy: ", "Listen by default: ": "Tryb dźwiękowy: ",
"Fallback captions: ": "Zastępcze napisy: ", "Proxy videos? ": "Filmy przez proxy? ",
"Show related videos? ": "Pokaż powiązane filmy? ", "Default speed: ": "Domyślna prędkość: ",
"Visual preferences": "Preferencje Wizualne", "Preferred video quality: ": "Preferowana jakość filmów: ",
"Dark mode: ": "Ciemny motyw: ", "Player volume: ": "Głośność odtwarzacza: ",
"Thin mode: ": "Tryb minimalny: ", "Default comments: ": "Domyślne komentarze: ",
"Subscription preferences": "Preferencje subskrybcji", "youtube": "",
"Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ", "reddit": "",
"Number of videos shown in feed: ": "Liczba filmów widoczna na stronie subskrybcji: ", "Default captions: ": "Domyślne napisy: ",
"Sort videos by: ": "Sortuj filmy: ", "Fallback captions: ": "Zastępcze napisy: ",
"published": "po czasie publikacji", "Show related videos? ": "Pokaż powiązane filmy? ",
"published - reverse": "po czasie publikacji od najstarszych", "Show annotations by default? ": "",
"alphabetically": "alfabetycznie", "Visual preferences": "Preferencje Wizualne",
"alphabetically - reverse": "alfabetycznie od tyłu", "Dark mode: ": "Ciemny motyw: ",
"channel name": "po nazwie kanału", "Thin mode: ": "Tryb minimalny: ",
"channel name - reverse": "po nazwie kanału od tyłu", "Subscription preferences": "Preferencje subskrybcji",
"Only show latest video from channel: ": "Pokazuj tylko najnowszy film z kanału: ", "Show annotations by default for subscribed channels? ": "",
"Only show latest unwatched video from channel: ": "Pokazuj tylko najnowszy nie obejrzany film z kanału: ", "Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ",
"Only show unwatched: ": "Pokazuj tylko nie obejrzane: ", "Number of videos shown in feed: ": "Liczba filmów widoczna na stronie subskrybcji: ",
"Only show notifications (if there are any): ": "Pokazuj tylko powiadomienia (jeśli są): ", "Sort videos by: ": "Sortuj filmy: ",
"Data preferences": "Preferencje danych", "published": "po czasie publikacji",
"Clear watch history": "Wyczyść historię", "published - reverse": "po czasie publikacji od najstarszych",
"Import/Export data": "Import/Eksport danych", "alphabetically": "alfabetycznie",
"Manage subscriptions": "Organizuj subskrybcje", "alphabetically - reverse": "alfabetycznie od tyłu",
"Watch history": "Historia", "channel name": "po nazwie kanału",
"Delete account": "Usuń konto", "channel name - reverse": "po nazwie kanału od tyłu",
"Administrator preferences": "Preferencje administratora", "Only show latest video from channel: ": "Pokazuj tylko najnowszy film z kanału: ",
"Default homepage: ": "Domyślna strona główna: ", "Only show latest unwatched video from channel: ": "Pokazuj tylko najnowszy nie obejrzany film z kanału: ",
"Feed menu: ": "", "Only show unwatched: ": "Pokazuj tylko nie obejrzane: ",
"Top enabled? ": "", "Only show notifications (if there are any): ": "Pokazuj tylko powiadomienia (jeśli są): ",
"CAPTCHA enabled? ": "CAPTCHA aktywna? ", "Data preferences": "Preferencje danych",
"Login enabled? ": "Logowanie włączone? ", "Clear watch history": "Wyczyść historię",
"Registration enabled? ": "Rejestracja włączona? ", "Import/export data": "Import/Eksport danych",
"Report statistics? ": "Raportować statystyki? ", "Change password": "",
"Save preferences": "Zapisz preferencje", "Manage subscriptions": "Organizuj subskrybcje",
"Subscription manager": "Manager subskrybcji", "Manage tokens": "",
"`x` subscriptions": "`x` subskrybcji", "Watch history": "Historia",
"Import/Export": "Import/Eksport", "Delete account": "Usuń konto",
"unsubscribe": "odsubskrybuj", "Administrator preferences": "Preferencje administratora",
"Subscriptions": "Subskrybcje", "Default homepage: ": "Domyślna strona główna: ",
"`x` unseen notifications": "`x` niewidzianych powiadomień", "Feed menu: ": "",
"search": "szukaj", "Top enabled? ": "",
"Sign out": "Wyloguj", "CAPTCHA enabled? ": "CAPTCHA aktywna? ",
"Released under the AGPLv3 by Omar Roth.": "Wydano na licencji AGPLv3 przez Omar Roth.", "Login enabled? ": "Logowanie włączone? ",
"Source available here.": "Kod źródłowy dostępny tutaj.", "Registration enabled? ": "Rejestracja włączona? ",
"View JavaScript license information.": "Wyświetl informację o licencji JavaScript.", "Report statistics? ": "Raportować statystyki? ",
"View privacy policy.": "Polityka prywatności.", "Save preferences": "Zapisz preferencje",
"Trending": "Na czasie", "Subscription manager": "Manager subskrybcji",
"Unlisted": "", "Token manager": "",
"Watch video on Youtube": "Zobacz film na YouTube", "Token": "",
"Genre: ": "Gatunek: ", "`x` subscriptions": "`x` subskrybcji",
"License: ": "Licencja: ", "`x` tokens": "",
"Family friendly? ": "Przyjazny rodzinie? ", "Import/export": "Import/Eksport",
"Wilson score: ": "Punktacja Wilsona: ", "unsubscribe": "odsubskrybuj",
"Engagement: ": "Zaangażowanie: ", "revoke": "",
"Whitelisted regions: ": "Dostępny na obszarach: ", "Subscriptions": "Subskrybcje",
"Blacklisted regions: ": "Niedostępny na obszarach: ", "`x` unseen notifications": "`x` nowych powiadomień",
"Shared `x`": "Udostępniono `x`", "search": "szukaj",
"Premieres in `x`": "", "Log out": "Wyloguj",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Cześć! Wygląda na to, że masz wyłączoną obsługę JavaScriptu. Kliknij tutaj, żeby zobaczyć komentarze. Pamiętaj, że wczytywanie może potrwać dłużej.", "Released under the AGPLv3 by Omar Roth.": "Wydano na licencji AGPLv3 przez Omar Roth.",
"View YouTube comments": "Wyświetl komentarze z YouTube", "Source available here.": "Kod źródłowy dostępny tutaj.",
"View more comments on Reddit": "Wyświetl więcej komentarzy na Reddicie", "View JavaScript license information.": "Wyświetl informację o licencji JavaScript.",
"View `x` comments": "Wyświetl `x` komentarzy", "View privacy policy.": "Polityka prywatności.",
"View Reddit comments": "Wyświetl komentarze z Redditta", "Trending": "Na czasie",
"Hide replies": "Ukryj odpowiedzi", "Unlisted": "",
"Show replies": "Pokaż odpowiedzi", "Watch on YouTube": "Zobacz film na YouTube",
"Incorrect password": "Niepoprawne hasło", "Hide annotations": "",
"Quota exceeded, try again in a few hours": "Przekroczony limit zapytań, spróbuj ponownie za kilka godzin", "Show annotations": "",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Nie udało się zalogować, upewnij się, że dwuetapowe uwierzytelnianie (Autentykator lub SMS) jest aktywne.", "Genre: ": "Gatunek: ",
"Invalid TFA code": "Niepoprawny kod TFA", "License: ": "Licencja: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Nie udało się zalogować. To może być spowodowane wyłączoną dwustopniową autoryzacją na twoim koncie.", "Family friendly? ": "Przyjazny rodzinie? ",
"Invalid answer": "Niepoprawna odpowiedź", "Wilson score: ": "Punktacja Wilsona: ",
"Invalid CAPTCHA": "CAPTCHA wykonane błędnie", "Engagement: ": "Zaangażowanie: ",
"CAPTCHA is a required field": "CAPTCHA jest polem wymaganym", "Whitelisted regions: ": "Dostępny na obszarach: ",
"User ID is a required field": "ID użytkownika jest polem wymaganym", "Blacklisted regions: ": "Niedostępny na obszarach: ",
"Password is a required field": "Hasło jest polem wymaganym", "Shared `x`": "Udostępniono `x`",
"Invalid username or password": "Niepoprawny login lub hasło", "`x` views": "`x` wyświetleń",
"Please sign in using 'Sign in with Google'": "Zaloguj się używając \"Zaloguj się przez Google\"", "Premieres in `x`": "Publikacja za `x`",
"Password cannot be empty": "Hasło nie może być puste", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Cześć! Wygląda na to, że masz wyłączoną obsługę JavaScriptu. Kliknij tutaj, żeby zobaczyć komentarze. Pamiętaj, że wczytywanie może potrwać dłużej.",
"Password cannot be longer than 55 characters": "Hasło nie może być dłuższe niż 55 znaków", "View YouTube comments": "Wyświetl komentarze z YouTube",
"Please sign in": "Proszę się zalogować", "View more comments on Reddit": "Wyświetl więcej komentarzy na Reddicie",
"Invidious Private Feed for `x`": "", "View `x` comments": "Wyświetl `x` komentarzy",
"channel:`x`": "kanał:`x", "View Reddit comments": "Wyświetl komentarze z Redditta",
"Deleted or invalid channel": "Usunięty lub niepoprawny kanał", "Hide replies": "Ukryj odpowiedzi",
"This channel does not exist.": "Ten kanał nie istnieje.", "Show replies": "Pokaż odpowiedzi",
"Could not get channel info.": "Nie udało się uzyskać informacji o kanale.", "Incorrect password": "Niepoprawne hasło",
"Could not fetch comments": "Nie udało się pobrać komentarzy", "Quota exceeded, try again in a few hours": "Przekroczony limit zapytań, spróbuj ponownie za kilka godzin",
"View `x` replies": "Wyświetl `x` odpowiedzi", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Nie udało się zalogować, upewnij się, że dwuetapowe uwierzytelnianie (Autentykator lub SMS) jest aktywne.",
"`x` ago": "`x` temu", "Invalid TFA code": "Niepoprawny kod TFA",
"Load more": "Wczytaj więcej", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Nie udało się zalogować. To może być spowodowane wyłączoną dwustopniową autoryzacją na twoim koncie.",
"`x` points": "`x` punktów", "Wrong answer": "Niepoprawna odpowiedź",
"Could not create mix.": "Nie udało się utworzyć miksu.", "Erroneous CAPTCHA": "CAPTCHA wykonane błędnie",
"Playlist is empty": "Lista odtwarzania jest pusta", "CAPTCHA is a required field": "CAPTCHA jest polem wymaganym",
"Invalid playlist.": "Niepoprawna lista.", "User ID is a required field": "ID użytkownika jest polem wymaganym",
"Playlist does not exist.": "Lista odtwarzania nie istnieje.", "Password is a required field": "Hasło jest polem wymaganym",
"Could not pull trending pages.": "Nie udało się pobrać strony na czasie.", "Wrong username or password": "Niepoprawny login lub hasło",
"Hidden field \"challenge\" is a required field": "Ukryte pole \"wyzwanie\" jest polem wymaganym", "Please sign in using 'Log in with Google'": "Zaloguj się używając \"Zaloguj się przez Google\"",
"Hidden field \"token\" is a required field": "Ukryte pole \"token\" jest polem wymaganym", "Password cannot be empty": "Hasło nie może być puste",
"Invalid challenge": "Niepoprawne wyzwanie", "Password cannot be longer than 55 characters": "Hasło nie może być dłuższe niż 55 znaków",
"Invalid token": "Niepoprawny token", "Please log in": "Proszę się zalogować",
"Invalid user": "Niepoprawny użytkownik", "Invidious Private Feed for `x`": "",
"Token is expired, please try again": "Token wygasł, spróbuj ponownie", "channel:`x`": "kanał:`x",
"English": "angielski", "Deleted or invalid channel": "Usunięty lub niepoprawny kanał",
"English (auto-generated)": "angielski (automatycznie generowane)", "This channel does not exist.": "Ten kanał nie istnieje.",
"Afrikaans": "afrykanerski", "Could not get channel info.": "Nie udało się uzyskać informacji o kanale.",
"Albanian": "albański", "Could not fetch comments": "Nie udało się pobrać komentarzy",
"Amharic": "amharski", "View `x` replies": "Wyświetl `x` odpowiedzi",
"Arabic": "arabski", "`x` ago": "`x` temu",
"Armenian": "armeński", "Load more": "Wczytaj więcej",
"Azerbaijani": "azerski", "`x` points": "`x` punktów",
"Bangla": "bengalski", "Could not create mix.": "Nie udało się utworzyć miksu.",
"Basque": "baskijski", "Empty playlist": "Lista odtwarzania jest pusta",
"Belarusian": "białoruski", "Not a playlist.": "Niepoprawna lista.",
"Bosnian": "bośniacki", "Playlist does not exist.": "Lista odtwarzania nie istnieje.",
"Bulgarian": "bułgarski", "Could not pull trending pages.": "Nie udało się pobrać strony na czasie.",
"Burmese": "birmański", "Hidden field \"challenge\" is a required field": "Ukryte pole \"wyzwanie\" jest polem wymaganym",
"Catalan": "kataloński", "Hidden field \"token\" is a required field": "Ukryte pole \"token\" jest polem wymaganym",
"Cebuano": "cebuański", "Erroneous challenge": "Niepoprawne wyzwanie",
"Chinese (Simplified)": "chiński (uproszczony)", "Erroneous token": "Niepoprawny token",
"Chinese (Traditional)": "chiński (tradycyjny)", "No such user": "Niepoprawny użytkownik",
"Corsican": "korsykański", "Token is expired, please try again": "Token wygasł, spróbuj ponownie",
"Croatian": "chorwacki", "English": "angielski",
"Czech": "czeski", "English (auto-generated)": "angielski (automatycznie generowane)",
"Danish": "duński", "Afrikaans": "afrykanerski",
"Dutch": "holenderski", "Albanian": "albański",
"Esperanto": "esperanto", "Amharic": "amharski",
"Estonian": "estoński", "Arabic": "arabski",
"Filipino": "filipiński", "Armenian": "armeński",
"Finnish": "fiński", "Azerbaijani": "azerski",
"French": "francuski", "Bangla": "bengalski",
"Galician": "galicyjski", "Basque": "baskijski",
"Georgian": "gruziński", "Belarusian": "białoruski",
"German": "niemiecki", "Bosnian": "bośniacki",
"Greek": "grecki", "Bulgarian": "bułgarski",
"Gujarati": "gudźarati", "Burmese": "birmański",
"Haitian Creole": "kreolski haitański", "Catalan": "kataloński",
"Hausa": "hausa", "Cebuano": "cebuański",
"Hawaiian": "hawajski", "Chinese (Simplified)": "chiński (uproszczony)",
"Hebrew": "hebrajski", "Chinese (Traditional)": "chiński (tradycyjny)",
"Hindi": "hindi", "Corsican": "korsykański",
"Hmong": "hmong", "Croatian": "chorwacki",
"Hungarian": "węgierski", "Czech": "czeski",
"Icelandic": "islandzki", "Danish": "duński",
"Igbo": "ibo", "Dutch": "holenderski",
"Indonesian": "indonezyjski", "Esperanto": "esperanto",
"Irish": "irlandzki", "Estonian": "estoński",
"Italian": "włoski", "Filipino": "filipiński",
"Japanese": "japoński", "Finnish": "fiński",
"Javanese": "jawajski", "French": "francuski",
"Kannada": "kannada", "Galician": "galicyjski",
"Kazakh": "kazachski", "Georgian": "gruziński",
"Khmer": "khmerski", "German": "niemiecki",
"Korean": "koreański", "Greek": "grecki",
"Kurdish": "kurdyjski", "Gujarati": "gudźarati",
"Kyrgyz": "kirgiski", "Haitian Creole": "kreolski haitański",
"Lao": "laotański", "Hausa": "hausa",
"Latin": "łaciński", "Hawaiian": "hawajski",
"Latvian": "łotewski", "Hebrew": "hebrajski",
"Lithuanian": "litewski", "Hindi": "hindi",
"Luxembourgish": "luksemburski", "Hmong": "hmong",
"Macedonian": "macedoński", "Hungarian": "węgierski",
"Malagasy": "malgaski", "Icelandic": "islandzki",
"Malay": "malajski", "Igbo": "ibo",
"Malayalam": "malajalam", "Indonesian": "indonezyjski",
"Maltese": "maltański", "Irish": "irlandzki",
"Maori": "maoryski", "Italian": "włoski",
"Marathi": "marathi", "Japanese": "japoński",
"Mongolian": "mongolski", "Javanese": "jawajski",
"Nepali": "nepalski", "Kannada": "kannada",
"Norwegian": "norweski", "Kazakh": "kazachski",
"Nyanja": "njandża", "Khmer": "khmerski",
"Pashto": "paszto", "Korean": "koreański",
"Persian": "perski", "Kurdish": "kurdyjski",
"Polish": "polski", "Kyrgyz": "kirgiski",
"Portuguese": "portugalski", "Lao": "laotański",
"Punjabi": "pendżabski", "Latin": "łaciński",
"Romanian": "rumuński", "Latvian": "łotewski",
"Russian": "rosyjski", "Lithuanian": "litewski",
"Samoan": "samoański", "Luxembourgish": "luksemburski",
"Scottish Gaelic": "gaelicki szkocki", "Macedonian": "macedoński",
"Serbian": "serbski", "Malagasy": "malgaski",
"Shona": "shona", "Malay": "malajski",
"Sindhi": "sindhi", "Malayalam": "malajalam",
"Sinhala": "syngaleski", "Maltese": "maltański",
"Slovak": "słowacki", "Maori": "maoryski",
"Slovenian": "słoweński", "Marathi": "marathi",
"Somali": "somalijski", "Mongolian": "mongolski",
"Southern Sotho": "sotho południowy", "Nepali": "nepalski",
"Spanish": "hiszpański", "Norwegian Bokmål": "norweski",
"Spanish (Latin America)": "hiszpański (ameryka łacińska)", "Nyanja": "njandża",
"Sundanese": "sundajski", "Pashto": "paszto",
"Swahili": "suahili", "Persian": "perski",
"Swedish": "szwedzki", "Polish": "polski",
"Tajik": "tadżycki", "Portuguese": "portugalski",
"Tamil": "tamilski", "Punjabi": "pendżabski",
"Telugu": "telugu", "Romanian": "rumuński",
"Thai": "tajski", "Russian": "rosyjski",
"Turkish": "turecki", "Samoan": "samoański",
"Ukrainian": "ukraiński", "Scottish Gaelic": "gaelicki szkocki",
"Urdu": "urdu", "Serbian": "serbski",
"Uzbek": "uzbecki", "Shona": "shona",
"Vietnamese": "wietnamski", "Sindhi": "sindhi",
"Welsh": "walijski", "Sinhala": "syngaleski",
"Western Frisian": "zachodniofryzyjski", "Slovak": "słowacki",
"Xhosa": "xhosa", "Slovenian": "słoweński",
"Yiddish": "jidysz", "Somali": "somalijski",
"Yoruba": "joruba", "Southern Sotho": "sotho południowy",
"Zulu": "zuluski", "Spanish": "hiszpański",
"`x` years": "`x` lat", "Spanish (Latin America)": "hiszpański (ameryka łacińska)",
"`x` months": "`x` miesięcy", "Sundanese": "sundajski",
"`x` weeks": "`x` tygodni", "Swahili": "suahili",
"`x` days": "`x` dni", "Swedish": "szwedzki",
"`x` hours": "`x` godzin", "Tajik": "tadżycki",
"`x` minutes": "`x` minut", "Tamil": "tamilski",
"`x` seconds": "`x` sekund", "Telugu": "telugu",
"Fallback comments: ": "Zastępcze komentarze: ", "Thai": "tajski",
"Popular": "Popularne", "Turkish": "turecki",
"Top": "Najczęściej oglądane", "Ukrainian": "ukraiński",
"About": "Informacje", "Urdu": "urdu",
"Rating: ": "Ocena: ", "Uzbek": "uzbecki",
"Language: ": "Język: ", "Vietnamese": "wietnamski",
"Default": "Domyślnie", "Welsh": "walijski",
"Music": "Muzyka", "Western Frisian": "zachodniofryzyjski",
"Gaming": "Gry", "Xhosa": "xhosa",
"News": "Wiadomości", "Yiddish": "jidysz",
"Movies": "Filmy", "Yoruba": "joruba",
"Download": "Pobierz", "Zulu": "zuluski",
"Download as: ": "Pobierz jako: ", "`x` years": "`x` lat",
"%A %B %-d, %Y": "", "`x` months": "`x` miesięcy",
"(edited)": "(edytowany)", "`x` weeks": "`x` tygodni",
"Youtube permalink of the comment": "Odnośnik bezpośredni do komentarza na YouTube", "`x` days": "`x` dni",
"`x` marked it with a ❤": "'x' oznaczonych ❤", "`x` hours": "`x` godzin",
"Audio mode": "Tryb audio", "`x` minutes": "`x` minut",
"Video mode": "Tryb wideo", "`x` seconds": "`x` sekund",
"Videos": "Filmy", "Fallback comments: ": "Zastępcze komentarze: ",
"Playlists": "Playlisty", "Popular": "Popularne",
"Current version: ": "Aktualna wersja: " "Top": "Najczęściej oglądane",
} "About": "Informacje",
"Rating: ": "Ocena: ",
"Language: ": "Język: ",
"View as playlist": "Obejrzyj w playliście",
"Default": "Domyślnie",
"Music": "Muzyka",
"Gaming": "Gry",
"News": "Wiadomości",
"Movies": "Filmy",
"Download": "Pobierz",
"Download as: ": "Pobierz jako: ",
"%A %B %-d, %Y": "",
"(edited)": "(edytowany)",
"YouTube comment permalink": "Odnośnik bezpośredni do komentarza na YouTube",
"`x` marked it with a ❤": "`x` oznaczonych ❤",
"Audio mode": "Tryb audio",
"Video mode": "Tryb wideo",
"Videos": "Filmy",
"Playlists": "Playlisty",
"Current version: ": "Aktualna wersja: "
}

View File

@ -1,297 +1,315 @@
{ {
"`x` subscribers": "`x` подписчиков", "`x` subscribers": "`x` подписчиков",
"`x` videos": "`x` видео", "`x` videos": "`x` видео",
"LIVE": "ПРЯМОЙ ЭФИР", "LIVE": "ПРЯМОЙ ЭФИР",
"Shared `x` ago": "Опубликовано `x` назад", "Shared `x` ago": "Опубликовано `x` назад",
"Unsubscribe": "Отписаться", "Unsubscribe": "Отписаться",
"Subscribe": "Подписаться", "Subscribe": "Подписаться",
"Login to subscribe to `x`": "Войти, чтобы подписаться на `x`", "View channel on YouTube": "Смотреть канал на YouTube",
"View channel on YouTube": "Канал на YouTube", "View playlist on YouTube": "",
"newest": "новые", "newest": "самые свежие",
"oldest": "старые", "oldest": "самые старые",
"popular": "популярные", "popular": "популярные",
"last": "недавно обновленные", "last": "недавние",
"Next page": "Следующая страница", "Next page": "Следующая страница",
"Previous page": "Предыдущая страница", "Previous page": "Предыдущая страница",
"Clear watch history?": "Очистить историю просмотров?", "Clear watch history?": "Очистить историю просмотров?",
"Yes": "Да", "New password": "Новый пароль",
"No": "Нет", "New passwords must match": "Новые пароли не совпадают",
"Import and Export Data": "Импорт и экспорт данных", "Cannot change password for Google accounts": "Изменить пароль аккаунта Google невозможно",
"Import": "Импорт", "Authorize token?": "Авторизовать токен?",
"Import Invidious data": "Импортировать данные Invidious", "Authorize token for `x`?": "Авторизовать токен для `x`?",
"Import YouTube subscriptions": "Импортировать YouTube подписки", "Yes": "Да",
"Import FreeTube subscriptions (.db)": "Импортировать FreeTube подписки (.db)", "No": "Нет",
"Import NewPipe subscriptions (.json)": "Импортировать NewPipe подписки (.json)", "Import and Export Data": "Импорт и экспорт данных",
"Import NewPipe data (.zip)": "Импортировать данные NewPipe (.zip)", "Import": "Импорт",
"Export": "Экспорт", "Import Invidious data": "Импортировать данные Invidious",
"Export subscriptions as OPML": "Экспортировать подписки в OPML", "Import YouTube subscriptions": "Импортировать подписки из YouTube",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Экспортировать подписки в OPML (для NewPipe и FreeTube)", "Import FreeTube subscriptions (.db)": "Импортировать подписки из FreeTube (.db)",
"Export data as JSON": "Экспортировать данные в JSON", "Import NewPipe subscriptions (.json)": "Импортировать подписки из NewPipe (.json)",
"Delete account?": "Удалить аккаунт?", "Import NewPipe data (.zip)": "Импортировать данные из NewPipe (.zip)",
"History": "История", "Export": "Экспорт",
"An alternative front-end to YouTube": "Альтернативный фронтенд для YouTube", "Export subscriptions as OPML": "Экспортировать подписки в формате OPML",
"JavaScript license information": "Лицензии JavaScript", "Export subscriptions as OPML (for NewPipe & FreeTube)": "Экспортировать подписки в формате OPML (для NewPipe и FreeTube)",
"source": "источник", "Export data as JSON": "Экспортировать данные в формате JSON",
"Login": "Войти", "Delete account?": "Удалить аккаунт?",
"Login/Register": "Войти/Регистрация", "History": "История",
"Login to Google": "Войти через Google", "An alternative front-end to YouTube": "Альтернативный фронтенд для YouTube",
"User ID:": "ID пользователя:", "JavaScript license information": "Информация о лицензиях JavaScript",
"Password:": "Пароль:", "source": "источник",
"Time (h:mm:ss):": "Время (ч:мм:сс):", "Log in": "Войти",
"Text CAPTCHA": "Текст капчи", "Log in/register": "Войти или зарегистрироваться",
"Image CAPTCHA": "Изображение капчи", "Log in with Google": "Войти через Google",
"Sign In": "Войти", "User ID": "ID пользователя",
"Register": "Регистрация", "Password": "Пароль",
"Email:": "Эл. почта:", "Time (h:mm:ss):": "Время (ч:мм:сс):",
"Google verification code:": "Код подтверждения Google:", "Text CAPTCHA": "Текст капчи",
"Preferences": "Настройки", "Image CAPTCHA": "Изображение капчи",
"Player preferences": "Настройки проигрывателя", "Sign In": "Войти",
"Always loop: ": "Всегда повторять: ", "Register": "Зарегистрироваться",
"Autoplay: ": "Автовоспроизведение: ", "E-mail": "Электронная почта",
"Autoplay next video: ": "Автовоспроизведение следующего видео: ", "Google verification code": "Код подтверждения Google",
"Listen by default: ": "Режим \"только аудио\" по-умолчанию: ", "Preferences": "Настройки",
"Proxy videos? ": "Проксировать видео? ", "Player preferences": "Настройки проигрывателя",
"Default speed: ": "Скорость по-умолчанию: ", "Always loop: ": "Всегда повторять: ",
"Preferred video quality: ": "Предпочтительное качество видео: ", "Autoplay: ": "Автовоспроизведение: ",
"Player volume: ": "Громкость воспроизведения: ", "Play next by default: ": "Всегда включать следующее видео? ",
"Default comments: ": "Источник комментариев: ", "Autoplay next video: ": "Автопроигрывание следующего видео: ",
"youtube": "YouTube", "Listen by default: ": "Режим «только аудио» по умолчанию: ",
"reddit": "Reddit", "Proxy videos? ": "Проигрывать видео через прокси? ",
"Default captions: ": "Субтитры по-умолчанию: ", "Default speed: ": "Скорость видео по умолчанию: ",
"Fallback captions: ": "Резервные субтитры: ", "Preferred video quality: ": "Предпочтительное качество видео: ",
"Show related videos? ": "Показывать похожие видео? ", "Player volume: ": "Громкость видео: ",
"Visual preferences": "Визуальные настройки", "Default comments: ": "Источник комментариев: ",
"Dark mode: ": "Темная тема: ", "youtube": "YouTube",
"Thin mode: ": "Облегченный режим: ", "reddit": "Reddit",
"Subscription preferences": "Настройки подписок", "Default captions: ": "Основной язык субтитров: ",
"Redirect homepage to feed: ": "Отображать ленту вместо главной страницы: ", "Fallback captions: ": "Дополнительный язык субтитров: ",
"Number of videos shown in feed: ": "Число видео в ленте: ", "Show related videos? ": "Показывать похожие видео? ",
"Sort videos by: ": "Сортировать видео по: ", "Show annotations by default? ": "Всегда показывать аннотации? ",
"published": "дате публикации", "Visual preferences": "Настройки сайта",
"published - reverse": "дате - обратный порядок", "Dark mode: ": "Тёмное оформление: ",
"alphabetically": "алфавиту", "Thin mode: ": "Облегчённое оформление: ",
"alphabetically - reverse": "алфавиту - обратный порядок", "Subscription preferences": "Настройки подписок",
"channel name": "имени канала", "Show annotations by default for subscribed channels? ": "Всегда показывать аннотации в видео каналов, на которые вы подписаны? ",
"channel name - reverse": "имени канала - обратный порядок", "Redirect homepage to feed: ": "Отображать видео с каналов, на которые вы подписаны, как главную страницу: ",
"Only show latest video from channel: ": "Отображать только последние видео с каждого канала: ", "Number of videos shown in feed: ": "Число видео, на которые вы подписаны, в ленте: ",
"Only show latest unwatched video from channel: ": "Отображать только непросмотренные видео с каждого канала: ", "Sort videos by: ": "Сортировать видео: ",
"Only show unwatched: ": "Отображать только непросмотренные видео: ", "published": "по дате публикации",
"Only show notifications (if there are any): ": "Отображать только оповещения (если есть): ", "published - reverse": "по дате публикации в обратном порядке",
"Data preferences": "Настройки данных", "alphabetically": "по алфавиту",
"Clear watch history": "Очистить историю просмотра", "alphabetically - reverse": "по алфавиту в обратном порядке",
"Import/Export data": "Импорт/Экспорт данных", "channel name": "по названию канала",
"Manage subscriptions": "Управление подписками", "channel name - reverse": "по названию канала в обратном порядке",
"Watch history": "История просмотров", "Only show latest video from channel: ": "Показывать только последние видео с каналов: ",
"Delete account": "Удалить аккаунт", "Only show latest unwatched video from channel: ": "Показывать только непросмотренные видео с каналов: ",
"Administrator preferences": "Настройки администратора", "Only show unwatched: ": "Показывать только непросмотренные видео: ",
"Default homepage: ": "Главная страница по умолчанию: ", "Only show notifications (if there are any): ": "Показывать только оповещения, если они есть: ",
"Feed menu: ": "Меню ленты: ", "Data preferences": "Настройки данных",
"Top enabled? ": "Включить ТОП? ", "Clear watch history": "Очистить историю просмотров",
"CAPTCHA enabled? ": "Включить капчу? ", "Import/export data": "Импорт/Экспорт данных",
"Login enabled? ": "Включить логин? ", "Change password": "Изменить пароль",
"Registration enabled? ": "Включить регистрацию? ", "Manage subscriptions": "Управлять подписками",
"Report statistics? ": "Отображать статистику? ", "Manage tokens": "Управлять токенами",
"Save preferences": "Сохранить настройки", "Watch history": "История просмотров",
"Subscription manager": "Менеджер подписок", "Delete account": "Удалить аккаунт",
"`x` subscriptions": "`x` подписок", "Administrator preferences": "Администраторские настройки",
"Import/Export": "Импорт/Экспорт", "Default homepage: ": "Главная страница по умолчанию: ",
"unsubscribe": "отписаться", "Feed menu: ": "Меню ленты видео: ",
"Subscriptions": "Подписки", "Top enabled? ": "Включить топ видео? ",
"`x` unseen notifications": "`x` новых оповещений", "CAPTCHA enabled? ": "Включить капчу? ",
"search": "поиск", "Login enabled? ": "Включить авторизацию? ",
"Sign out": "Выйти", "Registration enabled? ": "Включить регистрацию? ",
"Released under the AGPLv3 by Omar Roth.": "Распространяется Omar Roth по AGPLv3.", "Report statistics? ": "Сообщать статистику? ",
"Source available here.": "Исходный код доступен здесь.", "Save preferences": "Сохранить настройки",
"View JavaScript license information.": "Посмотреть лицензии JavaScript кода.", "Subscription manager": "Менеджер подписок",
"View privacy policy.": "См. политику конфиденциальности.", "Token manager": "Менеджер токенов",
"Trending": "В тренде", "Token": "Токен",
"Unlisted": "", "`x` subscriptions": "`x` подписок",
"Watch video on Youtube": "Смотреть на YouTube", "`x` tokens": "`x` токенов",
"Genre: ": "Жанр: ", "Import/export": "Импорт и экспорт",
"License: ": "Лицензия: ", "unsubscribe": "отписаться",
"Family friendly? ": "Семейный просмотр: ", "revoke": "отозвать",
"Wilson score: ": "Рейтинг Вильсона: ", "Subscriptions": "Подписки",
"Engagement: ": "Вовлеченность: ", "`x` unseen notifications": "`x` непросмотренных оповещений",
"Whitelisted regions: ": "Доступно для: ", "search": "поиск",
"Blacklisted regions: ": "Недоступно для: ", "Log out": "Выйти",
"Shared `x`": "Опубликовано `x`", "Released under the AGPLv3 by Omar Roth.": "Реализовано Омаром Ротом по лицензии AGPLv3.",
"Premieres in `x`": "", "Source available here.": "Исходный код доступен здесь.",
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Похоже, что у Вас отключен JavaScript. Нажмите сюда, чтобы увидеть комментарии (учтите, что они могут загружаться дольше).", "View JavaScript license information.": "Посмотреть информацию по лицензии JavaScript.",
"View YouTube comments": "Смотреть комментарии с YouTube", "View privacy policy.": "Посмотреть политику конфиденциальности.",
"View more comments on Reddit": "Больше комментариев на Reddit", "Trending": "В тренде",
"View `x` comments": "Показать `x` комментариев", "Unlisted": "Нет в списке",
"View Reddit comments": "Смотреть комментарии с Reddit", "Watch on YouTube": "Смотреть на YouTube",
"Hide replies": "Скрыть ответы", "Hide annotations": "Скрыть аннотации",
"Show replies": "Показать ответы", "Show annotations": "Показать аннотации",
"Incorrect password": "Неправильный пароль", "Genre: ": "Жанр: ",
"Quota exceeded, try again in a few hours": "Превышена квота, попробуйте снова через несколько часов", "License: ": "Лицензия: ",
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Вход не выполнен, проверьте, не включена ли двухфакторная аутентификация.", "Family friendly? ": "Семейный просмотр: ",
"Invalid TFA code": "Неправильный TFA код", "Wilson score: ": "Рейтинг Уилсона: ",
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Не удалось войти. Это может быть из-за того, что в вашем аккаунте не включена двухфакторная аутентификация.", "Engagement: ": "Вовлечённость: ",
"Invalid answer": "Неверный ответ", "Whitelisted regions: ": "Доступно в регионах: ",
"Invalid CAPTCHA": "Неверная капча", "Blacklisted regions: ": "Недоступно в регионах: ",
"CAPTCHA is a required field": "Необходимо ввести капчу", "Shared `x`": "Опубликовано `x`",
"User ID is a required field": "Необходимо ввести идентификатор пользователя", "`x` views": "`x` просмотров",
"Password is a required field": "Необходимо ввести пароль", "Premieres in `x`": "Премьера через `x`",
"Invalid username or password": "Недопустимый пароль или имя пользователя", "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Похоже, у вас отключён JavaScript. Чтобы увидить комментарии, нажмите сюда, но учтите: они могут загружаться немного медленнее.",
"Please sign in using 'Sign in with Google'": "Пожалуйста войдите через Google", "View YouTube comments": "Смотреть комментарии с YouTube",
"Password cannot be empty": "Пароль не может быть пустым", "View more comments on Reddit": "Посмотреть больше комментариев на Reddit",
"Password cannot be longer than 55 characters": "Пароль не может быть длиннее 55 символов", "View `x` comments": "Показать `x` комментариев",
"Please sign in": "Пожалуйста, войдите", "View Reddit comments": "Смотреть комментарии с Reddit",
"Invidious Private Feed for `x`": "Приватная лента Invidious для `x`", "Hide replies": "Скрыть ответы",
"channel:`x`": "канал: `x`", "Show replies": "Показать ответы",
"Deleted or invalid channel": "Канал удален или не найден", "Incorrect password": "Неправильный пароль",
"This channel does not exist.": "Такой канал не существует.", "Quota exceeded, try again in a few hours": "Лимит превышен, попробуйте снова через несколько часов",
"Could not get channel info.": "Невозможно получить информацию о канале.", "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Войти не удаётся. Проверьте, не включена ли двухфакторная аутентификация (по коду или смс).",
"Could not fetch comments": "Невозможно получить комментарии", "Invalid TFA code": "Неправильный код двухфакторной аутентификации",
"View `x` replies": "Показать `x` ответов", "Login failed. This may be because two-factor authentication is not turned on for your account.": "Не удаётся войти. Это может быть из-за того, что в вашем аккаунте не включена двухфакторная аутентификация.",
"`x` ago": "`x` назад", "Wrong answer": "Неправильный ответ",
"Load more": "Загрузить больше", "Erroneous CAPTCHA": "Неправильная капча",
"`x` points": "`x` очков", "CAPTCHA is a required field": "Необходимо пройти капчу",
"Could not create mix.": "Невозможно создать \"микс\".", "User ID is a required field": "Необходимо ввести ID пользователя",
"Playlist is empty": "Плейлист пуст", "Password is a required field": "Необходимо ввести пароль",
"Invalid playlist.": "Некорректный плейлист.", "Wrong username or password": "Неправильный логин или пароль",
"Playlist does not exist.": "Плейлист не существует.", "Please sign in using 'Log in with Google'": "Пожалуйста, нажмите «Войти через Google»",
"Could not pull trending pages.": "Невозможно получить страницы \"в тренде\".", "Password cannot be empty": "Пароль не может быть пустым",
"Hidden field \"challenge\" is a required field": "Необходимо заполнить скрытое поле \"challenge\"", "Password cannot be longer than 55 characters": "Пароль не может быть длиннее 55 символов",
"Hidden field \"token\" is a required field": "Необходимо заполнить скрытое поле \"токен\"", "Please log in": "Пожалуйста, войдите",
"Invalid challenge": "Неправильный ответ в \"challenge\"", "Invidious Private Feed for `x`": "Приватная лента Invidious для `x`",
"Invalid token": "Неправильный токен", "channel:`x`": "канал: `x`",
"Invalid user": "Недопустимое имя пользователя", "Deleted or invalid channel": "Канал удалён или не найден",
"Token is expired, please try again": "Срок действия токена истек, попробуйте позже", "This channel does not exist.": "Такого канала не существует.",
"English": "Английский", "Could not get channel info.": "Не удаётся получить информацию об этом канале.",
"English (auto-generated)": "Английский (созданы автоматически)", "Could not fetch comments": "Не удаётся загрузить комментарии",
"Afrikaans": "Африкаанс", "View `x` replies": "Показать `x` ответов",
"Albanian": "Албанский", "`x` ago": "`x` назад",
"Amharic": "Амхарский", "Load more": "Загрузить больше",
"Arabic": "Арабский", "`x` points": "`x` очков",
"Armenian": "Армянский", "Could not create mix.": "Не удаётся создать микс.",
"Azerbaijani": "Азербайджанский", "Empty playlist": "Плейлист пуст",
"Bangla": "Бенгальский", "Not a playlist.": "Некорректный плейлист.",
"Basque": "Баскский", "Playlist does not exist.": "Плейлист не существует.",
"Belarusian": "Белорусский", "Could not pull trending pages.": "Не удаётся загрузить страницы «в тренде».",
"Bosnian": "Боснийский", "Hidden field \"challenge\" is a required field": "Необходимо заполнить скрытое поле «challenge»",
"Bulgarian": "Болгарский", "Hidden field \"token\" is a required field": "Необходимо заполнить скрытое поле «токен»",
"Burmese": "Бирманский", "Erroneous challenge": "Неправильный ответ в «challenge»",
"Catalan": "Каталонский", "Erroneous token": "Неправильный токен",
"Cebuano": "Себуанский", "No such user": "Недопустимое имя пользователя",
"Chinese (Simplified)": "Китайский (упрощенный)", "Token is expired, please try again": "Срок действия токена истёк, попробуйте позже",
"Chinese (Traditional)": "Китайский (традиционный)", "English": "Английский",
"Corsican": "Корсиканский", "English (auto-generated)": "Английский (созданы автоматически)",
"Croatian": "Хорватский", "Afrikaans": "Африкаанс",
"Czech": "Чешский", "Albanian": "Албанский",
"Danish": "Датский", "Amharic": "Амхарский",
"Dutch": "Нидерландский", "Arabic": "Арабский",
"Esperanto": "Эсперанто", "Armenian": "Армянский",
"Estonian": "Эстонский", "Azerbaijani": "Азербайджанский",
"Filipino": "Филиппинский", "Bangla": "Бенгальский",
"Finnish": "Финский", "Basque": "Баскский",
"French": "Французский", "Belarusian": "Белорусский",
"Galician": "Галисийский", "Bosnian": "Боснийский",
"Georgian": "Грузинский", "Bulgarian": "Болгарский",
"German": "Немецкий", "Burmese": "Бирманский",
"Greek": "Греческий", "Catalan": "Каталонский",
"Gujarati": "Гуджаратский", "Cebuano": "Себуанский",
"Haitian Creole": "Гаит. креольский", "Chinese (Simplified)": "Китайский (упрощенный)",
"Hausa": "Хауса", "Chinese (Traditional)": "Китайский (традиционный)",
"Hawaiian": "Гавайский", "Corsican": "Корсиканский",
"Hebrew": "Иврит", "Croatian": "Хорватский",
"Hindi": "Хинди", "Czech": "Чешский",
"Hmong": "Хмонг (мяо)", "Danish": "Датский",
"Hungarian": "Венгерский", "Dutch": "Нидерландский",
"Icelandic": "Исландский", "Esperanto": "Эсперанто",
"Igbo": "Игбо", "Estonian": "Эстонский",
"Indonesian": "Индонезийский", "Filipino": "Филиппинский",
"Irish": "Ирландский", "Finnish": "Финский",
"Italian": "Итальянский", "French": "Французский",
"Japanese": "Японский", "Galician": "Галисийский",
"Javanese": "Яванский", "Georgian": "Грузинский",
"Kannada": "Каннада", "German": "Немецкий",
"Kazakh": "Казахский", "Greek": "Греческий",
"Khmer": "Кхмерский", "Gujarati": "Гуджаратский",
"Korean": "Корейский", "Haitian Creole": "Гаит. креольский",
"Kurdish": "Курдский", "Hausa": "Хауса",
"Kyrgyz": "Киргизский", "Hawaiian": "Гавайский",
"Lao": "Лаосский", "Hebrew": "Иврит",
"Latin": "Латинский", "Hindi": "Хинди",
"Latvian": "Латышский", "Hmong": "Хмонг (мяо)",
"Lithuanian": "Литовский", "Hungarian": "Венгерский",
"Luxembourgish": "Люксембургский", "Icelandic": "Исландский",
"Macedonian": "Македонский", "Igbo": "Игбо",
"Malagasy": "Малагасийский", "Indonesian": "Индонезийский",
"Malay": "Малайский", "Irish": "Ирландский",
"Malayalam": "Малаялам", "Italian": "Итальянский",
"Maltese": "Мальтийский", "Japanese": "Японский",
"Maori": "Маори", "Javanese": "Яванский",
"Marathi": "Маратхи", "Kannada": "Каннада",
"Mongolian": "Монгольская", "Kazakh": "Казахский",
"Nepali": "Непальский", "Khmer": "Кхмерский",
"Norwegian": "Норвежский", "Korean": "Корейский",
"Nyanja": "Ньянджа", "Kurdish": "Курдский",
"Pashto": "Пушту", "Kyrgyz": "Киргизский",
"Persian": "Персидский", "Lao": "Лаосский",
"Polish": "Польский", "Latin": "Латинский",
"Portuguese": "Португальский", "Latvian": "Латышский",
"Punjabi": "Панджаби", "Lithuanian": "Литовский",
"Romanian": "Румынский", "Luxembourgish": "Люксембургский",
"Russian": "Русский", "Macedonian": "Македонский",
"Samoan": "Самоанский", "Malagasy": "Малагасийский",
"Scottish Gaelic": "Шотландский (гэльский)", "Malay": "Малайский",
"Serbian": "Сербский", "Malayalam": "Малаялам",
"Shona": "Шона", "Maltese": "Мальтийский",
"Sindhi": "Синдхи", "Maori": "Маори",
"Sinhala": "Сингальский", "Marathi": "Маратхи",
"Slovak": "Словацкий", "Mongolian": "Монгольская",
"Slovenian": "Словенский", "Nepali": "Непальский",
"Somali": "Сомалийский", "Norwegian Bokmål": "Норвежский",
"Southern Sotho": "Сесото (южный сото)", "Nyanja": "Ньянджа",
"Spanish": "Испанский", "Pashto": "Пушту",
"Spanish (Latin America)": "Испанский (Латинская Америка)", "Persian": "Персидский",
"Sundanese": "Сунданский", "Polish": "Польский",
"Swahili": "Суахили", "Portuguese": "Португальский",
"Swedish": "Шведский", "Punjabi": "Панджаби",
"Tajik": "Таджикский", "Romanian": "Румынский",
"Tamil": "Тамильский", "Russian": "Русский",
"Telugu": "Телугу", "Samoan": "Самоанский",
"Thai": "Тайский", "Scottish Gaelic": "Шотландский (гэльский)",
"Turkish": "Турецкий", "Serbian": "Сербский",
"Ukrainian": "Украинский", "Shona": "Шона",
"Urdu": "Урду", "Sindhi": "Синдхи",
"Uzbek": "Узбекский", "Sinhala": "Сингальский",
"Vietnamese": "Вьетнамский", "Slovak": "Словацкий",
"Welsh": "Валлийский", "Slovenian": "Словенский",
"Western Frisian": "Западнофризский", "Somali": "Сомалийский",
"Xhosa": "Коса", "Southern Sotho": "Сесото (южный сото)",
"Yiddish": "Идиш", "Spanish": "Испанский",
"Yoruba": "Йоруба", "Spanish (Latin America)": "Испанский (Латинская Америка)",
"Zulu": "Зулусский", "Sundanese": "Сунданский",
"`x` years": "`x` лет", "Swahili": "Суахили",
"`x` months": "`x` месяцев", "Swedish": "Шведский",
"`x` weeks": "`x` недель", "Tajik": "Таджикский",
"`x` days": "`x` дней", "Tamil": "Тамильский",
"`x` hours": "`x` часов", "Telugu": "Телугу",
"`x` minutes": "`x` минут", "Thai": "Тайский",
"`x` seconds": "`x` секунд", "Turkish": "Турецкий",
"Fallback comments: ": "Резервные комментарии: ", "Ukrainian": "Украинский",
"Popular": "Популярное", "Urdu": "Урду",
"Top": "Топ", "Uzbek": "Узбекский",
"About": "О сайте", "Vietnamese": "Вьетнамский",
"Rating: ": "Рейтинг: ", "Welsh": "Валлийский",
"Language: ": "Язык: ", "Western Frisian": "Западнофризский",
"Default": "По-умолчанию", "Xhosa": "Коса",
"Music": "Музыка", "Yiddish": "Идиш",
"Gaming": "Игры", "Yoruba": "Йоруба",
"News": "Новости", "Zulu": "Зулусский",
"Movies": "Фильмы", "`x` years": "`x` лет",
"Download": "Скачать", "`x` months": "`x` месяцев",
"Download as: ": "Скачать как: ", "`x` weeks": "`x` недель",
"%A %B %-d, %Y": "%-d %B %Y, %A", "`x` days": "`x` дней",
"(edited)": "(изменено)", "`x` hours": "`x` часов",
"Youtube permalink of the comment": "Прямая ссылка на YouTube", "`x` minutes": "`x` минут",
"`x` marked it with a ❤": "❤ от автора канала \"`x`\"", "`x` seconds": "`x` секунд",
"Audio mode": "Аудио режим", "Fallback comments: ": "Резервные комментарии: ",
"Video mode": "Видео режим", "Popular": "Популярное",
"Videos": "Видео", "Top": "Топ",
"Playlists": "Плейлисты", "About": "О сайте",
"Current version: ": "Текущая версия: " "Rating: ": "Рейтинг: ",
} "Language: ": "Язык: ",
"View as playlist": "Смотреть как плейлист",
"Default": "По-умолчанию",
"Music": "Музыка",
"Gaming": "Игры",
"News": "Новости",
"Movies": "Фильмы",
"Download": "Скачать",
"Download as: ": "Скачать как: ",
"%A %B %-d, %Y": "%-d %B %Y, %A",
"(edited)": "(изменено)",
"YouTube comment permalink": "Прямая ссылка на YouTube",
"`x` marked it with a ❤": "❤ от автора канала \"`x`\"",
"Audio mode": "Аудио режим",
"Video mode": "Видео режим",
"Videos": "Видео",
"Playlists": "Плейлисты",
"Current version: ": "Текущая версия: "
}

315
locales/uk.json Normal file
View File

@ -0,0 +1,315 @@
{
"`x` subscribers": "`x` підписників",
"`x` videos": "`x` відео",
"LIVE": "ПРЯМИЙ ЕФІР",
"Shared `x` ago": "Розміщено `x` назад",
"Unsubscribe": "Відписатися",
"Subscribe": "Підписатися",
"View channel on YouTube": "Подивитися канал на YouTube",
"View playlist on YouTube": "",
"newest": "найновіше",
"oldest": "найстаріше",
"popular": "популярне",
"last": "останнє",
"Next page": "Наступна сторінка",
"Previous page": "Попередня сторінка",
"Clear watch history?": "Очистити історію переглядів?",
"New password": "Новий пароль",
"New passwords must match": "Нові паролі не співпадають",
"Cannot change password for Google accounts": "Змінити пароль обліківки Google неможливо",
"Authorize token?": "Авторизувати токен?",
"Authorize token for `x`?": "Авторизувати токен для `x`?",
"Yes": "Так",
"No": "Ні",
"Import and Export Data": "Імпорт і експорт даних",
"Import": "Імпорт",
"Import Invidious data": "Імпортувати дані Invidious",
"Import YouTube subscriptions": "Імпортувати підписки з YouTube",
"Import FreeTube subscriptions (.db)": "Імпортувати підписки з FreeTube (.db)",
"Import NewPipe subscriptions (.json)": "Імпортувати підписки з NewPipe (.json)",
"Import NewPipe data (.zip)": "Імпортувати дані з NewPipe (.zip)",
"Export": "Експорт",
"Export subscriptions as OPML": "Експортувати підписки у форматі OPML",
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Експортувати підписки у форматі OPML (для NewPipe та FreeTube)",
"Export data as JSON": "Експортувати дані у форматі JSON",
"Delete account?": "Видалити обліківку?",
"History": "Історія",
"An alternative front-end to YouTube": "Альтернативний фронтенд до YouTube",
"JavaScript license information": "Інформація щодо ліцензій JavaScript",
"source": "джерело",
"Log in": "Увійти",
"Log in/register": "Увійти або зареєструватися",
"Log in with Google": "Увійти через Google",
"User ID": "ID користувача",
"Password": "Пароль",
"Time (h:mm:ss):": "Час (г:мм:сс):",
"Text CAPTCHA": "Текст капчі",
"Image CAPTCHA": "Зображення капчі",
"Sign In": "Увійти",
"Register": "Зареєструватися",
"E-mail": "Електронна пошта",
"Google verification code": "Код підтвердження Google",
"Preferences": "Налаштування",
"Player preferences": "Налаштування програвача",
"Always loop: ": "Завжди повторювати: ",
"Autoplay: ": "Автовідтворення: ",
"Play next by default: ": "Завжди вмикати наступне відео: ",
"Autoplay next video: ": "Автовідтворення наступного відео: ",
"Listen by default: ": "Режим «тільки звук» як усталений: ",
"Proxy videos? ": "Програвати відео через проксі? ",
"Default speed: ": "Усталена швидкість відео: ",
"Preferred video quality: ": "Пріорітетна якість відео: ",
"Player volume: ": "Гучність відео: ",
"Default comments: ": "Джерело коментарів: ",
"youtube": "YouTube",
"reddit": "Reddit",
"Default captions: ": "Основна мова субтитрів: ",
"Fallback captions: ": "Запасна мова субтитрів: ",
"Show related videos? ": "Показувати схожі відео? ",
"Show annotations by default? ": "Завжди показувати анотації? ",
"Visual preferences": "Налаштування сайту",
"Dark mode: ": "Темне оформлення: ",
"Thin mode: ": "Полегшене оформлення: ",
"Subscription preferences": "Налаштування підписок",
"Show annotations by default for subscribed channels? ": "Завжди показувати анотації у відео каналів, на які ви підписані? ",
"Redirect homepage to feed: ": "Показувати відео з каналів, на які підписані, як головну сторінку: ",
"Number of videos shown in feed: ": "Кількість відео з каналів, на які підписані, у потоці: ",
"Sort videos by: ": "Сортувати відео: ",
"published": "за датою розміщення",
"published - reverse": "за датою розміщення в зворотному порядку",
"alphabetically": "за абеткою",
"alphabetically - reverse": "за абеткою в зворотному порядку",
"channel name": "за назвою каналу",
"channel name - reverse": "за назвою каналу в зворотному порядку",
"Only show latest video from channel: ": "Показувати тільки останнє відео з каналів: ",
"Only show latest unwatched video from channel: ": "Показувати тільки непереглянуті відео з каналів: ",
"Only show unwatched: ": "Показувати тільки непереглянуті відео: ",
"Only show notifications (if there are any): ": "Показувати лише сповіщення, якщо вони є: ",
"Data preferences": "Налаштування даних",
"Clear watch history": "Очистити історію переглядів",
"Import/export data": "Імпорт і експорт даних",
"Change password": "Змінити пароль",
"Manage subscriptions": "Керування підписками",
"Manage tokens": "Керувати токенами",
"Watch history": "Історія переглядів",
"Delete account": "Видалити обліківку",
"Administrator preferences": "Адміністраторські налаштування",
"Default homepage: ": "Усталена домашня сторінка: ",
"Feed menu: ": "Меню потоку з відео: ",
"Top enabled? ": "Увімкнути топ відео? ",
"CAPTCHA enabled? ": "Увімкнути капчу? ",
"Login enabled? ": "Увімкнути авторизацію? ",
"Registration enabled? ": "Увімкнути реєстрацію? ",
"Report statistics? ": "Повідомляти статистику? ",
"Save preferences": "Зберегти налаштування",
"Subscription manager": "Менеджер підписок",
"Token manager": "Менеджер токенів",
"Token": "Токен",
"`x` subscriptions": "`x` підписка / підписок / підписки",
"`x` tokens": "`x` токенів",
"Import/export": "Імпорт і експорт",
"unsubscribe": "відписатися",
"revoke": "скасувати",
"Subscriptions": "Підписки",
"`x` unseen notifications": "`x` непереглянуте сповіщення / непереглянутих сповіщень / непереглянутих сповіщення",
"search": "пошук",
"Log out": "Вийти",
"Released under the AGPLv3 by Omar Roth.": "Реалізовано Омаром Ротом за ліцензією AGPLv3.",
"Source available here.": "Програмний код доступний тут.",
"View JavaScript license information.": "Переглянути інформацію щодо ліцензії JavaScript.",
"View privacy policy.": "Переглянути політику приватності.",
"Trending": "У тренді",
"Unlisted": "Немає в списку",
"Watch on YouTube": "Дивитися на YouTube",
"Hide annotations": "Приховати анотації",
"Show annotations": "Показати анотації",
"Genre: ": "Жанр: ",
"License: ": "Ліцензія: ",
"Family friendly? ": "Перегляд із родиною? ",
"Wilson score: ": "Рейтинг Вілсона: ",
"Engagement: ": "Залученість: ",
"Whitelisted regions: ": "Доступно у регіонах: ",
"Blacklisted regions: ": "Недоступно у регіонах: ",
"Shared `x`": "Розміщено `x`",
"`x` views": "`x` переглядів",
"Premieres in `x`": "Прем’єра через `x`",
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Схоже, у вас відключений JavaScript. Щоб побачити коментарі, натисніть сюда, але майте на увазі, що вони можуть завантажуватися трохи довше.",
"View YouTube comments": "Переглянути коментарі з YouTube",
"View more comments on Reddit": "Переглянути більше коментарів на Reddit",
"View `x` comments": "Переглянути `x` коментар / коментарів / коментаря",
"View Reddit comments": "Переглянути коментарі з Reddit",
"Hide replies": "Сховати відповіді",
"Show replies": "Показати відповіді",
"Incorrect password": "Неправильний пароль",
"Quota exceeded, try again in a few hours": "Ліміт перевищено, спробуйте знову за декілька годин",
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Не вдається увійти. Перевірте, чи не ввімкнена двофакторна аутентифікація (за кодом чи смс).",
"Invalid TFA code": "Неправильний код двофакторної аутентифікації",
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Не вдається увійти. Це може бути через те, що у вашій обліківці не ввімкнена двофакторна аутентифікація.",
"Wrong answer": "Неправильна відповідь",
"Erroneous CAPTCHA": "Неправильна капча",
"CAPTCHA is a required field": "Необхідно пройти капчу",
"User ID is a required field": "Необхідно ввести ID користувача",
"Password is a required field": "Необхідно ввести пароль",
"Wrong username or password": "Неправильний логін чи пароль",
"Please sign in using 'Log in with Google'": "Будь ласка, натисніть «Увійти через Google»",
"Password cannot be empty": "Пароль не може бути порожнім",
"Password cannot be longer than 55 characters": "Пароль не може бути довшим за 55 знаків",
"Please log in": "Будь ласка, увійдіть",
"Invidious Private Feed for `x`": "Приватний поток відео Invidious для `x`",
"channel:`x`": "канал: `x`",
"Deleted or invalid channel": "Канал видалено або не знайдено",
"This channel does not exist.": "Такого каналу не існує.",
"Could not get channel info.": "Не вдається отримати інформацію щодо цього каналу.",
"Could not fetch comments": "Не вдається завантажити коментарі",
"View `x` replies": "Переглянути `x` відповідь / відповідей / відповіді",
"`x` ago": "`x` тому",
"Load more": "Завантажити більше",
"`x` points": "`x` очко / очок / очка",
"Could not create mix.": "Не вдається створити мікс.",
"Empty playlist": "Плейлист порожній",
"Not a playlist.": "Недійсний плейлист.",
"Playlist does not exist.": "Плейлист не існує.",
"Could not pull trending pages.": "Не вдається завантажити сторінки «у тренді».",
"Hidden field \"challenge\" is a required field": "Необхідно заповнити приховане поле «challenge»",
"Hidden field \"token\" is a required field": "Необхідно заповнити приховане поле «token»",
"Erroneous challenge": "Неправильна відповідь у «challenge»",
"Erroneous token": "Недійсний токен",
"No such user": "Недопустиме ім’я користувача",
"Token is expired, please try again": "Термін дії токена закінчився, спробуйте пізніше",
"English": "Англійська",
"English (auto-generated)": "Англійська (сгенеровано автоматично)",
"Afrikaans": "Африкаанс",
"Albanian": "Албанська",
"Amharic": "Амхарська",
"Arabic": "Арабська",
"Armenian": "Вірменська",
"Azerbaijani": "Азербайджанська",
"Bangla": "Бенгальска",
"Basque": "Баскська",
"Belarusian": "Білоруська",
"Bosnian": "Боснійська",
"Bulgarian": "Болгарська",
"Burmese": "Бірманська",
"Catalan": "Каталонська",
"Cebuano": "Себуанська",
"Chinese (Simplified)": "Китайська (спрощена)",
"Chinese (Traditional)": "Китайська (традиційна)",
"Corsican": "Корсиканська",
"Croatian": "Хорватська",
"Czech": "Чеська",
"Danish": "Данська",
"Dutch": "Нідерландська",
"Esperanto": "Есперанто",
"Estonian": "Естонська",
"Filipino": "Філіппінська",
"Finnish": "Фінська",
"French": "Французька",
"Galician": "Галісійська",
"Georgian": "Грузинська",
"German": "Німецька",
"Greek": "Грецька",
"Gujarati": "Гуджаратська",
"Haitian Creole": "Гаїтянська креольська",
"Hausa": "Хауса",
"Hawaiian": "Гавайська",
"Hebrew": "Іврит",
"Hindi": "Гінді",
"Hmong": "Хмонгська",
"Hungarian": "Угорська",
"Icelandic": "Ісландська",
"Igbo": "Ігбо",
"Indonesian": "Індонезійська",
"Irish": "Ірландська",
"Italian": "Італійська",
"Japanese": "Японська",
"Javanese": "Яванська",
"Kannada": "Каннада",
"Kazakh": "Казахська",
"Khmer": "Кхмерська",
"Korean": "Корейська",
"Kurdish": "Курдська",
"Kyrgyz": "Киргизька",
"Lao": "Лаоська",
"Latin": "Латинська",
"Latvian": "Латиська",
"Lithuanian": "Литовська",
"Luxembourgish": "Люксембурзька",
"Macedonian": "Македонська",
"Malagasy": "Малагасійська",
"Malay": "Малайська",
"Malayalam": "Малаялам",
"Maltese": "Мальтійська",
"Maori": "Маорі",
"Marathi": "Маратхі",
"Mongolian": "Монгольська",
"Nepali": "Непальська",
"Norwegian Bokmål": "Норвезька",
"Nyanja": "Ньянджа",
"Pashto": "Пушту",
"Persian": "Перська",
"Polish": "Польська",
"Portuguese": "Португальська",
"Punjabi": "Пенджабська",
"Romanian": "Румунська",
"Russian": "Російська",
"Samoan": "Самоанська",
"Scottish Gaelic": "Шотландська ґельська",
"Serbian": "Сербська",
"Shona": "Шона",
"Sindhi": "Сіндгі",
"Sinhala": "Сингальська",
"Slovak": "Словацька",
"Slovenian": "Словенська",
"Somali": "Сомалійська",
"Southern Sotho": "Сесото (південна сото)",
"Spanish": "Іспанська",
"Spanish (Latin America)": "Испанська (Латинська Америка)",
"Sundanese": "Сунданська",
"Swahili": "Суахілі",
"Swedish": "Шведська",
"Tajik": "Таджицька",
"Tamil": "Тамільська",
"Telugu": "Телугу",
"Thai": "Тайська",
"Turkish": "Турецька",
"Ukrainian": "Українська",
"Urdu": "Урду",
"Uzbek": "Узбецька",
"Vietnamese": "В’єтнамська",
"Welsh": "Валлійська",
"Western Frisian": "Західнофризька",
"Xhosa": "Коса",
"Yiddish": "Їдиш",
"Yoruba": "Йоруба",
"Zulu": "Зулу",
"`x` years": "`x` років",
"`x` months": "`x` місяців",
"`x` weeks": "`x` тижнів",
"`x` days": "`x` днів",
"`x` hours": "`x` годин",
"`x` minutes": "`x` хвилин",
"`x` seconds": "`x` секунд",
"Fallback comments: ": "Резервні коментарі: ",
"Popular": "Популярне",
"Top": "Топ",
"About": "Про сайт",
"Rating: ": "Рейтинг: ",
"Language: ": "Мова: ",
"View as playlist": "Дивитися як плейлист",
"Default": "Усталено",
"Music": "Музика",
"Gaming": "Ігри",
"News": "Новини",
"Movies": "Фільми",
"Download": "Завантажити",
"Download as: ": "Завантажити як: ",
"%A %B %-d, %Y": "%-d %B %Y, %A",
"(edited)": "(змінено)",
"YouTube comment permalink": "Пряме посилання на коментар в YouTube",
"`x` marked it with a ❤": "❤ цьому від каналу `x`",
"Audio mode": "Аудіорежим",
"Video mode": "Відеорежим",
"Videos": "Відео",
"Playlists": "Плейлисти",
"Current version: ": "Поточна версія: "
}

View File

@ -1,5 +1,5 @@
name: invidious name: invidious
version: 0.15.0 version: 0.17.0
authors: authors:
- Omar Roth <omarroth@protonmail.com> - Omar Roth <omarroth@protonmail.com>
@ -16,6 +16,6 @@ dependencies:
sqlite3: sqlite3:
github: crystal-lang/crystal-sqlite3 github: crystal-lang/crystal-sqlite3
crystal: 0.27.2 crystal: 0.28.0
license: AGPLv3 license: AGPLv3

View File

@ -1,13 +1,16 @@
require "kemal" require "kemal"
require "openssl/hmac"
require "pg" require "pg"
require "spec" require "spec"
require "yaml" require "yaml"
require "../src/invidious/helpers/*" require "../src/invidious/helpers/*"
require "../src/invidious/channels" require "../src/invidious/channels"
require "../src/invidious/comments"
require "../src/invidious/playlists" require "../src/invidious/playlists"
require "../src/invidious/search" require "../src/invidious/search"
require "../src/invidious/users"
describe "Helpers" do describe "Helper" do
describe "#produce_channel_videos_url" do describe "#produce_channel_videos_url" do
it "correctly produces url for requesting page `x` of a channel's videos" do it "correctly produces url for requesting page `x` of a channel's videos" do
produce_channel_videos_url(ucid: "UCXuqSBlHAE6Xw-yeJA0Tunw").should eq("/browse_ajax?continuation=4qmFsgI8EhhVQ1h1cVNCbEhBRTZYdy15ZUpBMFR1bncaIEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQUFlZ0V4&gl=US&hl=en") produce_channel_videos_url(ucid: "UCXuqSBlHAE6Xw-yeJA0Tunw").should eq("/browse_ajax?continuation=4qmFsgI8EhhVQ1h1cVNCbEhBRTZYdy15ZUpBMFR1bncaIEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQUFlZ0V4&gl=US&hl=en")
@ -16,9 +19,7 @@ describe "Helpers" do
produce_channel_videos_url(ucid: "UCXuqSBlHAE6Xw-yeJA0Tunw", page: 20).should eq("/browse_ajax?continuation=4qmFsgJEEhhVQ1h1cVNCbEhBRTZYdy15ZUpBMFR1bncaKEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQUFlZ0l5TUElM0QlM0Q%3D&gl=US&hl=en") produce_channel_videos_url(ucid: "UCXuqSBlHAE6Xw-yeJA0Tunw", page: 20).should eq("/browse_ajax?continuation=4qmFsgJEEhhVQ1h1cVNCbEhBRTZYdy15ZUpBMFR1bncaKEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQUFlZ0l5TUElM0QlM0Q%3D&gl=US&hl=en")
produce_channel_videos_url(ucid: "UC-9-kyTW8ZkZNDHQJ6FgpwQ", auto_generated: true).should eq("/browse_ajax?continuation=4qmFsgJIEhhVQy05LWt5VFc4WmtaTkRIUUo2Rmdwd1EaLEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQTJlZ294TlRVeU1ESXlPVFE1&gl=US&hl=en") produce_channel_videos_url(ucid: "UC-9-kyTW8ZkZNDHQJ6FgpwQ", page: 20, sort_by: "popular").should eq("/browse_ajax?continuation=4qmFsgJAEhhVQy05LWt5VFc4WmtaTkRIUUo2Rmdwd1EaJEVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQUFlZ0l5TUJnQg%3D%3D&gl=US&hl=en")
produce_channel_videos_url(ucid: "UC-9-kyTW8ZkZNDHQJ6FgpwQ", page: 20, auto_generated: true, sort_by: "popular").should eq("/browse_ajax?continuation=4qmFsgJOEhhVQy05LWt5VFc4WmtaTkRIUUo2Rmdwd1EaMkVnWjJhV1JsYjNNd0FqZ0JZQUZxQUxnQkFDQTJlZ294TlRBeU1UY3dNVFE1R0FFJTNE&gl=US&hl=en")
end end
end end
@ -59,4 +60,49 @@ describe "Helpers" do
produce_search_params(content_type: "channel").should eq("CAASAhAC") produce_search_params(content_type: "channel").should eq("CAASAhAC")
end end
end end
describe "#produce_comment_continuation" do
it "correctly produces a continuation token for comments" do
produce_comment_continuation("_cE8xSu6swE", "ADSJ_i2qvJeFtL0htmS5_K5Ctj3eGFVBMWL9Wd42o3kmUL6_mAzdLp85-liQZL0mYr_16BhaggUqX652Sv9JqV6VXinShSP-ZT6rL4NolPBaPXVtJsO5_rA_qE3GubAuLFw9uzIIXU2-HnpXbdgPLWTFavfX206hqWmmpHwUOrmxQV_OX6tYkM3ux3rPAKCDrT8eWL7MU3bLiNcnbgkW8o0h8KYLL_8BPa8LcHbTv8pAoNkjerlX1x7K4pqxaXPoyz89qNlnh6rRx6AXgAzzoHH1dmcyQ8CIBeOHg-m4i8ZxdX4dP88XWrIFg-jJGhpGP8JUMDgZgavxVx225hUEYZMyrLGler5em4FgbG62YWC51moLDLeYEA").should eq("EiYSC19jRTh4U3U2c3dFwAEByAEB4AEBogINKP___________wFAABgGMowDCvYCQURTSl9pMnF2SmVGdEwwaHRtUzVfSzVDdGozZUdGVkJNV0w5V2Q0Mm8za21VTDZfbUF6ZExwODUtbGlRWkwwbVlyXzE2QmhhZ2dVcVg2NTJTdjlKcVY2VlhpblNoU1AtWlQ2ckw0Tm9sUEJhUFhWdEpzTzVfckFfcUUzR3ViQXVMRnc5dXpJSVhVMi1IbnBYYmRnUExXVEZhdmZYMjA2aHFXbW1wSHdVT3JteFFWX09YNnRZa00zdXgzclBBS0NEclQ4ZVdMN01VM2JMaU5jbmJna1c4bzBoOEtZTExfOEJQYThMY0hiVHY4cEFvTmtqZXJsWDF4N0s0cHF4YVhQb3l6ODlxTmxuaDZyUng2QVhnQXp6b0hIMWRtY3lROENJQmVPSGctbTRpOFp4ZFg0ZFA4OFhXcklGZy1qSkdocEdQOEpVTURnWmdhdnhWeDIyNWhVRVlaTXlyTEdsZXI1ZW00RmdiRzYyWVdDNTFtb0xETGVZRUEiDyILX2NFOHhTdTZzd0UwACgU")
produce_comment_continuation("_cE8xSu6swE", "ADSJ_i1yz21HI4xrtsYXVC-2_kfZ6kx1yjYQumXAAxqH3CAd7ZxKxfLdZS1__fqhCtOASRbbpSBGH_tH1J96Dxux-Qfjk-lUbupMqv08Q3aHzGu7p70VoUMHhI2-GoJpnbpmcOxkGzeIuenRS_ym2Y8fkDowhqLPFgsS0n4djnZ2UmC17F3Ch3N1S1UYf1ZVOc991qOC1iW9kJDzyvRQTWCPsJUPneSaAKW-Rr97pdesOkR4i8cNvHZRnQKe2HEfsvlJOb2C3lF1dJBfJeNfnQYeh5hv6_fZN7bt3-JL1Xk3Qc9NXNxmmbDpwAC_yFR8dthFfUJdyIO9Nu1D79MLYeR-H5HxqUJokkJiGIz4lTE_CXXbhAI").should eq("EiYSC19jRTh4U3U2c3dFwAEByAEB4AEBogINKP___________wFAABgGMokDCvMCQURTSl9pMXl6MjFISTR4cnRzWVhWQy0yX2tmWjZreDF5allRdW1YQUF4cUgzQ0FkN1p4S3hmTGRaUzFfX2ZxaEN0T0FTUmJicFNCR0hfdEgxSjk2RHh1eC1RZmprLWxVYnVwTXF2MDhRM2FIekd1N3A3MFZvVU1IaEkyLUdvSnBuYnBtY094a0d6ZUl1ZW5SU195bTJZOGZrRG93aHFMUEZnc1MwbjRkam5aMlVtQzE3RjNDaDNOMVMxVVlmMVpWT2M5OTFxT0MxaVc5a0pEenl2UlFUV0NQc0pVUG5lU2FBS1ctUnI5N3BkZXNPa1I0aThjTnZIWlJuUUtlMkhFZnN2bEpPYjJDM2xGMWRKQmZKZU5mblFZZWg1aHY2X2ZaTjdidDMtSkwxWGszUWM5TlhOeG1tYkRwd0FDX3lGUjhkdGhGZlVKZHlJTzlOdTFENzlNTFllUi1INUh4cVVKb2trSmlHSXo0bFRFX0NYWGJoQUkiDyILX2NFOHhTdTZzd0UwACgU")
produce_comment_continuation("29-q7YnyUmY", "").should eq("EiYSCzI5LXE3WW55VW1ZwAEByAEB4AEBogINKP___________wFAABgGMhMiDyILMjktcTdZbnlVbVkwAHgC")
produce_comment_continuation("CvFH_6DNRCY", "").should eq("EiYSC0N2RkhfNkROUkNZwAEByAEB4AEBogINKP___________wFAABgGMhMiDyILQ3ZGSF82RE5SQ1kwAHgC")
end
end
describe "#produce_comment_reply_continuation" do
it "correctly produces a continuation token for replies to a given comment" do
produce_comment_reply_continuation("cIHQWOoJeag", "UCq6VFHwMzcMXbuKyG7SQYIg", "Ugx1IP_wGVv3WtGWcdV4AaABAg").should eq("EiYSC2NJSFFXT29KZWFnwAEByAEB4AEBogINKP___________wFAABgGMk0aSxIaVWd4MUlQX3dHVnYzV3RHV2NkVjRBYUFCQWciAggAKhhVQ3E2VkZId016Y01YYnVLeUc3U1FZSWcyC2NJSFFXT29KZWFnQAFICg%3D%3D")
produce_comment_reply_continuation("cIHQWOoJeag", "UCq6VFHwMzcMXbuKyG7SQYIg", "Ugza62y_TlmTu9o2RfF4AaABAg").should eq("EiYSC2NJSFFXT29KZWFnwAEByAEB4AEBogINKP___________wFAABgGMk0aSxIaVWd6YTYyeV9UbG1UdTlvMlJmRjRBYUFCQWciAggAKhhVQ3E2VkZId016Y01YYnVLeUc3U1FZSWcyC2NJSFFXT29KZWFnQAFICg%3D%3D")
produce_comment_reply_continuation("_cE8xSu6swE", "UC1AZY74-dGVPe6bfxFwwEMg", "UgyBUaRGHB9Jmt1dsUZ4AaABAg").should eq("EiYSC19jRTh4U3U2c3dFwAEByAEB4AEBogINKP___________wFAABgGMk0aSxIaVWd5QlVhUkdIQjlKbXQxZHNVWjRBYUFCQWciAggAKhhVQzFBWlk3NC1kR1ZQZTZiZnhGd3dFTWcyC19jRTh4U3U2c3dFQAFICg%3D%3D")
end
end
describe "#sign_token" do
it "correctly signs a given hash" do
token = {
"session" => "v1:AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA",
"expires" => 1554680038,
"scopes" => [
":notifications",
":subscriptions/*",
"GET:tokens*",
],
"signature" => "f__2hS20th8pALF305PJFK-D2aVtvefNnQheILHD2vU=",
}
sign_token("SECRET_KEY", token).should eq(token["signature"])
token = {
"session" => "v1:AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA",
"scopes" => [":notifications", "POST:subscriptions/*"],
"signature" => "fNvXoT0MRAL9eE6lTE33CEg8HitYJDOL9a22rSN2Ihg=",
}
sign_token("SECRET_KEY", token).should eq(token["signature"])
end
end
end end

File diff suppressed because it is too large Load Diff

View File

@ -1,5 +1,5 @@
struct InvidiousChannel struct InvidiousChannel
add_mapping({ db_mapping({
id: String, id: String,
author: String, author: String,
updated: Time, updated: Time,
@ -9,7 +9,37 @@ struct InvidiousChannel
end end
struct ChannelVideo struct ChannelVideo
add_mapping({ def to_json(locale, config, kemal_config, json : JSON::Builder)
json.object do
json.field "title", self.title
json.field "videoId", self.id
json.field "videoThumbnails" do
generate_thumbnails(json, self.id, config, Kemal.config)
end
json.field "lengthSeconds", self.length_seconds
json.field "author", self.author
json.field "authorId", self.ucid
json.field "authorUrl", "/channel/#{self.ucid}"
json.field "published", self.published.to_unix
json.field "publishedText", translate(locale, "`x` ago", recode_date(self.published, locale))
json.field "viewCount", self.views
end
end
def to_json(locale, config, kemal_config, json : JSON::Builder | Nil = nil)
if json
to_json(locale, config, kemal_config, json)
else
JSON.build do |json|
to_json(locale, config, kemal_config, json)
end
end
end
db_mapping({
id: String, id: String,
title: String, title: String,
published: Time, published: Time,
@ -19,30 +49,41 @@ struct ChannelVideo
length_seconds: {type: Int32, default: 0}, length_seconds: {type: Int32, default: 0},
live_now: {type: Bool, default: false}, live_now: {type: Bool, default: false},
premiere_timestamp: {type: Time?, default: nil}, premiere_timestamp: {type: Time?, default: nil},
views: {type: Int64?, default: nil},
}) })
end end
def get_batch_channels(channels, db, refresh = false, pull_all_videos = true, max_threads = 10) def get_batch_channels(channels, db, refresh = false, pull_all_videos = true, max_threads = 10)
active_threads = 0 finished_channel = Channel(String | Nil).new
active_channel = Channel(String | Nil).new
final = [] of String spawn do
channels.map do |ucid| active_threads = 0
if active_threads >= max_threads active_channel = Channel(Nil).new
if response = active_channel.receive
channels.each do |ucid|
if active_threads >= max_threads
active_channel.receive
active_threads -= 1 active_threads -= 1
final << response end
active_threads += 1
spawn do
begin
get_channel(ucid, db, refresh, pull_all_videos)
finished_channel.send(ucid)
rescue ex
finished_channel.send(nil)
ensure
active_channel.send(nil)
end
end end
end end
end
active_threads += 1 final = [] of String
spawn do channels.size.times do
begin if ucid = finished_channel.receive
get_channel(ucid, db, refresh, pull_all_videos) final << ucid
active_channel.send(ucid)
rescue ex
active_channel.send(nil)
end
end end
end end
@ -91,71 +132,84 @@ def fetch_channel(ucid, db, pull_all_videos = true, locale = nil)
auto_generated = true auto_generated = true
end end
if !pull_all_videos page = 1
url = produce_channel_videos_url(ucid, 1, auto_generated: auto_generated)
response = client.get(url)
json = JSON.parse(response.body)
if json["content_html"]? && !json["content_html"].as_s.empty? url = produce_channel_videos_url(ucid, page, auto_generated: auto_generated)
document = XML.parse_html(json["content_html"].as_s) response = client.get(url)
nodeset = document.xpath_nodes(%q(//li[contains(@class, "feed-item-container")])) json = JSON.parse(response.body)
if auto_generated if json["content_html"]? && !json["content_html"].as_s.empty?
videos = extract_videos(nodeset) document = XML.parse_html(json["content_html"].as_s)
else nodeset = document.xpath_nodes(%q(//li[contains(@class, "feed-item-container")]))
videos = extract_videos(nodeset, ucid)
videos.each { |video| video.ucid = ucid } if auto_generated
videos.each { |video| video.author = author } videos = extract_videos(nodeset)
end else
videos = extract_videos(nodeset, ucid, author)
end end
end
videos ||= [] of ChannelVideo videos ||= [] of ChannelVideo
rss.xpath_nodes("//feed/entry").each do |entry| rss.xpath_nodes("//feed/entry").each do |entry|
video_id = entry.xpath_node("videoid").not_nil!.content video_id = entry.xpath_node("videoid").not_nil!.content
title = entry.xpath_node("title").not_nil!.content title = entry.xpath_node("title").not_nil!.content
published = Time.parse_rfc3339(entry.xpath_node("published").not_nil!.content) published = Time.parse_rfc3339(entry.xpath_node("published").not_nil!.content)
updated = Time.parse_rfc3339(entry.xpath_node("updated").not_nil!.content) updated = Time.parse_rfc3339(entry.xpath_node("updated").not_nil!.content)
author = entry.xpath_node("author/name").not_nil!.content author = entry.xpath_node("author/name").not_nil!.content
ucid = entry.xpath_node("channelid").not_nil!.content ucid = entry.xpath_node("channelid").not_nil!.content
views = entry.xpath_node("group/community/statistics").try &.["views"]?.try &.to_i64?
views ||= 0_i64
channel_video = videos.select { |video| video.id == video_id }[0]? channel_video = videos.select { |video| video.id == video_id }[0]?
length_seconds = channel_video.try &.length_seconds length_seconds = channel_video.try &.length_seconds
length_seconds ||= 0 length_seconds ||= 0
live_now = channel_video.try &.live_now live_now = channel_video.try &.live_now
live_now ||= false live_now ||= false
premiere_timestamp = channel_video.try &.premiere_timestamp premiere_timestamp = channel_video.try &.premiere_timestamp
video = ChannelVideo.new( video = ChannelVideo.new(
video_id, id: video_id,
title, title: title,
published, published: published,
Time.now, updated: Time.now,
ucid, ucid: ucid,
author, author: author,
length_seconds, length_seconds: length_seconds,
live_now, live_now: live_now,
premiere_timestamp premiere_timestamp: premiere_timestamp,
) views: views,
)
db.exec("UPDATE users SET notifications = notifications || $1 \ users = db.query_all("UPDATE users SET notifications = notifications || $1 \
WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications)", video.id, video.published, ucid) WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications) RETURNING email",
video.id, video.published, ucid, as: String)
video_array = video.to_a video_array = video.to_a
args = arg_array(video_array) args = arg_array(video_array)
# We don't include the 'premire_timestamp' here because channel pages don't include them, # We don't include the 'premiere_timestamp' here because channel pages don't include them,
# meaning the above timestamp is always null # meaning the above timestamp is always null
db.exec("INSERT INTO channel_videos VALUES (#{args}) \ db.exec("INSERT INTO channel_videos VALUES (#{args}) \
ON CONFLICT (id) DO UPDATE SET title = $2, published = $3, \ ON CONFLICT (id) DO UPDATE SET title = $2, published = $3, \
updated = $4, ucid = $5, author = $6, length_seconds = $7, \ updated = $4, ucid = $5, author = $6, length_seconds = $7, \
live_now = $8", video_array) live_now = $8, views = $10", video_array)
users.each do |user|
payload = {
"email" => user,
"action" => "refresh",
}.to_json
PG_DB.exec("NOTIFY feeds, E'#{payload}'")
end end
else end
page = 1
if pull_all_videos
page += 1
ids = [] of String ids = [] of String
loop do loop do
@ -170,48 +224,59 @@ def fetch_channel(ucid, db, pull_all_videos = true, locale = nil)
break break
end end
nodeset = nodeset.not_nil!
if auto_generated if auto_generated
videos = extract_videos(nodeset) videos = extract_videos(nodeset)
else else
videos = extract_videos(nodeset, ucid) videos = extract_videos(nodeset, ucid, author)
videos.each { |video| video.ucid = ucid }
videos.each { |video| video.author = author }
end end
count = nodeset.size count = nodeset.size
videos = videos.map { |video| ChannelVideo.new( videos = videos.map { |video| ChannelVideo.new(
video.id, id: video.id,
video.title, title: video.title,
video.published, published: video.published,
Time.now, updated: Time.now,
video.ucid, ucid: video.ucid,
video.author, author: video.author,
video.length_seconds, length_seconds: video.length_seconds,
video.live_now, live_now: video.live_now,
video.premiere_timestamp premiere_timestamp: video.premiere_timestamp,
views: video.views
) } ) }
videos.each do |video| videos.each do |video|
ids << video.id ids << video.id
# FIXME: Red videos don't provide published date, so the best we can do is ignore them # We are notified of Red videos elsewhere (PubSub), which includes a correct published date,
# so since they don't provide a published date here we can safely ignore them.
if Time.now - video.published > 1.minute if Time.now - video.published > 1.minute
db.exec("UPDATE users SET notifications = notifications || $1 \ users = db.query_all("UPDATE users SET notifications = notifications || $1 \
WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications)", video.id, video.published, video.ucid) WHERE updated < $2 AND $3 = ANY(subscriptions) AND $1 <> ALL(notifications) RETURNING email",
video.id, video.published, video.ucid, as: String)
video_array = video.to_a video_array = video.to_a
args = arg_array(video_array) args = arg_array(video_array)
# We don't include the 'premire_timestamp' here because channel pages don't include them, # We don't update the 'premire_timestamp' here because channel pages don't include them
# meaning the above timestamp is always null
db.exec("INSERT INTO channel_videos VALUES (#{args}) \ db.exec("INSERT INTO channel_videos VALUES (#{args}) \
ON CONFLICT (id) DO UPDATE SET title = $2, published = $3, \ ON CONFLICT (id) DO UPDATE SET title = $2, published = $3, \
updated = $4, ucid = $5, author = $6, length_seconds = $7, \ updated = $4, ucid = $5, author = $6, length_seconds = $7, \
live_now = $8", video_array) live_now = $8, views = $10", video_array)
# Update all users affected by insert
users.each do |user|
payload = {
"email" => user,
"action" => "refresh",
}.to_json
PG_DB.exec("NOTIFY feeds, E'#{payload}'")
end
end end
end end
if count < 30 if count < 25
break break
end end

View File

@ -56,15 +56,14 @@ class RedditListing
}) })
end end
def fetch_youtube_comments(id, continuation, proxies, format, locale, thin_mode, region) def fetch_youtube_comments(id, continuation, proxies, format, locale, thin_mode, region, sort_by = "top")
video = fetch_video(id, proxies, region: region) video = fetch_video(id, proxies, region: region)
session_token = video.info["session_token"]? session_token = video.info["session_token"]?
itct = video.info["itct"]?
ctoken = video.info["ctoken"]? ctoken = produce_comment_continuation(id, cursor: "", sort_by: sort_by)
continuation ||= ctoken continuation ||= ctoken
if !continuation || !itct || !session_token if !continuation || !session_token
if format == "json" if format == "json"
return {"comments" => [] of String}.to_json return {"comments" => [] of String}.to_json
else else
@ -73,7 +72,7 @@ def fetch_youtube_comments(id, continuation, proxies, format, locale, thin_mode,
end end
post_req = { post_req = {
"session_token" => session_token.not_nil!, "session_token" => session_token,
} }
post_req = HTTP::Params.encode(post_req) post_req = HTTP::Params.encode(post_req)
@ -90,7 +89,7 @@ def fetch_youtube_comments(id, continuation, proxies, format, locale, thin_mode,
headers["x-youtube-client-name"] = "1" headers["x-youtube-client-name"] = "1"
headers["x-youtube-client-version"] = "2.20180719" headers["x-youtube-client-version"] = "2.20180719"
response = client.post("/comment_service_ajax?action_get_comments=1&pbj=1&ctoken=#{continuation}&continuation=#{continuation}&itct=#{itct}&hl=en&gl=US", headers, post_req) response = client.post("/comment_service_ajax?action_get_comments=1&ctoken=#{continuation}&continuation=#{continuation}&hl=en&gl=US", headers, post_req)
response = JSON.parse(response.body) response = JSON.parse(response.body)
if !response["response"]["continuationContents"]? if !response["response"]["continuationContents"]?
@ -250,7 +249,7 @@ def fetch_youtube_comments(id, continuation, proxies, format, locale, thin_mode,
return comments return comments
end end
def fetch_reddit_comments(id) def fetch_reddit_comments(id, sort_by = "confidence")
client = make_client(REDDIT_URL) client = make_client(REDDIT_URL)
headers = HTTP::Headers{"User-Agent" => "web:invidious:v#{CURRENT_VERSION} (by /u/omarroth)"} headers = HTTP::Headers{"User-Agent" => "web:invidious:v#{CURRENT_VERSION} (by /u/omarroth)"}
@ -260,12 +259,16 @@ def fetch_reddit_comments(id)
if search_results.status_code == 200 if search_results.status_code == 200
search_results = RedditThing.from_json(search_results.body) search_results = RedditThing.from_json(search_results.body)
# For videos that have more than one thread, choose the one with the highest score
thread = search_results.data.as(RedditListing).children.sort_by { |child| child.data.as(RedditLink).score }[-1] thread = search_results.data.as(RedditListing).children.sort_by { |child| child.data.as(RedditLink).score }[-1]
thread = thread.data.as(RedditLink) thread = thread.data.as(RedditLink)
result = client.get("/r/#{thread.subreddit}/comments/#{thread.id}.json?limit=100&sort=top", headers).body result = client.get("/r/#{thread.subreddit}/comments/#{thread.id}.json?limit=100&sort=#{sort_by}", headers).body
result = Array(RedditThing).from_json(result) result = Array(RedditThing).from_json(result)
elsif search_results.status_code == 302 elsif search_results.status_code == 302
# Previously, if there was only one result then the API would redirect to that result.
# Now, it appears it will still return a listing so this section is likely unnecessary.
result = client.get(search_results.headers["Location"], headers).body result = client.get(search_results.headers["Location"], headers).body
result = Array(RedditThing).from_json(result) result = Array(RedditThing).from_json(result)
@ -306,7 +309,7 @@ def template_youtube_comments(comments, locale, thin_mode)
html += <<-END_HTML html += <<-END_HTML
<div class="pure-g"> <div class="pure-g">
<div class="pure-u-4-24 pure-u-md-2-24"> <div class="pure-u-4-24 pure-u-md-2-24">
<img style="width:90%; padding-right:1em; padding-top:1em;" src="#{author_thumbnail}"> <img style="width:90%;padding-right:1em;padding-top:1em" src="#{author_thumbnail}">
</div> </div>
<div class="pure-u-20-24 pure-u-md-22-24"> <div class="pure-u-20-24 pure-u-md-22-24">
<p> <p>
@ -316,7 +319,7 @@ def template_youtube_comments(comments, locale, thin_mode)
<p style="white-space:pre-wrap">#{child["contentHtml"]}</p> <p style="white-space:pre-wrap">#{child["contentHtml"]}</p>
<span title="#{Time.unix(child["published"].as_i64).to_s(translate(locale, "%A %B %-d, %Y"))}">#{translate(locale, "`x` ago", recode_date(Time.unix(child["published"].as_i64), locale))} #{child["isEdited"] == true ? translate(locale, "(edited)") : ""}</span> <span title="#{Time.unix(child["published"].as_i64).to_s(translate(locale, "%A %B %-d, %Y"))}">#{translate(locale, "`x` ago", recode_date(Time.unix(child["published"].as_i64), locale))} #{child["isEdited"] == true ? translate(locale, "(edited)") : ""}</span>
| |
<a href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "Youtube permalink of the comment")}">[YT]</a> <a href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
| |
<i class="icon ion-ios-thumbs-up"></i> #{number_with_separator(child["likeCount"])} <i class="icon ion-ios-thumbs-up"></i> #{number_with_separator(child["likeCount"])}
END_HTML END_HTML
@ -442,8 +445,12 @@ def replace_links(html)
end end
end end
html = html.to_xml(options: XML::SaveOptions::NO_DECL) html = html.xpath_node(%q(//body)).not_nil!
return html if node = html.xpath_node(%q(./p))
html = node
end
return html.to_xml(options: XML::SaveOptions::NO_DECL)
end end
def fill_links(html, scheme, host) def fill_links(html, scheme, host)
@ -460,12 +467,10 @@ def fill_links(html, scheme, host)
end end
if host == "www.youtube.com" if host == "www.youtube.com"
html = html.xpath_node(%q(//body)).not_nil!.to_xml html = html.xpath_node(%q(//body/p)).not_nil!
else
html = html.to_xml(options: XML::SaveOptions::NO_DECL)
end end
return html return html.to_xml(options: XML::SaveOptions::NO_DECL)
end end
def content_to_comment_html(content) def content_to_comment_html(content)
@ -516,3 +521,111 @@ def content_to_comment_html(content)
return comment_html return comment_html
end end
def produce_comment_continuation(video_id, cursor = "", sort_by = "top")
continuation = IO::Memory.new
continuation.write(Bytes[0x12, 0x26])
continuation.write(Bytes[0x12, video_id.size])
continuation.print(video_id)
continuation.write(Bytes[0xc0, 0x01, 0x01])
continuation.write(Bytes[0xc8, 0x01, 0x01])
continuation.write(Bytes[0xe0, 0x01, 0x01])
continuation.write(Bytes[0xa2, 0x02, 0x0d])
continuation.write(Bytes[0x28, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01])
continuation.write(Bytes[0x40, 0x00])
continuation.write(Bytes[0x18, 0x06])
if cursor.empty?
continuation.write(Bytes[0x32])
continuation.write(write_var_int(video_id.size + 8))
continuation.write(Bytes[0x22, video_id.size + 4])
continuation.write(Bytes[0x22, video_id.size])
continuation.print(video_id)
case sort_by
when "top"
continuation.write(Bytes[0x30, 0x00])
when "new", "newest"
continuation.write(Bytes[0x30, 0x01])
end
continuation.write(Bytes[0x78, 0x02])
else
continuation.write(Bytes[0x32])
continuation.write(write_var_int(cursor.size + video_id.size + 11))
continuation.write(Bytes[0x0a])
continuation.write(write_var_int(cursor.size))
continuation.print(cursor)
continuation.write(Bytes[0x22, video_id.size + 4])
continuation.write(Bytes[0x22, video_id.size])
continuation.print(video_id)
case sort_by
when "top"
continuation.write(Bytes[0x30, 0x00])
when "new", "newest"
continuation.write(Bytes[0x30, 0x01])
end
continuation.write(Bytes[0x28, 0x14])
end
continuation.rewind
continuation = continuation.gets_to_end
continuation = Base64.urlsafe_encode(continuation.to_slice)
continuation = URI.escape(continuation)
return continuation
end
def produce_comment_reply_continuation(video_id, ucid, comment_id)
continuation = IO::Memory.new
continuation.write(Bytes[0x12, 0x26])
continuation.write(Bytes[0x12, video_id.size])
continuation.print(video_id)
continuation.write(Bytes[0xc0, 0x01, 0x01])
continuation.write(Bytes[0xc8, 0x01, 0x01])
continuation.write(Bytes[0xe0, 0x01, 0x01])
continuation.write(Bytes[0xa2, 0x02, 0x0d])
continuation.write(Bytes[0x28, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01])
continuation.write(Bytes[0x40, 0x00])
continuation.write(Bytes[0x18, 0x06])
continuation.write(Bytes[0x32, ucid.size + video_id.size + comment_id.size + 16])
continuation.write(Bytes[0x1a, ucid.size + video_id.size + comment_id.size + 14])
continuation.write(Bytes[0x12, comment_id.size])
continuation.print(comment_id)
continuation.write(Bytes[0x22, 0x02, 0x08, 0x00]) # ??
continuation.write(Bytes[ucid.size + video_id.size + 7])
continuation.write(Bytes[ucid.size])
continuation.print(ucid)
continuation.write(Bytes[0x32, video_id.size])
continuation.print(video_id)
continuation.write(Bytes[0x40, 0x01])
continuation.write(Bytes[0x48, 0x0a])
continuation.rewind
continuation = continuation.gets_to_end
continuation = Base64.urlsafe_encode(continuation.to_slice)
continuation = URI.escape(continuation)
return continuation
end

View File

@ -20,7 +20,9 @@ module HTTP::Handler
end end
class Kemal::RouteHandler class Kemal::RouteHandler
exclude ["/api/v1/*"] {% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
exclude ["/api/v1/*"], {{method}}
{% end %}
# Processes the route if it's a match. Otherwise renders 404. # Processes the route if it's a match. Otherwise renders 404.
private def process_request(context) private def process_request(context)
@ -31,13 +33,20 @@ class Kemal::RouteHandler
raise Kemal::Exceptions::CustomException.new(context) raise Kemal::Exceptions::CustomException.new(context)
end end
if context.request.method == "HEAD" &&
context.request.path.ends_with? ".jpg"
context.response.headers["Content-Type"] = "image/jpeg"
end
context.response.print(content) context.response.print(content)
context context
end end
end end
class Kemal::ExceptionHandler class Kemal::ExceptionHandler
exclude ["/api/v1/*"] {% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
exclude ["/api/v1/*"], {{method}}
{% end %}
private def call_exception_with_status_code(context : HTTP::Server::Context, exception : Exception, status_code : Int32) private def call_exception_with_status_code(context : HTTP::Server::Context, exception : Exception, status_code : Int32)
return if context.response.closed? return if context.response.closed?
@ -53,7 +62,8 @@ class Kemal::ExceptionHandler
end end
class FilteredCompressHandler < Kemal::Handler class FilteredCompressHandler < Kemal::Handler
exclude ["/videoplayback", "/videoplayback/*", "/vi/*", "/ggpht/*"] exclude ["/videoplayback", "/videoplayback/*", "/vi/*", "/ggpht/*", "/api/v1/auth/notifications"]
exclude ["/api/v1/auth/notifications", "/data_control"], "POST"
def call(env) def call(env)
return call_next env if exclude_match? env return call_next env if exclude_match? env
@ -77,13 +87,21 @@ class FilteredCompressHandler < Kemal::Handler
end end
class APIHandler < Kemal::Handler class APIHandler < Kemal::Handler
only ["/api/v1/*"] {% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
only ["/api/v1/*"], {{method}}
{% end %}
exclude ["/api/v1/auth/notifications"], "GET"
exclude ["/api/v1/auth/notifications"], "POST"
def call(env) def call(env)
return call_next env unless only_match? env return call_next env unless only_match? env
env.response.headers["Access-Control-Allow-Origin"] = "*" env.response.headers["Access-Control-Allow-Origin"] = "*"
# Since /api/v1/notifications is an event-stream, we don't want
# to wrap the response
return call_next env if exclude_match? env
# Here we swap out the socket IO so we can modify the response as needed # Here we swap out the socket IO so we can modify the response as needed
output = env.response.output output = env.response.output
env.response.output = IO::Memory.new env.response.output = IO::Memory.new
@ -97,13 +115,22 @@ class APIHandler < Kemal::Handler
if env.response.headers["Content-Type"]?.try &.== "application/json" if env.response.headers["Content-Type"]?.try &.== "application/json"
response = JSON.parse(response) response = JSON.parse(response)
if fields_text = env.params.query["fields"]?
begin
JSONFilter.filter(response, fields_text)
rescue ex
env.response.status_code = 400
response = {"error" => ex.message}
end
end
if env.params.query["pretty"]? && env.params.query["pretty"] == "1" if env.params.query["pretty"]? && env.params.query["pretty"] == "1"
response = response.to_pretty_json response = response.to_pretty_json
else else
response = response.to_json response = response.to_json
end end
end end
rescue rescue ex
ensure ensure
env.response.output = output env.response.output = output
env.response.puts response env.response.puts response
@ -124,10 +151,28 @@ class DenyFrame < Kemal::Handler
end end
end end
# Temp fix for https://github.com/crystal-lang/crystal/issues/7383 # Temp fixes for https://github.com/crystal-lang/crystal/issues/7383
class HTTP::UnknownLengthContent
def read_byte
ensure_send_continue
if @io.is_a?(OpenSSL::SSL::Socket::Client)
return if @io.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
end
@io.read_byte
end
end
class HTTP::Client class HTTP::Client
private def handle_response(response) private def handle_response(response)
# close unless response.keep_alive? if @socket.is_a?(OpenSSL::SSL::Socket::Client)
close unless response.keep_alive? || @socket.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
if @socket.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
@socket = nil
end
else
close unless response.keep_alive?
end
response response
end end
end end

View File

@ -1,28 +1,139 @@
require "./macros"
struct Nonce
db_mapping({
nonce: String,
expire: Time,
})
end
struct SessionId
db_mapping({
id: String,
email: String,
issued: String,
})
end
struct Annotation
db_mapping({
id: String,
annotations: String,
})
end
struct ConfigPreferences
module StringToArray
def self.to_yaml(value : Array(String), yaml : YAML::Nodes::Builder)
yaml.sequence do
value.each do |element|
yaml.scalar element
end
end
end
def self.from_yaml(ctx : YAML::ParseContext, node : YAML::Nodes::Node) : Array(String)
begin
unless node.is_a?(YAML::Nodes::Sequence)
node.raise "Expected sequence, not #{node.class}"
end
result = [] of String
node.nodes.each do |item|
unless item.is_a?(YAML::Nodes::Scalar)
node.raise "Expected scalar, not #{item.class}"
end
result << item.value
end
rescue ex
if node.is_a?(YAML::Nodes::Scalar)
result = [node.value, ""]
else
result = ["", ""]
end
end
result
end
end
yaml_mapping({
annotations: {type: Bool, default: false},
annotations_subscribed: {type: Bool, default: false},
autoplay: {type: Bool, default: false},
captions: {type: Array(String), default: ["", "", ""], converter: StringToArray},
comments: {type: Array(String), default: ["youtube", ""], converter: StringToArray},
continue: {type: Bool, default: false},
continue_autoplay: {type: Bool, default: true},
dark_mode: {type: Bool, default: false},
latest_only: {type: Bool, default: false},
listen: {type: Bool, default: false},
local: {type: Bool, default: false},
locale: {type: String, default: "en-US"},
max_results: {type: Int32, default: 40},
notifications_only: {type: Bool, default: false},
quality: {type: String, default: "hd720"},
redirect_feed: {type: Bool, default: false},
related_videos: {type: Bool, default: true},
sort: {type: String, default: "published"},
speed: {type: Float32, default: 1.0_f32},
thin_mode: {type: Bool, default: false},
unseen_only: {type: Bool, default: false},
video_loop: {type: Bool, default: false},
volume: {type: Int32, default: 100},
})
end
struct Config struct Config
module ConfigPreferencesConverter
def self.from_yaml(ctx : YAML::ParseContext, node : YAML::Nodes::Node) : Preferences
Preferences.new(*ConfigPreferences.new(ctx, node).to_tuple)
end
def self.to_yaml(value : Preferences, yaml : YAML::Nodes::Builder)
value.to_yaml(yaml)
end
end
YAML.mapping({ YAML.mapping({
channel_threads: Int32, # Number of threads to use for crawling videos from channels (for updating subscriptions) channel_threads: Int32, # Number of threads to use for crawling videos from channels (for updating subscriptions)
feed_threads: Int32, # Number of threads to use for updating feeds feed_threads: Int32, # Number of threads to use for updating feeds
db: NamedTuple( # Database configuration db: DBConfig, # Database configuration
user: String, full_refresh: Bool, # Used for crawling channels: threads should check all videos uploaded by a channel
password: String, https_only: Bool?, # Used to tell Invidious it is behind a proxy, so links to resources should be https://
host: String, hmac_key: String?, # HMAC signing key for CSRF tokens and verifying pubsub subscriptions
port: Int32, domain: String?, # Domain to be used for links to resources on the site where an absolute URL is required
dbname: String, use_pubsub_feeds: {type: Bool | Int32, default: false}, # Subscribe to channels using PubSubHubbub (requires domain, hmac_key)
), use_feed_events: {type: Bool | Int32, default: false}, # Update feeds on receiving notifications
full_refresh: Bool, # Used for crawling channels: threads should check all videos uploaded by a channel default_home: {type: String, default: "Top"},
https_only: Bool?, # Used to tell Invidious it is behind a proxy, so links to resources should be https:// feed_menu: {type: Array(String), default: ["Popular", "Top", "Trending", "Subscriptions"]},
hmac_key: String?, # HMAC signing key for CSRF tokens and verifying pubsub subscriptions top_enabled: {type: Bool, default: true},
domain: String?, # Domain to be used for links to resources on the site where an absolute URL is required captcha_enabled: {type: Bool, default: true},
use_pubsub_feeds: {type: Bool, default: false}, # Subscribe to channels using PubSubHubbub (requires domain, hmac_key) login_enabled: {type: Bool, default: true},
default_home: {type: String, default: "Top"}, registration_enabled: {type: Bool, default: true},
feed_menu: {type: Array(String), default: ["Popular", "Top", "Trending", "Subscriptions"]}, statistics_enabled: {type: Bool, default: false},
top_enabled: {type: Bool, default: true}, admins: {type: Array(String), default: [] of String},
captcha_enabled: {type: Bool, default: true}, external_port: {type: Int32?, default: nil},
login_enabled: {type: Bool, default: true}, default_user_preferences: {type: Preferences,
registration_enabled: {type: Bool, default: true}, default: Preferences.new(*ConfigPreferences.from_yaml("").to_tuple),
statistics_enabled: {type: Bool, default: false}, converter: ConfigPreferencesConverter,
admins: {type: Array(String), default: [] of String}, },
external_port: {type: Int32 | Nil, default: nil}, dmca_content: {type: Array(String), default: [] of String}, # For compliance with DMCA, disables download widget using list of video IDs
check_tables: {type: Bool, default: false}, # Check table integrity, automatically try to add any missing columns, create tables, etc.
cache_annotations: {type: Bool, default: false}, # Cache annotations requested from IA, will not cache empty annotations or annotations that only contain cards
banner: {type: String?, default: nil}, # Optional banner to be displayed along top of page for announcements, etc.
hsts: {type: Bool?, default: true}, # Enables 'Strict-Transport-Security'. Ensure that `domain` and all subdomains are served securely
})
end
struct DBConfig
yaml_mapping({
user: String,
password: String,
host: String,
port: Int32,
dbname: String,
}) })
end end
@ -86,8 +197,8 @@ def html_to_content(description_html)
return description_html, description return description_html, description
end end
def extract_videos(nodeset, ucid = nil) def extract_videos(nodeset, ucid = nil, author_name = nil)
videos = extract_items(nodeset, ucid) videos = extract_items(nodeset, ucid, author_name)
videos.select! { |item| !item.is_a?(SearchChannel | SearchPlaylist) } videos.select! { |item| !item.is_a?(SearchChannel | SearchPlaylist) }
videos.map { |video| video.as(SearchVideo) } videos.map { |video| video.as(SearchVideo) }
end end
@ -148,7 +259,7 @@ def extract_items(nodeset, ucid = nil, author_name = nil)
video_count = video_count.rchop("+") video_count = video_count.rchop("+")
end end
video_count = video_count.to_i? video_count = video_count.gsub(/\D/, "").to_i?
end end
video_count ||= 0 video_count ||= 0
@ -200,12 +311,18 @@ def extract_items(nodeset, ucid = nil, author_name = nil)
author_thumbnail = node.xpath_node(%q(.//div/span/img)).try &.["data-thumb"]? author_thumbnail = node.xpath_node(%q(.//div/span/img)).try &.["data-thumb"]?
author_thumbnail ||= node.xpath_node(%q(.//div/span/img)).try &.["src"] author_thumbnail ||= node.xpath_node(%q(.//div/span/img)).try &.["src"]
if author_thumbnail
author_thumbnail = URI.parse(author_thumbnail)
author_thumbnail.scheme = "https"
author_thumbnail = author_thumbnail.to_s
end
author_thumbnail ||= "" author_thumbnail ||= ""
subscriber_count = node.xpath_node(%q(.//span[contains(@class, "yt-subscriber-count")])).try &.["title"].delete(",").to_i? subscriber_count = node.xpath_node(%q(.//span[contains(@class, "yt-subscriber-count")])).try &.["title"].gsub(/\D/, "").to_i?
subscriber_count ||= 0 subscriber_count ||= 0
video_count = node.xpath_node(%q(.//ul[@class="yt-lockup-meta-info"]/li)).try &.content.split(" ")[0].delete(",").to_i? video_count = node.xpath_node(%q(.//ul[@class="yt-lockup-meta-info"]/li)).try &.content.split(" ")[0].gsub(/\D/, "").to_i?
video_count ||= 0 video_count ||= 0
items << SearchChannel.new( items << SearchChannel.new(
@ -367,7 +484,7 @@ def extract_shelf_items(nodeset, ucid = nil, author_name = nil)
video_count_label = child_node.xpath_node(%q(.//span[@class="formatted-video-count-label"])) video_count_label = child_node.xpath_node(%q(.//span[@class="formatted-video-count-label"]))
if video_count_label if video_count_label
video_count = video_count_label.content.strip.match(/^\d+/).try &.[0].to_i? video_count = video_count_label.content.gsub(/\D/, "").to_i?
end end
video_count ||= 50 video_count ||= 50
@ -400,3 +517,262 @@ def extract_shelf_items(nodeset, ucid = nil, author_name = nil)
return items return items
end end
def analyze_table(db, logger, table_name, struct_type = nil)
# Create table if it doesn't exist
begin
db.exec("SELECT * FROM #{table_name} LIMIT 0")
rescue ex
logger.write("CREATE TABLE #{table_name}\n")
db.using_connection do |conn|
conn.as(PG::Connection).exec_all(File.read("config/sql/#{table_name}.sql"))
end
end
if !struct_type
return
end
struct_array = struct_type.to_type_tuple
column_array = get_column_array(db, table_name)
column_types = File.read("config/sql/#{table_name}.sql").match(/CREATE TABLE public\.#{table_name}\n\((?<types>[\d\D]*?)\);/)
.try &.["types"].split(",").map { |line| line.strip }
if !column_types
return
end
struct_array.each_with_index do |name, i|
if name != column_array[i]?
if !column_array[i]?
new_column = column_types.select { |line| line.starts_with? name }[0]
logger.write("ALTER TABLE #{table_name} ADD COLUMN #{new_column}\n")
db.exec("ALTER TABLE #{table_name} ADD COLUMN #{new_column}")
next
end
# Column doesn't exist
if !column_array.includes? name
new_column = column_types.select { |line| line.starts_with? name }[0]
db.exec("ALTER TABLE #{table_name} ADD COLUMN #{new_column}")
end
# Column exists but in the wrong position, rotate
if struct_array.includes? column_array[i]
until name == column_array[i]
new_column = column_types.select { |line| line.starts_with? column_array[i] }[0]?.try &.gsub("#{column_array[i]}", "#{column_array[i]}_new")
# There's a column we didn't expect
if !new_column
logger.write("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]}\n")
db.exec("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]} CASCADE")
column_array = get_column_array(db, table_name)
next
end
logger.write("ALTER TABLE #{table_name} ADD COLUMN #{new_column}\n")
db.exec("ALTER TABLE #{table_name} ADD COLUMN #{new_column}")
logger.write("UPDATE #{table_name} SET #{column_array[i]}_new=#{column_array[i]}\n")
db.exec("UPDATE #{table_name} SET #{column_array[i]}_new=#{column_array[i]}")
logger.write("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]} CASCADE\n")
db.exec("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]} CASCADE")
logger.write("ALTER TABLE #{table_name} RENAME COLUMN #{column_array[i]}_new TO #{column_array[i]}\n")
db.exec("ALTER TABLE #{table_name} RENAME COLUMN #{column_array[i]}_new TO #{column_array[i]}")
column_array = get_column_array(db, table_name)
end
else
logger.write("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]} CASCADE\n")
db.exec("ALTER TABLE #{table_name} DROP COLUMN #{column_array[i]} CASCADE")
end
end
end
end
class PG::ResultSet
def field(index = @column_index)
@fields.not_nil![index]
end
end
def get_column_array(db, table_name)
column_array = [] of String
db.query("SELECT * FROM #{table_name} LIMIT 0") do |rs|
rs.column_count.times do |i|
column = rs.as(PG::ResultSet).field(i)
column_array << column.name
end
end
return column_array
end
def cache_annotation(db, id, annotations)
if !CONFIG.cache_annotations
return
end
body = XML.parse(annotations)
nodeset = body.xpath_nodes(%q(/document/annotations/annotation))
if nodeset == 0
return
end
has_legacy_annotations = false
nodeset.each do |node|
if !{"branding", "card", "drawer"}.includes? node["type"]?
has_legacy_annotations = true
break
end
end
if has_legacy_annotations
# TODO: Update on conflict?
db.exec("INSERT INTO annotations VALUES ($1, $2) ON CONFLICT DO NOTHING", id, annotations)
end
end
def proxy_file(response, env)
if !response.body_io?
return
end
if response.headers.includes_word?("Content-Encoding", "gzip")
Gzip::Writer.open(env.response) do |deflate|
copy_in_chunks(response.body_io, deflate)
end
elsif response.headers.includes_word?("Content-Encoding", "deflate")
Flate::Writer.open(env.response) do |deflate|
copy_in_chunks(response.body_io, deflate)
end
else
copy_in_chunks(response.body_io, env.response)
end
end
# https://stackoverflow.com/a/44802810 <3
def copy_in_chunks(input, output, chunk_size = 4096)
size = 1
while size > 0
size = IO.copy(input, output, chunk_size)
Fiber.yield
end
end
def create_notification_stream(env, proxies, config, kemal_config, decrypt_function, topics)
locale = LOCALES[env.get("preferences").as(Preferences).locale]?
env.response.content_type = "text/event-stream"
since = env.params.query["since"]?.try &.to_i?
begin
id = 0
if topics.includes? "debug"
spawn do
loop do
time_span = [0, 0, 0, 0]
time_span[rand(4)] = rand(30) + 5
published = Time.now - Time::Span.new(time_span[0], time_span[1], time_span[2], time_span[3])
video_id = TEST_IDS[rand(TEST_IDS.size)]
video = get_video(video_id, PG_DB, proxies)
video.published = published
response = JSON.parse(video.to_json(locale, config, kemal_config, decrypt_function))
if fields_text = env.params.query["fields"]?
begin
JSONFilter.filter(response, fields_text)
rescue ex
env.response.status_code = 400
response = {"error" => ex.message}
end
end
env.response.puts "id: #{id}"
env.response.puts "data: #{response.to_json}"
env.response.puts
env.response.flush
id += 1
sleep 1.minute
end
end
end
spawn do
if since
topics.try &.each do |topic|
case topic
when .match(/UC[A-Za-z0-9_-]{22}/)
PG_DB.query_all("SELECT * FROM channel_videos WHERE ucid = $1 AND published > $2 ORDER BY published DESC LIMIT 15",
topic, Time.unix(since.not_nil!), as: ChannelVideo).each do |video|
response = JSON.parse(video.to_json(locale, config, Kemal.config))
if fields_text = env.params.query["fields"]?
begin
JSONFilter.filter(response, fields_text)
rescue ex
env.response.status_code = 400
response = {"error" => ex.message}
end
end
env.response.puts "id: #{id}"
env.response.puts "data: #{response.to_json}"
env.response.puts
env.response.flush
id += 1
end
else
# TODO
end
end
end
PG.connect_listen(PG_URL, "notifications") do |event|
notification = JSON.parse(event.payload)
topic = notification["topic"].as_s
video_id = notification["videoId"].as_s
published = notification["published"].as_i64
video = get_video(video_id, PG_DB, proxies)
video.published = Time.unix(published)
response = JSON.parse(video.to_json(locale, config, Kemal.config, decrypt_function))
if fields_text = env.params.query["fields"]?
begin
JSONFilter.filter(response, fields_text)
rescue ex
env.response.status_code = 400
response = {"error" => ex.message}
end
end
if topics.try &.includes? topic
env.response.puts "id: #{id}"
env.response.puts "data: #{response.to_json}"
env.response.puts
env.response.flush
id += 1
end
end
end
# Send heartbeat
loop do
env.response.puts ":keepalive #{Time.now.to_unix}"
env.response.puts
env.response.flush
sleep (20 + rand(11)).seconds
end
rescue
end
end

View File

@ -7,8 +7,24 @@ def translate(locale : Hash(String, JSON::Any) | Nil, translation : String, text
# puts "Could not find translation for #{translation.dump}" # puts "Could not find translation for #{translation.dump}"
# end # end
if locale && locale[translation]? && !locale[translation].as_s.empty? if locale && locale[translation]?
translation = locale[translation].as_s case locale[translation]
when .as_h?
match_length = 0
locale[translation].as_h.each do |key, value|
if md = text.try &.match(/#{key}/)
if md[0].size >= match_length
translation = value.as_s
match_length = md[0].size
end
end
end
when .as_s?
if !locale[translation].as_s.empty?
translation = locale[translation].as_s
end
end
end end
if text if text
@ -17,3 +33,12 @@ def translate(locale : Hash(String, JSON::Any) | Nil, translation : String, text
return translation return translation
end end
def translate_bool(locale : Hash(String, JSON::Any) | Nil, translation : Bool)
case translation
when true
return translate(locale, "Yes")
when false
return translate(locale, "No")
end
end

View File

@ -0,0 +1,294 @@
def refresh_channels(db, logger, config)
max_channel = Channel(Int32).new
spawn do
max_threads = max_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
db.query("SELECT id FROM channels ORDER BY updated") do |rs|
rs.each do
id = rs.read(String)
if active_threads >= max_threads
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
channel = fetch_channel(id, db, config.full_refresh)
db.exec("UPDATE channels SET updated = $1, author = $2, deleted = false WHERE id = $3", Time.now, channel.author, id)
rescue ex
if ex.message == "Deleted or invalid channel"
db.exec("UPDATE channels SET updated = $1, deleted = true WHERE id = $2", Time.now, id)
end
logger.write("#{id} : #{ex.message}\n")
end
active_channel.send(true)
end
end
end
sleep 1.minute
end
end
max_channel.send(config.channel_threads)
end
def refresh_feeds(db, logger, config)
# Spawn thread to handle feed events
if config.use_feed_events
case config.use_feed_events
when Bool
max_feed_event_threads = config.use_feed_events.as(Bool).to_unsafe
when Int32
max_feed_event_threads = config.use_feed_events.as(Int32)
end
max_feed_event_channel = Channel(Int32).new
spawn do
queue = Deque(String).new(30)
PG.connect_listen(PG_URL, "feeds") do |event|
if !queue.includes? event.payload
queue << event.payload
end
end
max_threads = max_feed_event_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
until queue.empty?
event = queue.shift
if active_threads >= max_threads
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
feed = JSON.parse(event)
email = feed["email"].as_s
action = feed["action"].as_s
view_name = "subscriptions_#{sha256(email)}"
case action
when "refresh"
db.exec("REFRESH MATERIALIZED VIEW #{view_name}")
end
rescue ex
end
active_channel.send(true)
end
end
sleep 5.seconds
end
end
max_feed_event_channel.send(max_feed_event_threads.as(Int32))
end
max_channel = Channel(Int32).new
spawn do
max_threads = max_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
db.query("SELECT email FROM users") do |rs|
rs.each do
email = rs.read(String)
view_name = "subscriptions_#{sha256(email)}"
if active_threads >= max_threads
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
# Drop outdated views
column_array = get_column_array(db, view_name)
ChannelVideo.to_type_tuple.each_with_index do |name, i|
if name != column_array[i]?
logger.write("DROP MATERIALIZED VIEW #{view_name}\n")
db.exec("DROP MATERIALIZED VIEW #{view_name}")
raise "view does not exist"
end
end
db.exec("REFRESH MATERIALIZED VIEW #{view_name}")
rescue ex
# Rename old views
begin
legacy_view_name = "subscriptions_#{sha256(email)[0..7]}"
db.exec("SELECT * FROM #{legacy_view_name} LIMIT 0")
logger.write("RENAME MATERIALIZED VIEW #{legacy_view_name}\n")
db.exec("ALTER MATERIALIZED VIEW #{legacy_view_name} RENAME TO #{view_name}")
rescue ex
begin
# While iterating through, we may have an email stored from a deleted account
if db.query_one?("SELECT true FROM users WHERE email = $1", email, as: Bool)
logger.write("CREATE #{view_name}\n")
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \
SELECT * FROM channel_videos WHERE \
ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{email.gsub("'", "\\'")}')::text[]) \
ORDER BY published DESC;")
end
rescue ex
logger.write("REFRESH #{email} : #{ex.message}\n")
end
end
end
active_channel.send(true)
end
end
end
sleep 1.minute
end
end
max_channel.send(config.feed_threads)
end
def subscribe_to_feeds(db, logger, key, config)
if config.use_pubsub_feeds
case config.use_pubsub_feeds
when Bool
max_threads = config.use_pubsub_feeds.as(Bool).to_unsafe
when Int32
max_threads = config.use_pubsub_feeds.as(Int32)
end
max_channel = Channel(Int32).new
spawn do
max_threads = max_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
db.query_all("SELECT id FROM channels WHERE CURRENT_TIMESTAMP - subscribed > interval '4 days' OR subscribed IS NULL") do |rs|
rs.each do
ucid = rs.read(String)
if active_threads >= max_threads.as(Int32)
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
response = subscribe_pubsub(ucid, key, config)
if response.status_code >= 400
logger.write("#{ucid} : #{response.body}\n")
end
rescue ex
end
active_channel.send(true)
end
end
end
sleep 1.minute
end
end
max_channel.send(max_threads.as(Int32))
end
end
def pull_top_videos(config, db)
loop do
begin
top = rank_videos(db, 40)
rescue ex
next
end
if top.size > 0
args = arg_array(top)
else
next
end
videos = [] of Video
top.each do |id|
begin
videos << get_video(id, db)
rescue ex
next
end
end
yield videos
sleep 1.minute
end
end
def pull_popular_videos(db)
loop do
subscriptions = db.query_all("SELECT channel FROM \
(SELECT UNNEST(subscriptions) AS channel FROM users) AS d \
GROUP BY channel ORDER BY COUNT(channel) DESC LIMIT 40", as: String)
videos = db.query_all("SELECT DISTINCT ON (ucid) * FROM \
channel_videos WHERE ucid IN (#{arg_array(subscriptions)}) \
ORDER BY ucid, published DESC", subscriptions, as: ChannelVideo).sort_by { |video| video.published }.reverse
yield videos
sleep 1.minute
end
end
def update_decrypt_function
loop do
begin
decrypt_function = fetch_decrypt_function
rescue ex
next
end
yield decrypt_function
sleep 1.minute
end
end
def find_working_proxies(regions)
loop do
regions.each do |region|
proxies = get_proxies(region).first(20)
proxies = proxies.map { |proxy| {ip: proxy[:ip], port: proxy[:port]} }
# proxies = filter_proxies(proxies)
yield region, proxies
end
sleep 1.minute
end
end

View File

@ -0,0 +1,248 @@
module JSONFilter
alias BracketIndex = Hash(Int64, Int64)
alias GroupedFieldsValue = String | Array(GroupedFieldsValue)
alias GroupedFieldsList = Array(GroupedFieldsValue)
class FieldsParser
class ParseError < Exception
end
# Returns the `Regex` pattern used to match nest groups
def self.nest_group_pattern : Regex
# uses a '.' character to match json keys as they are allowed
# to contain any unicode codepoint
/(?:|,)(?<groupname>[^,\n]*?)\(/
end
# Returns the `Regex` pattern used to check if there are any empty nest groups
def self.unnamed_nest_group_pattern : Regex
/^\(|\(\(|\/\(/
end
def self.parse_fields(fields_text : String) : Nil
if fields_text.empty?
raise FieldsParser::ParseError.new "Fields is empty"
end
opening_bracket_count = fields_text.count('(')
closing_bracket_count = fields_text.count(')')
if opening_bracket_count != closing_bracket_count
bracket_type = opening_bracket_count > closing_bracket_count ? "opening" : "closing"
raise FieldsParser::ParseError.new "There are too many #{bracket_type} brackets (#{opening_bracket_count}:#{closing_bracket_count})"
elsif match_result = unnamed_nest_group_pattern.match(fields_text)
raise FieldsParser::ParseError.new "Unnamed nest group at position #{match_result.begin}"
end
# first, handle top-level single nested properties: items/id, playlistItems/snippet, etc
parse_single_nests(fields_text) { |nest_list| yield nest_list }
# next, handle nest groups: items(id, etag, etc)
parse_nest_groups(fields_text) { |nest_list| yield nest_list }
end
def self.parse_single_nests(fields_text : String) : Nil
single_nests = remove_nest_groups(fields_text)
if !single_nests.empty?
property_nests = single_nests.split(',')
property_nests.each do |nest|
nest_list = nest.split('/')
if nest_list.includes? ""
raise FieldsParser::ParseError.new "Empty key in nest list: #{nest_list}"
end
yield nest_list
end
# else
# raise FieldsParser::ParseError.new "Empty key in nest list 22: #{fields_text} | #{single_nests}"
end
end
def self.parse_nest_groups(fields_text : String) : Nil
nest_stack = [] of NamedTuple(group_name: String, closing_bracket_index: Int64)
bracket_pairs = get_bracket_pairs(fields_text, true)
text_index = 0
regex_index = 0
while regex_result = self.nest_group_pattern.match(fields_text, regex_index)
raw_match = regex_result[0]
group_name = regex_result["groupname"]
text_index = regex_result.begin
regex_index = regex_result.end
if text_index.nil? || regex_index.nil?
raise FieldsParser::ParseError.new "Received invalid index while parsing nest groups: text_index: #{text_index} | regex_index: #{regex_index}"
end
offset = raw_match.starts_with?(',') ? 1 : 0
opening_bracket_index = (text_index + group_name.size) + offset
closing_bracket_index = bracket_pairs[opening_bracket_index]
content_start = opening_bracket_index + 1
content = fields_text[content_start...closing_bracket_index]
if content.empty?
raise FieldsParser::ParseError.new "Empty nest group at position #{content_start}"
else
content = remove_nest_groups(content)
end
while nest_stack.size > 0 && closing_bracket_index > nest_stack[nest_stack.size - 1][:closing_bracket_index]
if nest_stack.size
nest_stack.pop
end
end
group_name.split('/').each do |group_name|
nest_stack.push({
group_name: group_name,
closing_bracket_index: closing_bracket_index,
})
end
if !content.empty?
properties = content.split(',')
properties.each do |prop|
nest_list = nest_stack.map { |nest_prop| nest_prop[:group_name] }
if !prop.empty?
if prop.includes?('/')
parse_single_nests(prop) { |list| nest_list += list }
else
nest_list.push prop
end
else
raise FieldsParser::ParseError.new "Empty key in nest list: #{nest_list << prop}"
end
yield nest_list
end
end
end
end
def self.remove_nest_groups(text : String) : String
content_bracket_pairs = get_bracket_pairs(text, false)
content_bracket_pairs.each_key.to_a.reverse.each do |opening_bracket|
closing_bracket = content_bracket_pairs[opening_bracket]
last_comma = text.rindex(',', opening_bracket) || 0
text = text[0...last_comma] + text[closing_bracket + 1...text.size]
end
return text.starts_with?(',') ? text[1...text.size] : text
end
def self.get_bracket_pairs(text : String, recursive = true) : BracketIndex
istart = [] of Int64
bracket_index = BracketIndex.new
text.each_char_with_index do |char, index|
if char == '('
istart.push(index.to_i64)
end
if char == ')'
begin
opening = istart.pop
if recursive || (!recursive && istart.size == 0)
bracket_index[opening] = index.to_i64
end
rescue
raise FieldsParser::ParseError.new "No matching opening parenthesis at: #{index}"
end
end
end
if istart.size != 0
idx = istart.pop
raise FieldsParser::ParseError.new "No matching closing parenthesis at: #{idx}"
end
return bracket_index
end
end
class FieldsGrouper
alias SkeletonValue = Hash(String, SkeletonValue)
def self.create_json_skeleton(fields_text : String) : SkeletonValue
root_hash = {} of String => SkeletonValue
FieldsParser.parse_fields(fields_text) do |nest_list|
current_item = root_hash
nest_list.each do |key|
if current_item[key]?
current_item = current_item[key]
else
current_item[key] = {} of String => SkeletonValue
current_item = current_item[key]
end
end
end
root_hash
end
def self.create_grouped_fields_list(json_skeleton : SkeletonValue) : GroupedFieldsList
grouped_fields_list = GroupedFieldsList.new
json_skeleton.each do |key, value|
grouped_fields_list.push key
nested_keys = create_grouped_fields_list(value)
grouped_fields_list.push nested_keys unless nested_keys.empty?
end
return grouped_fields_list
end
end
class FilterError < Exception
end
def self.filter(item : JSON::Any, fields_text : String, in_place : Bool = true)
skeleton = FieldsGrouper.create_json_skeleton(fields_text)
grouped_fields_list = FieldsGrouper.create_grouped_fields_list(skeleton)
filter(item, grouped_fields_list, in_place)
end
def self.filter(item : JSON::Any, grouped_fields_list : GroupedFieldsList, in_place : Bool = true) : JSON::Any
item = item.clone unless in_place
if !item.as_h? && !item.as_a?
raise FilterError.new "Can't filter '#{item}' by #{grouped_fields_list}"
end
top_level_keys = Array(String).new
grouped_fields_list.each do |value|
if value.is_a? String
top_level_keys.push value
elsif value.is_a? Array
if !top_level_keys.empty?
key_to_filter = top_level_keys.last
if item.as_h?
filter(item[key_to_filter], value, in_place: true)
elsif item.as_a?
item.as_a.each { |arr_item| filter(arr_item[key_to_filter], value, in_place: true) }
end
else
raise FilterError.new "Tried to filter while top level keys list is empty"
end
end
end
if item.as_h?
item.as_h.select! top_level_keys
elsif item.as_a?
item.as_a.map { |value| filter(value, top_level_keys, in_place: true) }
end
item
end
end

View File

@ -1,29 +1,49 @@
macro add_mapping(mapping) macro db_mapping(mapping)
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}}) def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
end end
def to_a def to_a
return [{{*mapping.keys.map { |id| "@#{id}".id }}}] return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
end end
DB.mapping({{mapping}}) def self.to_type_tuple
return { {{*mapping.keys.map { |id| "#{id}" }}} }
end
DB.mapping( {{mapping}} )
end end
macro json_mapping(mapping) macro json_mapping(mapping)
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}}) def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
end end
def to_a def to_a
return [{{*mapping.keys.map { |id| "@#{id}".id }}}] return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
end end
JSON.mapping({{mapping}}) patched_json_mapping( {{mapping}} )
YAML.mapping( {{mapping}} )
end
macro yaml_mapping(mapping)
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
end
def to_a
return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
end
def to_tuple
return { {{*mapping.keys.map { |id| "@#{id}".id }}} }
end
YAML.mapping({{mapping}})
end end
macro templated(filename, template = "template") macro templated(filename, template = "template")
render "src/invidious/views/#{{{filename}}}.ecr", "src/invidious/views/#{{{template}}}.ecr" render "src/invidious/views/#{{{filename}}}.ecr", "src/invidious/views/#{{{template}}}.ecr"
end end
macro rendered(filename) macro rendered(filename)
render "src/invidious/views/#{{{filename}}}.ecr" render "src/invidious/views/#{{{filename}}}.ecr"
end end

View File

@ -0,0 +1,166 @@
# Overloads https://github.com/crystal-lang/crystal/blob/0.28.0/src/json/from_json.cr#L24
def Object.from_json(string_or_io, default) : self
parser = JSON::PullParser.new(string_or_io)
new parser, default
end
# Adds configurable 'default' to
macro patched_json_mapping(_properties_, strict = false)
{% for key, value in _properties_ %}
{% _properties_[key] = {type: value} unless value.is_a?(HashLiteral) || value.is_a?(NamedTupleLiteral) %}
{% end %}
{% for key, value in _properties_ %}
{% _properties_[key][:key_id] = key.id.gsub(/\?$/, "") %}
{% end %}
{% for key, value in _properties_ %}
@{{value[:key_id]}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }}
{% if value[:setter] == nil ? true : value[:setter] %}
def {{value[:key_id]}}=(_{{value[:key_id]}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }})
@{{value[:key_id]}} = _{{value[:key_id]}}
end
{% end %}
{% if value[:getter] == nil ? true : value[:getter] %}
def {{key.id}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }}
@{{value[:key_id]}}
end
{% end %}
{% if value[:presence] %}
@{{value[:key_id]}}_present : Bool = false
def {{value[:key_id]}}_present?
@{{value[:key_id]}}_present
end
{% end %}
{% end %}
def initialize(%pull : ::JSON::PullParser, default = nil)
{% for key, value in _properties_ %}
%var{key.id} = nil
%found{key.id} = false
{% end %}
%location = %pull.location
begin
%pull.read_begin_object
rescue exc : ::JSON::ParseException
raise ::JSON::MappingError.new(exc.message, self.class.to_s, nil, *%location, exc)
end
while %pull.kind != :end_object
%key_location = %pull.location
key = %pull.read_object_key
case key
{% for key, value in _properties_ %}
when {{value[:key] || value[:key_id].stringify}}
%found{key.id} = true
begin
%var{key.id} =
{% if value[:nilable] || value[:default] != nil %} %pull.read_null_or { {% end %}
{% if value[:root] %}
%pull.on_key!({{value[:root]}}) do
{% end %}
{% if value[:converter] %}
{{value[:converter]}}.from_json(%pull)
{% elsif value[:type].is_a?(Path) || value[:type].is_a?(Generic) %}
{{value[:type]}}.new(%pull)
{% else %}
::Union({{value[:type]}}).new(%pull)
{% end %}
{% if value[:root] %}
end
{% end %}
{% if value[:nilable] || value[:default] != nil %} } {% end %}
rescue exc : ::JSON::ParseException
raise ::JSON::MappingError.new(exc.message, self.class.to_s, {{value[:key] || value[:key_id].stringify}}, *%key_location, exc)
end
{% end %}
else
{% if strict %}
raise ::JSON::MappingError.new("Unknown JSON attribute: #{key}", self.class.to_s, nil, *%key_location, nil)
{% else %}
%pull.skip
{% end %}
end
end
%pull.read_next
{% for key, value in _properties_ %}
{% unless value[:nilable] || value[:default] != nil %}
if %var{key.id}.nil? && !%found{key.id} && !::Union({{value[:type]}}).nilable?
raise ::JSON::MappingError.new("Missing JSON attribute: {{(value[:key] || value[:key_id]).id}}", self.class.to_s, nil, *%location, nil)
end
{% end %}
{% if value[:nilable] %}
{% if value[:default] != nil %}
@{{value[:key_id]}} = %found{key.id} ? %var{key.id} : (default.responds_to?(:{{value[:key_id]}}) ? default.{{value[:key_id]}} : {{value[:default]}})
{% else %}
@{{value[:key_id]}} = %var{key.id}
{% end %}
{% elsif value[:default] != nil %}
@{{value[:key_id]}} = %var{key.id}.nil? ? (default.responds_to?(:{{value[:key_id]}}) ? default.{{value[:key_id]}} : {{value[:default]}}) : %var{key.id}
{% else %}
@{{value[:key_id]}} = (%var{key.id}).as({{value[:type]}})
{% end %}
{% if value[:presence] %}
@{{value[:key_id]}}_present = %found{key.id}
{% end %}
{% end %}
end
def to_json(json : ::JSON::Builder)
json.object do
{% for key, value in _properties_ %}
_{{value[:key_id]}} = @{{value[:key_id]}}
{% unless value[:emit_null] %}
unless _{{value[:key_id]}}.nil?
{% end %}
json.field({{value[:key] || value[:key_id].stringify}}) do
{% if value[:root] %}
{% if value[:emit_null] %}
if _{{value[:key_id]}}.nil?
nil.to_json(json)
else
{% end %}
json.object do
json.field({{value[:root]}}) do
{% end %}
{% if value[:converter] %}
if _{{value[:key_id]}}
{{ value[:converter] }}.to_json(_{{value[:key_id]}}, json)
else
nil.to_json(json)
end
{% else %}
_{{value[:key_id]}}.to_json(json)
{% end %}
{% if value[:root] %}
{% if value[:emit_null] %}
end
{% end %}
end
end
{% end %}
end
{% unless value[:emit_null] %}
end
{% end %}
{% end %}
end
end
end

View File

@ -0,0 +1,146 @@
def generate_token(email, scopes, expire, key, db)
session = "v1:#{Base64.urlsafe_encode(Random::Secure.random_bytes(32))}"
PG_DB.exec("INSERT INTO session_ids VALUES ($1, $2, $3)", session, email, Time.now)
token = {
"session" => session,
"scopes" => scopes,
"expire" => expire,
}
if !expire
token.delete("expire")
end
token["signature"] = sign_token(key, token)
return token.to_json
end
def generate_response(session, scopes, key, db, expire = 6.hours, use_nonce = false)
expire = Time.now + expire
token = {
"session" => session,
"expire" => expire.to_unix,
"scopes" => scopes,
}
if use_nonce
nonce = Random::Secure.hex(16)
db.exec("INSERT INTO nonces VALUES ($1, $2) ON CONFLICT DO NOTHING", nonce, expire)
token["nonce"] = nonce
end
token["signature"] = sign_token(key, token)
return token.to_json
end
def sign_token(key, hash)
string_to_sign = [] of String
hash.each do |key, value|
if key == "signature"
next
end
if value.is_a?(JSON::Any)
case value
when .as_a?
value = value.as_a.map { |item| item.as_s }
end
end
case value
when Array
string_to_sign << "#{key}=#{value.sort.join(",")}"
when Tuple
string_to_sign << "#{key}=#{value.to_a.sort.join(",")}"
else
string_to_sign << "#{key}=#{value}"
end
end
string_to_sign = string_to_sign.sort.join("\n")
return Base64.urlsafe_encode(OpenSSL::HMAC.digest(:sha256, key, string_to_sign)).strip
end
def validate_request(token, session, request, key, db, locale = nil)
case token
when String
token = JSON.parse(URI.unescape(token)).as_h
when JSON::Any
token = token.as_h
when Nil
raise translate(locale, "Hidden field \"token\" is a required field")
end
if token["signature"] != sign_token(key, token)
raise translate(locale, "Invalid signature")
end
if token["session"] != session
raise translate(locale, "Erroneous token")
end
if token["nonce"]? && (nonce = db.query_one?("SELECT * FROM nonces WHERE nonce = $1", token["nonce"], as: {String, Time}))
if nonce[1] > Time.now
db.exec("UPDATE nonces SET expire = $1 WHERE nonce = $2", Time.new(1990, 1, 1), nonce[0])
else
raise translate(locale, "Erroneous token")
end
end
scopes = token["scopes"].as_a.map { |v| v.as_s }
scope = "#{request.method}:#{request.path.lchop("/api/v1/auth/").lstrip("/")}"
if !scopes_include_scope(scopes, scope)
raise translate(locale, "Invalid scope")
end
expire = token["expire"]?.try &.as_i
if expire.try &.< Time.now.to_unix
raise translate(locale, "Token is expired, please try again")
end
return {scopes, expire, token["signature"].as_s}
end
def scope_includes_scope(scope, subset)
methods, endpoint = scope.split(":")
methods = methods.split(";").map { |method| method.upcase }.reject { |method| method.empty? }.sort
endpoint = endpoint.downcase
subset_methods, subset_endpoint = subset.split(":")
subset_methods = subset_methods.split(";").map { |method| method.upcase }.sort
subset_endpoint = subset_endpoint.downcase
if methods.empty?
methods = %w(GET POST PUT HEAD DELETE PATCH OPTIONS)
end
if methods & subset_methods != subset_methods
return false
end
if endpoint.ends_with?("*") && !subset_endpoint.starts_with? endpoint.rchop("*")
return false
end
if !endpoint.ends_with?("*") && subset_endpoint != endpoint
return false
end
return true
end
def scopes_include_scope(scopes, subset)
scopes.each do |scope|
if scope_includes_scope(scope, subset)
return true
end
end
return false
end

View File

@ -18,13 +18,18 @@ def elapsed_text(elapsed)
"#{(millis * 1000).round(2)}µs" "#{(millis * 1000).round(2)}µs"
end end
def make_client(url, proxies = {} of String => Array({ip: String, port: Int32}), region = nil) def make_client(url : URI, proxies = {} of String => Array({ip: String, port: Int32}), region = nil)
context = OpenSSL::SSL::Context::Client.new context = nil
context.add_options(
OpenSSL::SSL::Options::ALL | if url.scheme == "https"
OpenSSL::SSL::Options::NO_SSL_V2 | context = OpenSSL::SSL::Context::Client.new
OpenSSL::SSL::Options::NO_SSL_V3 context.add_options(
) OpenSSL::SSL::Options::ALL |
OpenSSL::SSL::Options::NO_SSL_V2 |
OpenSSL::SSL::Options::NO_SSL_V3
)
end
client = HTTPClient.new(url, context) client = HTTPClient.new(url, context)
client.read_timeout = 10.seconds client.read_timeout = 10.seconds
client.connect_timeout = 10.seconds client.connect_timeout = 10.seconds
@ -59,8 +64,8 @@ def recode_length_seconds(time)
time = time.seconds time = time.seconds
text = "#{time.minutes.to_s.rjust(2, '0')}:#{time.seconds.to_s.rjust(2, '0')}" text = "#{time.minutes.to_s.rjust(2, '0')}:#{time.seconds.to_s.rjust(2, '0')}"
if time.hours > 0 if time.total_hours.to_i > 0
text = "#{time.hours.to_s.rjust(2, '0')}:#{text}" text = "#{time.total_hours.to_i.to_s.rjust(2, '0')}:#{text}"
end end
text = text.lchop('0') text = text.lchop('0')
@ -189,7 +194,9 @@ def number_to_short_text(number)
text = text.rchop(".0") text = text.rchop(".0")
if number / 1000000 != 0 if number / 1_000_000_000 != 0
text += "B"
elsif number / 1_000_000 != 0
text += "M" text += "M"
elsif number / 1000 != 0 elsif number / 1000 != 0
text += "K" text += "K"

View File

@ -1,195 +0,0 @@
def refresh_channels(db, logger, max_threads = 1, full_refresh = false)
max_channel = Channel(Int32).new
spawn do
max_threads = max_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
db.query("SELECT id FROM channels ORDER BY updated") do |rs|
rs.each do
id = rs.read(String)
if active_threads >= max_threads
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
channel = fetch_channel(id, db, full_refresh)
db.exec("UPDATE channels SET updated = $1, author = $2, deleted = false WHERE id = $3", Time.now, channel.author, id)
rescue ex
if ex.message == "Deleted or invalid channel"
db.exec("UPDATE channels SET updated = $1, deleted = true WHERE id = $2", Time.now, id)
end
logger.write("#{id} : #{ex.message}\n")
end
active_channel.send(true)
end
end
end
sleep 1.minute
end
end
max_channel.send(max_threads)
end
def refresh_feeds(db, logger, max_threads = 1)
max_channel = Channel(Int32).new
spawn do
max_threads = max_channel.receive
active_threads = 0
active_channel = Channel(Bool).new
loop do
db.query("SELECT email FROM users") do |rs|
rs.each do
email = rs.read(String)
view_name = "subscriptions_#{sha256(email)[0..7]}"
if active_threads >= max_threads
if active_channel.receive
active_threads -= 1
end
end
active_threads += 1
spawn do
begin
db.query("SELECT * FROM #{view_name} LIMIT 1") do |rs|
# View doesn't contain same number of rows as ChannelVideo
if ChannelVideo.from_rs(rs)[0]?.try &.to_a.size.try &.!= rs.column_count
db.exec("DROP MATERIALIZED VIEW #{view_name}")
raise "valid schema does not exist"
end
end
db.exec("REFRESH MATERIALIZED VIEW #{view_name}")
rescue ex
# Create view if it doesn't exist
if ex.message.try &.ends_with?("does not exist")
# While iterating through, we may have an email stored from a deleted account
if db.query_one?("SELECT true FROM users WHERE email = $1", email, as: Bool)
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \
SELECT * FROM channel_videos WHERE \
ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{email.gsub("'", "\\'")}')::text[]) \
ORDER BY published DESC;")
logger.write("CREATE #{view_name}\n")
end
else
logger.write("REFRESH #{email} : #{ex.message}\n")
end
end
active_channel.send(true)
end
end
end
sleep 1.minute
end
end
max_channel.send(max_threads)
end
def subscribe_to_feeds(db, logger, key, config)
if config.use_pubsub_feeds
spawn do
loop do
db.query_all("SELECT id FROM channels WHERE CURRENT_TIMESTAMP - subscribed > '4 days' OR subscribed IS NULL") do |rs|
rs.each do
ucid = rs.read(String)
response = subscribe_pubsub(ucid, key, config)
if response.status_code >= 400
logger.write("#{ucid} : #{response.body}\n")
end
end
end
sleep 1.minute
end
end
end
end
def pull_top_videos(config, db)
loop do
begin
top = rank_videos(db, 40)
rescue ex
next
end
if top.size > 0
args = arg_array(top)
else
next
end
videos = [] of Video
top.each do |id|
begin
videos << get_video(id, db)
rescue ex
next
end
end
yield videos
sleep 1.minute
end
end
def pull_popular_videos(db)
loop do
subscriptions = db.query_all("SELECT channel FROM \
(SELECT UNNEST(subscriptions) AS channel FROM users) AS d \
GROUP BY channel ORDER BY COUNT(channel) DESC LIMIT 40", as: String)
videos = db.query_all("SELECT DISTINCT ON (ucid) * FROM \
channel_videos WHERE ucid IN (#{arg_array(subscriptions)}) \
ORDER BY ucid, published DESC", subscriptions, as: ChannelVideo).sort_by { |video| video.published }.reverse
yield videos
sleep 1.minute
end
end
def update_decrypt_function
loop do
begin
decrypt_function = fetch_decrypt_function
rescue ex
next
end
yield decrypt_function
sleep 1.minute
end
end
def find_working_proxies(regions)
loop do
regions.each do |region|
proxies = get_proxies(region).first(20)
proxies = proxies.map { |proxy| {ip: proxy[:ip], port: proxy[:port]} }
# proxies = filter_proxies(proxies)
yield region, proxies
end
sleep 1.minute
end
end

View File

@ -1,5 +1,5 @@
struct MixVideo struct MixVideo
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
author: String, author: String,
@ -11,7 +11,7 @@ struct MixVideo
end end
struct Mix struct Mix
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
videos: Array(MixVideo), videos: Array(MixVideo),
@ -105,7 +105,7 @@ def template_mix(mix)
</div> </div>
<p style="width:100%">#{video["title"]}</p> <p style="width:100%">#{video["title"]}</p>
<p> <p>
<b style="width: 100%">#{video["author"]}</b> <b style="width:100%">#{video["author"]}</b>
</p> </p>
</a> </a>
</li> </li>

View File

@ -1,5 +1,5 @@
struct PlaylistVideo struct PlaylistVideo
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
author: String, author: String,
@ -13,7 +13,7 @@ struct PlaylistVideo
end end
struct Playlist struct Playlist
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
author: String, author: String,
@ -49,7 +49,7 @@ def fetch_playlist_videos(plid, page, video_count, continuation = nil, locale =
response = client.get(url) response = client.get(url)
response = JSON.parse(response.body) response = JSON.parse(response.body)
if !response["content_html"]? || response["content_html"].as_s.empty? if !response["content_html"]? || response["content_html"].as_s.empty?
raise translate(locale, "Playlist is empty") raise translate(locale, "Empty playlist")
end end
document = XML.parse_html(response["content_html"].as_s) document = XML.parse_html(response["content_html"].as_s)
@ -174,7 +174,7 @@ def fetch_playlist(plid, locale)
response = client.get("/playlist?list=#{plid}&hl=en&disable_polymer=1") response = client.get("/playlist?list=#{plid}&hl=en&disable_polymer=1")
if response.status_code != 200 if response.status_code != 200
raise translate(locale, "Invalid playlist.") raise translate(locale, "Not a playlist.")
end end
body = response.body.gsub(/<button[^>]+><span[^>]+>\s*less\s*<img[^>]+>\n<\/span><\/button>/, "") body = response.body.gsub(/<button[^>]+><span[^>]+>\s*less\s*<img[^>]+>\n<\/span><\/button>/, "")
@ -190,23 +190,27 @@ def fetch_playlist(plid, locale)
description_html ||= document.xpath_node(%q(//span[@class="pl-header-description-text"])) description_html ||= document.xpath_node(%q(//span[@class="pl-header-description-text"]))
description_html, description = html_to_content(description_html) description_html, description = html_to_content(description_html)
anchor = document.xpath_node(%q(//ul[@class="pl-header-details"])).not_nil! # YouTube allows anonymous playlists, so most of this can be empty or optional
author = anchor.xpath_node(%q(.//li[1]/a)).not_nil!.content anchor = document.xpath_node(%q(//ul[@class="pl-header-details"]))
author = anchor.try &.xpath_node(%q(.//li[1]/a)).try &.content
author ||= ""
author_thumbnail = document.xpath_node(%q(//img[@class="channel-header-profile-image"])).try &.["src"] author_thumbnail = document.xpath_node(%q(//img[@class="channel-header-profile-image"])).try &.["src"]
author_thumbnail ||= "" author_thumbnail ||= ""
ucid = anchor.xpath_node(%q(.//li[1]/a)).not_nil!["href"].split("/")[-1] ucid = anchor.try &.xpath_node(%q(.//li[1]/a)).try &.["href"].split("/")[-1]
ucid ||= ""
video_count = anchor.xpath_node(%q(.//li[2])).not_nil!.content.delete("videos, ").to_i video_count = anchor.try &.xpath_node(%q(.//li[2])).try &.content.gsub(/\D/, "").to_i?
views = anchor.xpath_node(%q(.//li[3])).not_nil!.content.delete("No views, ") video_count ||= 0
if views.empty? views = anchor.try &.xpath_node(%q(.//li[3])).try &.content.delete("No views, ").to_i64?
views = 0_i64 views ||= 0_i64
updated = anchor.try &.xpath_node(%q(.//li[4])).try &.content.lchop("Last updated on ").lchop("Updated ")
if updated
updated = decode_date(updated)
else else
views = views.to_i64 updated = Time.now
end end
updated = anchor.xpath_node(%q(.//li[4])).not_nil!.content.lchop("Last updated on ").lchop("Updated ")
updated = decode_date(updated)
playlist = Playlist.new( playlist = Playlist.new(
title: title, title: title,
id: plid, id: plid,
@ -244,7 +248,7 @@ def template_playlist(playlist)
</div> </div>
<p style="width:100%">#{video["title"]}</p> <p style="width:100%">#{video["title"]}</p>
<p> <p>
<b style="width: 100%">#{video["author"]}</b> <b style="width:100%">#{video["author"]}</b>
</p> </p>
</a> </a>
</li> </li>

View File

@ -1,5 +1,5 @@
struct SearchVideo struct SearchVideo
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
author: String, author: String,
@ -17,7 +17,7 @@ struct SearchVideo
end end
struct SearchPlaylistVideo struct SearchPlaylistVideo
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
length_seconds: Int32, length_seconds: Int32,
@ -25,7 +25,7 @@ struct SearchPlaylistVideo
end end
struct SearchPlaylist struct SearchPlaylist
add_mapping({ db_mapping({
title: String, title: String,
id: String, id: String,
author: String, author: String,
@ -37,7 +37,7 @@ struct SearchPlaylist
end end
struct SearchChannel struct SearchChannel
add_mapping({ db_mapping({
author: String, author: String,
ucid: String, ucid: String,
author_thumbnail: String, author_thumbnail: String,
@ -53,12 +53,18 @@ alias SearchItem = SearchVideo | SearchChannel | SearchPlaylist
def channel_search(query, page, channel) def channel_search(query, page, channel)
client = make_client(YT_URL) client = make_client(YT_URL)
response = client.get("/user/#{channel}?disable_polymer=1&hl=en&gl=US") response = client.get("/channel/#{channel}?disable_polymer=1&hl=en&gl=US")
document = XML.parse_html(response.body) document = XML.parse_html(response.body)
canonical = document.xpath_node(%q(//link[@rel="canonical"])) canonical = document.xpath_node(%q(//link[@rel="canonical"]))
if !canonical if !canonical
response = client.get("/channel/#{channel}?disable_polymer=1&hl=en&gl=US") response = client.get("/c/#{channel}?disable_polymer=1&hl=en&gl=US")
document = XML.parse_html(response.body)
canonical = document.xpath_node(%q(//link[@rel="canonical"]))
end
if !canonical
response = client.get("/user/#{channel}?disable_polymer=1&hl=en&gl=US")
document = XML.parse_html(response.body) document = XML.parse_html(response.body)
canonical = document.xpath_node(%q(//link[@rel="canonical"])) canonical = document.xpath_node(%q(//link[@rel="canonical"]))
end end

View File

@ -7,6 +7,8 @@ def fetch_trending(trending_type, proxies, region, locale)
region = region.upcase region = region.upcase
trending = "" trending = ""
plid = nil
if trending_type && trending_type != "Default" if trending_type && trending_type != "Default"
trending_type = trending_type.downcase.capitalize trending_type = trending_type.downcase.capitalize
@ -23,9 +25,11 @@ def fetch_trending(trending_type, proxies, region, locale)
url = tabs.select { |tab| tab["channelListSubMenuAvatarRenderer"]["title"]["simpleText"] == trending_type }[0]? url = tabs.select { |tab| tab["channelListSubMenuAvatarRenderer"]["title"]["simpleText"] == trending_type }[0]?
if url if url
url["channelListSubMenuAvatarRenderer"]["navigationEndpoint"]["commandMetadata"]["webCommandMetadata"]["url"]
url = url["channelListSubMenuAvatarRenderer"]["navigationEndpoint"]["commandMetadata"]["webCommandMetadata"]["url"].as_s url = url["channelListSubMenuAvatarRenderer"]["navigationEndpoint"]["commandMetadata"]["webCommandMetadata"]["url"].as_s
url += "&disable_polymer=1&gl=#{region}&hl=en" url += "&disable_polymer=1&gl=#{region}&hl=en"
trending = client.get(url).body trending = client.get(url).body
plid = extract_plid(url)
else else
trending = client.get("/feed/trending?gl=#{region}&hl=en&disable_polymer=1").body trending = client.get("/feed/trending?gl=#{region}&hl=en&disable_polymer=1").body
end end
@ -37,5 +41,37 @@ def fetch_trending(trending_type, proxies, region, locale)
nodeset = trending.xpath_nodes(%q(//ul/li[@class="expanded-shelf-content-item-wrapper"])) nodeset = trending.xpath_nodes(%q(//ul/li[@class="expanded-shelf-content-item-wrapper"]))
trending = extract_videos(nodeset) trending = extract_videos(nodeset)
return trending return {trending, plid}
end
def extract_plid(url)
wrapper = HTTP::Params.parse(URI.parse(url).query.not_nil!)["bp"]
wrapper = URI.unescape(wrapper)
wrapper = Base64.decode(wrapper)
# 0xe2 0x02 0x2e
wrapper += 3
# 0x0a
wrapper += 1
# Looks like "/m/[a-z0-9]{5}", not sure what it does here
item_size = wrapper[0]
wrapper += 1
item = wrapper[0, item_size]
wrapper += item.size
# 0x12
wrapper += 1
plid_size = wrapper[0]
wrapper += 1
plid = wrapper[0, plid_size]
wrapper += plid.size
plid = String.new(plid)
return plid
end end

View File

@ -11,7 +11,7 @@ struct User
end end
end end
add_mapping({ db_mapping({
updated: Time, updated: Time,
notifications: Array(String), notifications: Array(String),
subscriptions: Array(String), subscriptions: Array(String),
@ -26,29 +26,6 @@ struct User
}) })
end end
DEFAULT_USER_PREFERENCES = Preferences.new(
video_loop: false,
autoplay: false,
continue: false,
local: false,
listen: false,
speed: 1.0_f32,
quality: "hd720",
volume: 100,
comments: ["youtube", ""],
captions: ["", "", ""],
related_videos: true,
redirect_feed: false,
locale: "en-US",
dark_mode: false,
thin_mode: false,
max_results: 40,
sort: "published",
latest_only: false,
unseen_only: false,
notifications_only: false,
)
struct Preferences struct Preferences
module StringToArray module StringToArray
def self.to_json(value : Array(String), json : JSON::Builder) def self.to_json(value : Array(String), json : JSON::Builder)
@ -63,37 +40,91 @@ struct Preferences
begin begin
result = [] of String result = [] of String
value.read_array do value.read_array do
result << value.read_string result << HTML.escape(value.read_string)
end end
rescue ex rescue ex
result = [value.read_string, ""] result = [HTML.escape(value.read_string), ""]
end
result
end
def self.to_yaml(value : Array(String), yaml : YAML::Nodes::Builder)
yaml.sequence do
value.each do |element|
yaml.scalar element
end
end
end
def self.from_yaml(ctx : YAML::ParseContext, node : YAML::Nodes::Node) : Array(String)
begin
unless node.is_a?(YAML::Nodes::Sequence)
node.raise "Expected sequence, not #{node.class}"
end
result = [] of String
node.nodes.each do |item|
unless item.is_a?(YAML::Nodes::Scalar)
node.raise "Expected scalar, not #{item.class}"
end
result << HTML.escape(item.value)
end
rescue ex
if node.is_a?(YAML::Nodes::Scalar)
result = [HTML.escape(node.value), ""]
else
result = ["", ""]
end
end end
result result
end end
end end
module EscapeString
def self.to_json(value : String, json : JSON::Builder)
json.string value
end
def self.from_json(value : JSON::PullParser) : String
HTML.escape(value.read_string)
end
def self.to_yaml(value : String, yaml : YAML::Nodes::Builder)
yaml.scalar value
end
def self.from_yaml(ctx : YAML::ParseContext, node : YAML::Nodes::Node) : String
HTML.escape(node.value)
end
end
json_mapping({ json_mapping({
video_loop: {type: Bool, default: DEFAULT_USER_PREFERENCES.video_loop}, annotations: {type: Bool, default: CONFIG.default_user_preferences.annotations},
autoplay: {type: Bool, default: DEFAULT_USER_PREFERENCES.autoplay}, annotations_subscribed: {type: Bool, default: CONFIG.default_user_preferences.annotations_subscribed},
continue: {type: Bool, default: DEFAULT_USER_PREFERENCES.continue}, autoplay: {type: Bool, default: CONFIG.default_user_preferences.autoplay},
local: {type: Bool, default: DEFAULT_USER_PREFERENCES.local}, captions: {type: Array(String), default: CONFIG.default_user_preferences.captions, converter: StringToArray},
listen: {type: Bool, default: DEFAULT_USER_PREFERENCES.listen}, comments: {type: Array(String), default: CONFIG.default_user_preferences.comments, converter: StringToArray},
speed: {type: Float32, default: DEFAULT_USER_PREFERENCES.speed}, continue: {type: Bool, default: CONFIG.default_user_preferences.continue},
quality: {type: String, default: DEFAULT_USER_PREFERENCES.quality}, continue_autoplay: {type: Bool, default: CONFIG.default_user_preferences.continue_autoplay},
volume: {type: Int32, default: DEFAULT_USER_PREFERENCES.volume}, dark_mode: {type: Bool, default: CONFIG.default_user_preferences.dark_mode},
comments: {type: Array(String), default: DEFAULT_USER_PREFERENCES.comments, converter: StringToArray}, latest_only: {type: Bool, default: CONFIG.default_user_preferences.latest_only},
captions: {type: Array(String), default: DEFAULT_USER_PREFERENCES.captions, converter: StringToArray}, listen: {type: Bool, default: CONFIG.default_user_preferences.listen},
redirect_feed: {type: Bool, default: DEFAULT_USER_PREFERENCES.redirect_feed}, local: {type: Bool, default: CONFIG.default_user_preferences.local},
related_videos: {type: Bool, default: DEFAULT_USER_PREFERENCES.related_videos}, locale: {type: String, default: CONFIG.default_user_preferences.locale, converter: EscapeString},
dark_mode: {type: Bool, default: DEFAULT_USER_PREFERENCES.dark_mode}, max_results: {type: Int32, default: CONFIG.default_user_preferences.max_results},
thin_mode: {type: Bool, default: DEFAULT_USER_PREFERENCES.thin_mode}, notifications_only: {type: Bool, default: CONFIG.default_user_preferences.notifications_only},
max_results: {type: Int32, default: DEFAULT_USER_PREFERENCES.max_results}, quality: {type: String, default: CONFIG.default_user_preferences.quality, converter: EscapeString},
sort: {type: String, default: DEFAULT_USER_PREFERENCES.sort}, redirect_feed: {type: Bool, default: CONFIG.default_user_preferences.redirect_feed},
latest_only: {type: Bool, default: DEFAULT_USER_PREFERENCES.latest_only}, related_videos: {type: Bool, default: CONFIG.default_user_preferences.related_videos},
unseen_only: {type: Bool, default: DEFAULT_USER_PREFERENCES.unseen_only}, sort: {type: String, default: CONFIG.default_user_preferences.sort, converter: EscapeString},
notifications_only: {type: Bool, default: DEFAULT_USER_PREFERENCES.notifications_only}, speed: {type: Float32, default: CONFIG.default_user_preferences.speed},
locale: {type: String, default: DEFAULT_USER_PREFERENCES.locale}, thin_mode: {type: Bool, default: CONFIG.default_user_preferences.thin_mode},
unseen_only: {type: Bool, default: CONFIG.default_user_preferences.unseen_only},
video_loop: {type: Bool, default: CONFIG.default_user_preferences.video_loop},
volume: {type: Int32, default: CONFIG.default_user_preferences.volume},
}) })
end end
@ -115,7 +146,7 @@ def get_user(sid, headers, db, refresh = true)
ON CONFLICT (id) DO NOTHING", sid, user.email, Time.now) ON CONFLICT (id) DO NOTHING", sid, user.email, Time.now)
begin begin
view_name = "subscriptions_#{sha256(user.email)[0..7]}" view_name = "subscriptions_#{sha256(user.email)}"
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \ db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \
SELECT * FROM channel_videos WHERE \ SELECT * FROM channel_videos WHERE \
ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{user.email.gsub("'", "\\'")}')::text[]) \ ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{user.email.gsub("'", "\\'")}')::text[]) \
@ -137,7 +168,7 @@ def get_user(sid, headers, db, refresh = true)
ON CONFLICT (id) DO NOTHING", sid, user.email, Time.now) ON CONFLICT (id) DO NOTHING", sid, user.email, Time.now)
begin begin
view_name = "subscriptions_#{sha256(user.email)[0..7]}" view_name = "subscriptions_#{sha256(user.email)}"
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \ db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \
SELECT * FROM channel_videos WHERE \ SELECT * FROM channel_videos WHERE \
ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{user.email.gsub("'", "\\'")}')::text[]) \ ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{user.email.gsub("'", "\\'")}')::text[]) \
@ -174,7 +205,7 @@ def fetch_user(sid, headers, db)
token = Base64.urlsafe_encode(Random::Secure.random_bytes(32)) token = Base64.urlsafe_encode(Random::Secure.random_bytes(32))
user = User.new(Time.now, [] of String, channels, email, DEFAULT_USER_PREFERENCES, nil, token, [] of String) user = User.new(Time.now, [] of String, channels, email, CONFIG.default_user_preferences, nil, token, [] of String)
return user, sid return user, sid
end end
@ -182,74 +213,11 @@ def create_user(sid, email, password)
password = Crypto::Bcrypt::Password.create(password, cost: 10) password = Crypto::Bcrypt::Password.create(password, cost: 10)
token = Base64.urlsafe_encode(Random::Secure.random_bytes(32)) token = Base64.urlsafe_encode(Random::Secure.random_bytes(32))
user = User.new(Time.now, [] of String, [] of String, email, DEFAULT_USER_PREFERENCES, password.to_s, token, [] of String) user = User.new(Time.now, [] of String, [] of String, email, CONFIG.default_user_preferences, password.to_s, token, [] of String)
return user, sid return user, sid
end end
def create_response(user_id, operation, key, db, expire = 6.hours)
expire = Time.now + expire
nonce = Random::Secure.hex(16)
db.exec("INSERT INTO nonces VALUES ($1, $2) ON CONFLICT DO NOTHING", nonce, expire)
challenge = "#{expire.to_unix}-#{nonce}-#{user_id}-#{operation}"
token = OpenSSL::HMAC.digest(:sha256, key, challenge)
challenge = Base64.urlsafe_encode(challenge)
token = Base64.urlsafe_encode(token)
return challenge, token
end
def validate_response(challenge, token, user_id, operation, key, db, locale)
if !challenge
raise translate(locale, "Hidden field \"challenge\" is a required field")
end
if !token
raise translate(locale, "Hidden field \"token\" is a required field")
end
challenge = Base64.decode_string(challenge)
if challenge.split("-").size == 4
expire, nonce, challenge_user_id, challenge_operation = challenge.split("-")
expire = expire.to_i?
expire ||= 0
else
raise translate(locale, "Invalid challenge")
end
challenge = OpenSSL::HMAC.digest(:sha256, key, challenge)
challenge = Base64.urlsafe_encode(challenge)
if nonce = db.query_one?("SELECT * FROM nonces WHERE nonce = $1", nonce, as: {String, Time})
if nonce[1] > Time.now
db.exec("UPDATE nonces SET expire = $1 WHERE nonce = $2", Time.new(1990, 1, 1), nonce[0])
else
raise translate(locale, "Invalid token")
end
else
raise translate(locale, "Invalid token")
end
if challenge != token
raise translate(locale, "Invalid token")
end
if challenge_operation != operation
raise translate(locale, "Invalid token")
end
if challenge_user_id != user_id
raise translate(locale, "Invalid token")
end
if expire < Time.now.to_unix
raise translate(locale, "Token is expired, please try again")
end
end
def generate_captcha(key, db) def generate_captcha(key, db)
second = Random::Secure.rand(12) second = Random::Secure.rand(12)
second_angle = second * 30 second_angle = second * 30
@ -302,16 +270,16 @@ def generate_captcha(key, db)
return { return {
question: image, question: image,
tokens: [create_response(answer, "sign_in", key, db)], tokens: {generate_response(answer, {":login"}, key, db, use_nonce: true)},
} }
end end
def generate_text_captcha(key, db) def generate_text_captcha(key, db)
response = HTTP::Client.get(TEXTCAPTCHA_URL).body response = make_client(TEXTCAPTCHA_URL).get("/omarroth@protonmail.com.json").body
response = JSON.parse(response) response = JSON.parse(response)
tokens = response["a"].as_a.map do |answer| tokens = response["a"].as_a.map do |answer|
create_response(answer.as_s, "sign_in", key, db) generate_response(answer.as_s, {":login"}, key, db, use_nonce: true)
end end
return { return {
@ -319,3 +287,36 @@ def generate_text_captcha(key, db)
tokens: tokens, tokens: tokens,
} }
end end
def subscribe_ajax(channel_id, action, env_headers)
headers = HTTP::Headers.new
headers["Cookie"] = env_headers["Cookie"]
client = make_client(YT_URL)
html = client.get("/subscription_manager?disable_polymer=1", headers)
cookies = HTTP::Cookies.from_headers(headers)
html.cookies.each do |cookie|
if {"VISITOR_INFO1_LIVE", "YSC", "SIDCC"}.includes? cookie.name
if cookies[cookie.name]?
cookies[cookie.name] = cookie
else
cookies << cookie
end
end
end
headers = cookies.add_request_headers(headers)
if match = html.body.match(/'XSRF_TOKEN': "(?<session_token>[A-Za-z0-9\_\-\=]+)"/)
session_token = match["session_token"]
headers["content-type"] = "application/x-www-form-urlencoded"
post_req = {
"session_token" => session_token,
}
post_url = "/subscription_ajax?#{action}=1&c=#{channel_id}"
client.post(post_url, headers, form: post_req)
end
end

View File

@ -67,7 +67,7 @@ CAPTION_LANGUAGES = {
"Marathi", "Marathi",
"Mongolian", "Mongolian",
"Nepali", "Nepali",
"Norwegian", "Norwegian Bokmål",
"Nyanja", "Nyanja",
"Pashto", "Pashto",
"Persian", "Persian",
@ -241,6 +241,29 @@ VIDEO_FORMATS = {
"251" => {"ext" => "webm", "format" => "DASH audio", "acodec" => "opus", "abr" => 160}, "251" => {"ext" => "webm", "format" => "DASH audio", "acodec" => "opus", "abr" => 160},
} }
struct VideoPreferences
json_mapping({
annotations: Bool,
autoplay: Bool,
comments: Array(String),
continue: Bool,
continue_autoplay: Bool,
controls: Bool,
listen: Bool,
local: Bool,
preferred_captions: Array(String),
quality: String,
raw: Bool,
region: String?,
related_videos: Bool,
speed: (Float32 | Float64),
video_end: (Float64 | Int32),
video_loop: Bool,
video_start: (Float64 | Int32),
volume: Int32,
})
end
struct Video struct Video
property player_json : JSON::Any? property player_json : JSON::Any?
@ -250,6 +273,190 @@ struct Video
end end
end end
def to_json(locale, config, kemal_config, decrypt_function)
JSON.build do |json|
json.object do
json.field "title", self.title
json.field "videoId", self.id
json.field "videoThumbnails" do
generate_thumbnails(json, self.id, config, kemal_config)
end
json.field "storyboards" do
generate_storyboards(json, self.id, self.storyboards, config, kemal_config)
end
description_html, description = html_to_content(self.description)
json.field "description", description
json.field "descriptionHtml", description_html
json.field "published", self.published.to_unix
json.field "publishedText", translate(locale, "`x` ago", recode_date(self.published, locale))
json.field "keywords", self.keywords
json.field "viewCount", self.views
json.field "likeCount", self.likes
json.field "dislikeCount", self.dislikes
json.field "paid", self.paid
json.field "premium", self.premium
json.field "isFamilyFriendly", self.is_family_friendly
json.field "allowedRegions", self.allowed_regions
json.field "genre", self.genre
json.field "genreUrl", self.genre_url
json.field "author", self.author
json.field "authorId", self.ucid
json.field "authorUrl", "/channel/#{self.ucid}"
json.field "authorThumbnails" do
json.array do
qualities = {32, 48, 76, 100, 176, 512}
qualities.each do |quality|
json.object do
json.field "url", self.author_thumbnail.gsub("=s48-", "=s#{quality}-")
json.field "width", quality
json.field "height", quality
end
end
end
end
json.field "subCountText", self.sub_count_text
json.field "lengthSeconds", self.info["length_seconds"].to_i
json.field "allowRatings", self.allow_ratings
json.field "rating", self.info["avg_rating"].to_f32
json.field "isListed", self.is_listed
json.field "liveNow", self.live_now
json.field "isUpcoming", self.is_upcoming
if self.premiere_timestamp
json.field "premiereTimestamp", self.premiere_timestamp.not_nil!.to_unix
end
if self.player_response["streamingData"]?.try &.["hlsManifestUrl"]?
host_url = make_host_url(config, kemal_config)
hlsvp = self.player_response["streamingData"]["hlsManifestUrl"].as_s
hlsvp = hlsvp.gsub("https://manifest.googlevideo.com", host_url)
json.field "hlsUrl", hlsvp
end
json.field "dashUrl", "#{make_host_url(config, kemal_config)}/api/manifest/dash/id/#{id}"
json.field "adaptiveFormats" do
json.array do
self.adaptive_fmts(decrypt_function).each do |fmt|
json.object do
json.field "index", fmt["index"]
json.field "bitrate", fmt["bitrate"]
json.field "init", fmt["init"]
json.field "url", fmt["url"]
json.field "itag", fmt["itag"]
json.field "type", fmt["type"]
json.field "clen", fmt["clen"]
json.field "lmt", fmt["lmt"]
json.field "projectionType", fmt["projection_type"]
fmt_info = itag_to_metadata?(fmt["itag"])
if fmt_info
fps = fmt_info["fps"]?.try &.to_i || fmt["fps"]?.try &.to_i || 30
json.field "fps", fps
json.field "container", fmt_info["ext"]
json.field "encoding", fmt_info["vcodec"]? || fmt_info["acodec"]
if fmt_info["height"]?
json.field "resolution", "#{fmt_info["height"]}p"
quality_label = "#{fmt_info["height"]}p"
if fps > 30
quality_label += "60"
end
json.field "qualityLabel", quality_label
if fmt_info["width"]?
json.field "size", "#{fmt_info["width"]}x#{fmt_info["height"]}"
end
end
end
end
end
end
end
json.field "formatStreams" do
json.array do
self.fmt_stream(decrypt_function).each do |fmt|
json.object do
json.field "url", fmt["url"]
json.field "itag", fmt["itag"]
json.field "type", fmt["type"]
json.field "quality", fmt["quality"]
fmt_info = itag_to_metadata?(fmt["itag"])
if fmt_info
fps = fmt_info["fps"]?.try &.to_i || fmt["fps"]?.try &.to_i || 30
json.field "fps", fps
json.field "container", fmt_info["ext"]
json.field "encoding", fmt_info["vcodec"]? || fmt_info["acodec"]
if fmt_info["height"]?
json.field "resolution", "#{fmt_info["height"]}p"
quality_label = "#{fmt_info["height"]}p"
if fps > 30
quality_label += "60"
end
json.field "qualityLabel", quality_label
if fmt_info["width"]?
json.field "size", "#{fmt_info["width"]}x#{fmt_info["height"]}"
end
end
end
end
end
end
end
json.field "captions" do
json.array do
self.captions.each do |caption|
json.object do
json.field "label", caption.name.simpleText
json.field "languageCode", caption.languageCode
json.field "url", "/api/v1/captions/#{id}?label=#{URI.escape(caption.name.simpleText)}"
end
end
end
end
json.field "recommendedVideos" do
json.array do
self.info["rvs"]?.try &.split(",").each do |rv|
rv = HTTP::Params.parse(rv)
if rv["id"]?
json.object do
json.field "videoId", rv["id"]
json.field "title", rv["title"]
json.field "videoThumbnails" do
generate_thumbnails(json, rv["id"], config, kemal_config)
end
json.field "author", rv["author"]
json.field "lengthSeconds", rv["length_seconds"].to_i
json.field "viewCountText", rv["short_view_count_text"]
end
end
end
end
end
end
end
end
def allow_ratings def allow_ratings
allow_ratings = player_response["videoDetails"]?.try &.["allowRatings"]?.try &.as_bool allow_ratings = player_response["videoDetails"]?.try &.["allowRatings"]?.try &.as_bool
@ -473,6 +680,78 @@ struct Video
return @player_json.not_nil! return @player_json.not_nil!
end end
def storyboards
storyboards = self.player_response["storyboards"]?
.try &.as_h
.try &.["playerStoryboardSpecRenderer"]?
if !storyboards
storyboards = self.player_response["storyboards"]?
.try &.as_h
.try &.["playerLiveStoryboardSpecRenderer"]?
if storyboard = storyboards.try &.["spec"]?
.try &.as_s
return [{
url: storyboard.split("#")[0],
width: 106,
height: 60,
count: -1,
interval: 5000,
storyboard_width: 3,
storyboard_height: 3,
storyboard_count: -1,
}]
end
end
storyboards = storyboards.try &.["spec"]?
.try &.as_s.split("|")
items = [] of NamedTuple(
url: String,
width: Int32,
height: Int32,
count: Int32,
interval: Int32,
storyboard_width: Int32,
storyboard_height: Int32,
storyboard_count: Int32)
if !storyboards
return items
end
url = URI.parse(storyboards.shift)
params = HTTP::Params.parse(url.query || "")
storyboards.each_with_index do |storyboard, i|
width, height, count, storyboard_width, storyboard_height, interval, _, sigh = storyboard.split("#")
params["sigh"] = sigh
url.query = params.to_s
width = width.to_i
height = height.to_i
count = count.to_i
interval = interval.to_i
storyboard_width = storyboard_width.to_i
storyboard_height = storyboard_height.to_i
items << {
url: url.to_s.sub("$L", i).sub("$N", "M$M"),
width: width,
height: height,
count: count,
interval: interval,
storyboard_width: storyboard_width,
storyboard_height: storyboard_height,
storyboard_count: (count.to_f / (storyboard_width.to_f * storyboard_height.to_f)).ceil.to_i,
}
end
items
end
def paid def paid
reason = self.player_response["playabilityStatus"]?.try &.["reason"]? reason = self.player_response["playabilityStatus"]?.try &.["reason"]?
@ -521,7 +800,7 @@ struct Video
return self.info["length_seconds"].to_i return self.info["length_seconds"].to_i
end end
add_mapping({ db_mapping({
id: String, id: String,
info: { info: {
type: HTTP::Params, type: HTTP::Params,
@ -550,28 +829,28 @@ struct Video
end end
struct Caption struct Caption
JSON.mapping( json_mapping({
name: CaptionName, name: CaptionName,
baseUrl: String, baseUrl: String,
languageCode: String languageCode: String,
) })
end end
struct CaptionName struct CaptionName
JSON.mapping( json_mapping({
simpleText: String, simpleText: String,
) })
end end
class VideoRedirect < Exception class VideoRedirect < Exception
end end
def get_video(id, db, proxies = {} of String => Array({ip: String, port: Int32}), refresh = true, region = nil) def get_video(id, db, proxies = {} of String => Array({ip: String, port: Int32}), refresh = true, region = nil, force_refresh = false)
if db.query_one?("SELECT EXISTS (SELECT true FROM videos WHERE id = $1)", id, as: Bool) && !region if db.query_one?("SELECT EXISTS (SELECT true FROM videos WHERE id = $1)", id, as: Bool) && !region
video = db.query_one("SELECT * FROM videos WHERE id = $1", id, as: Video) video = db.query_one("SELECT * FROM videos WHERE id = $1", id, as: Video)
# If record was last updated over 10 minutes ago, refresh (expire param in response lasts for 6 hours) # If record was last updated over 10 minutes ago, refresh (expire param in response lasts for 6 hours)
if refresh && Time.now - video.updated > 10.minutes if (refresh && Time.now - video.updated > 10.minutes) || force_refresh
begin begin
video = fetch_video(id, proxies, region: region) video = fetch_video(id, proxies, region: region)
video_array = video.to_a video_array = video.to_a
@ -601,6 +880,168 @@ def get_video(id, db, proxies = {} of String => Array({ip: String, port: Int32})
return video return video
end end
def extract_polymer_config(body, html)
params = HTTP::Params.new
params["session_token"] = body.match(/"XSRF_TOKEN":"(?<session_token>[A-Za-z0-9\_\-\=]+)"/).try &.["session_token"] || ""
html_info = JSON.parse(body.match(/ytplayer\.config = (?<info>.*?);ytplayer\.load/).try &.["info"] || "{}").try &.["args"]?.try &.as_h
if html_info
html_info.each do |key, value|
params[key] = value.to_s
end
end
initial_data = JSON.parse(body.match(/window\["ytInitialData"\] = (?<info>.*?);\n/).try &.["info"] || "{}")
primary_results = initial_data["contents"]?
.try &.["twoColumnWatchNextResults"]?
.try &.["results"]?
.try &.["results"]?
.try &.["contents"]?
comment_continuation = primary_results.try &.as_a.select { |object| object["itemSectionRenderer"]? }[0]?
.try &.["itemSectionRenderer"]?
.try &.["continuations"]?
.try &.[0]?
.try &.["nextContinuationData"]?
params["ctoken"] = comment_continuation.try &.["continuation"]?.try &.as_s || ""
params["itct"] = comment_continuation.try &.["clickTrackingParams"]?.try &.as_s || ""
recommended_videos = initial_data["contents"]?
.try &.["twoColumnWatchNextResults"]?
.try &.["secondaryResults"]?
.try &.["secondaryResults"]?
.try &.["results"]?
.try &.as_a
rvs = [] of String
recommended_videos.try &.each do |compact_renderer|
if compact_renderer["compactRadioRenderer"]? || compact_renderer["compactPlaylistRenderer"]?
# TODO
elsif compact_renderer["compactVideoRenderer"]?
compact_renderer = compact_renderer["compactVideoRenderer"]
recommended_video = HTTP::Params.new
recommended_video["id"] = compact_renderer["videoId"].as_s
recommended_video["title"] = compact_renderer["title"]["simpleText"].as_s
recommended_video["author"] = compact_renderer["shortBylineText"]["runs"].as_a[0]["text"].as_s
recommended_video["ucid"] = compact_renderer["shortBylineText"]["runs"].as_a[0]["navigationEndpoint"]["browseEndpoint"]["browseId"].as_s
recommended_video["author_thumbnail"] = compact_renderer["channelThumbnail"]["thumbnails"][0]["url"].as_s
recommended_video["short_view_count_text"] = compact_renderer["shortViewCountText"]["simpleText"].as_s
recommended_video["view_count"] = compact_renderer["viewCountText"]?.try &.["simpleText"]?.try &.as_s.delete(", views watching").to_i64?.try &.to_s || "0"
recommended_video["length_seconds"] = decode_length_seconds(compact_renderer["lengthText"]?.try &.["simpleText"]?.try &.as_s || "0:00").to_s
rvs << recommended_video.to_s
end
end
params["rvs"] = rvs.join(",")
# TODO: Watching now
params["views"] = primary_results.try &.as_a.select { |object| object["videoPrimaryInfoRenderer"]? }[0]?
.try &.["videoPrimaryInfoRenderer"]?
.try &.["viewCount"]?
.try &.["videoViewCountRenderer"]?
.try &.["viewCount"]?
.try &.["simpleText"]?
.try &.as_s.gsub(/\D/, "").to_i64.to_s || "0"
sentiment_bar = primary_results.try &.as_a.select { |object| object["videoPrimaryInfoRenderer"]? }[0]?
.try &.["videoPrimaryInfoRenderer"]?
.try &.["sentimentBar"]?
.try &.["sentimentBarRenderer"]?
.try &.["tooltip"]?
.try &.as_s
likes, dislikes = sentiment_bar.try &.split(" / ").map { |a| a.delete(", ").to_i32 }[0, 2] || {0, 0}
params["likes"] = "#{likes}"
params["dislikes"] = "#{dislikes}"
published = primary_results.try &.as_a.select { |object| object["videoSecondaryInfoRenderer"]? }[0]?
.try &.["videoSecondaryInfoRenderer"]?
.try &.["dateText"]?
.try &.["simpleText"]?
.try &.as_s.split(" ")[-3..-1].join(" ")
if published
params["published"] = Time.parse(published, "%b %-d, %Y", Time::Location.local).to_unix.to_s
else
params["published"] = Time.new(1990, 1, 1).to_unix.to_s
end
params["description_html"] = "<p></p>"
description_html = primary_results.try &.as_a.select { |object| object["videoSecondaryInfoRenderer"]? }[0]?
.try &.["videoSecondaryInfoRenderer"]?
.try &.["description"]?
.try &.["runs"]?
.try &.as_a
if description_html
params["description_html"] = content_to_comment_html(description_html)
end
metadata = primary_results.try &.as_a.select { |object| object["videoSecondaryInfoRenderer"]? }[0]?
.try &.["videoSecondaryInfoRenderer"]?
.try &.["metadataRowContainer"]?
.try &.["metadataRowContainerRenderer"]?
.try &.["rows"]?
.try &.as_a
params["genre"] = ""
params["genre_ucid"] = ""
params["license"] = ""
metadata.try &.each do |row|
title = row["metadataRowRenderer"]?.try &.["title"]?.try &.["simpleText"]?.try &.as_s
contents = row["metadataRowRenderer"]?
.try &.["contents"]?
.try &.as_a[0]?
if title.try &.== "Category"
contents = contents.try &.["runs"]?
.try &.as_a[0]?
params["genre"] = contents.try &.["text"]?
.try &.as_s || ""
params["genre_ucid"] = contents.try &.["navigationEndpoint"]?
.try &.["browseEndpoint"]?
.try &.["browseId"]?.try &.as_s || ""
elsif title.try &.== "License"
contents = contents.try &.["runs"]?
.try &.as_a[0]?
params["license"] = contents.try &.["text"]?
.try &.as_s || ""
elsif title.try &.== "Licensed to YouTube by"
params["license"] = contents.try &.["simpleText"]?
.try &.as_s || ""
end
end
author_info = primary_results.try &.as_a.select { |object| object["videoSecondaryInfoRenderer"]? }[0]?
.try &.["videoSecondaryInfoRenderer"]?
.try &.["owner"]?
.try &.["videoOwnerRenderer"]?
params["author_thumbnail"] = author_info.try &.["thumbnail"]?
.try &.["thumbnails"]?
.try &.as_a[0]?
.try &.["url"]?
.try &.as_s || ""
params["sub_count_text"] = author_info.try &.["subscriberCountText"]?
.try &.["simpleText"]?
.try &.as_s.gsub(/\D/, "") || "0"
return params
end
def extract_player_config(body, html) def extract_player_config(body, html)
params = HTTP::Params.new params = HTTP::Params.new
@ -608,14 +1049,6 @@ def extract_player_config(body, html)
params["session_token"] = md["session_token"] params["session_token"] = md["session_token"]
end end
if md = body.match(/itct=(?<itct>[^"]+)"/)
params["itct"] = md["itct"]
end
if md = body.match(/'COMMENTS_TOKEN': "(?<ctoken>[^"]+)"/)
params["ctoken"] = md["ctoken"]
end
if md = body.match(/'RELATED_PLAYER_ARGS': (?<rvs>{"rvs":"[^"]+"})/) if md = body.match(/'RELATED_PLAYER_ARGS': (?<rvs>{"rvs":"[^"]+"})/)
params["rvs"] = JSON.parse(md["rvs"])["rvs"].as_s params["rvs"] = JSON.parse(md["rvs"])["rvs"].as_s
end end
@ -726,7 +1159,7 @@ def fetch_video(id, proxies, region)
info["avg_rating"] = "#{avg_rating}" info["avg_rating"] = "#{avg_rating}"
description = html.xpath_node(%q(//p[@id="eow-description"])) description = html.xpath_node(%q(//p[@id="eow-description"]))
description = description ? description.to_xml : "" description = description ? description.to_xml(options: XML::SaveOptions::NO_DECL) : ""
wilson_score = ci_lower_bound(likes, likes + dislikes) wilson_score = ci_lower_bound(likes, likes + dislikes)
@ -745,8 +1178,10 @@ def fetch_video(id, proxies, region)
genre_url = html.xpath_node(%(//ul[contains(@class, "watch-info-tag-list")]/li/a[text()="#{genre}"])).try &.["href"] genre_url = html.xpath_node(%(//ul[contains(@class, "watch-info-tag-list")]/li/a[text()="#{genre}"])).try &.["href"]
# Sometimes YouTube tries to link to invalid/missing channels, so we fix that here # YouTube provides invalid URLs for some genres, so we fix that here
case genre case genre
when "Comedy"
genre_url = "/channel/UCQZ43c4dAA9eXCQuXWu9aTw"
when "Education" when "Education"
genre_url = "/channel/UCdxpofrI-dO6oYfsqHDHphw" genre_url = "/channel/UCdxpofrI-dO6oYfsqHDHphw"
when "Gaming" when "Gaming"
@ -792,8 +1227,11 @@ def itag_to_metadata?(itag : String)
end end
def process_video_params(query, preferences) def process_video_params(query, preferences)
annotations = query["iv_load_policy"]?.try &.to_i?
autoplay = query["autoplay"]?.try &.to_i? autoplay = query["autoplay"]?.try &.to_i?
comments = query["comments"]?.try &.split(",").map { |a| a.downcase }
continue = query["continue"]?.try &.to_i? continue = query["continue"]?.try &.to_i?
continue_autoplay = query["continue_autoplay"]?.try &.to_i?
listen = query["listen"]? && (query["listen"] == "true" || query["listen"] == "1").to_unsafe listen = query["listen"]? && (query["listen"] == "true" || query["listen"] == "1").to_unsafe
local = query["local"]? && (query["local"] == "true").to_unsafe local = query["local"]? && (query["local"] == "true").to_unsafe
preferred_captions = query["subtitles"]?.try &.split(",").map { |a| a.downcase } preferred_captions = query["subtitles"]?.try &.split(",").map { |a| a.downcase }
@ -806,8 +1244,11 @@ def process_video_params(query, preferences)
if preferences if preferences
# region ||= preferences.region # region ||= preferences.region
annotations ||= preferences.annotations.to_unsafe
autoplay ||= preferences.autoplay.to_unsafe autoplay ||= preferences.autoplay.to_unsafe
comments ||= preferences.comments
continue ||= preferences.continue.to_unsafe continue ||= preferences.continue.to_unsafe
continue_autoplay ||= preferences.continue_autoplay.to_unsafe
listen ||= preferences.listen.to_unsafe listen ||= preferences.listen.to_unsafe
local ||= preferences.local.to_unsafe local ||= preferences.local.to_unsafe
preferred_captions ||= preferences.captions preferred_captions ||= preferences.captions
@ -818,19 +1259,24 @@ def process_video_params(query, preferences)
volume ||= preferences.volume volume ||= preferences.volume
end end
autoplay ||= DEFAULT_USER_PREFERENCES.autoplay.to_unsafe annotations ||= CONFIG.default_user_preferences.annotations.to_unsafe
continue ||= DEFAULT_USER_PREFERENCES.continue.to_unsafe autoplay ||= CONFIG.default_user_preferences.autoplay.to_unsafe
listen ||= DEFAULT_USER_PREFERENCES.listen.to_unsafe comments ||= CONFIG.default_user_preferences.comments
local ||= DEFAULT_USER_PREFERENCES.local.to_unsafe continue ||= CONFIG.default_user_preferences.continue.to_unsafe
preferred_captions ||= DEFAULT_USER_PREFERENCES.captions continue_autoplay ||= CONFIG.default_user_preferences.continue_autoplay.to_unsafe
quality ||= DEFAULT_USER_PREFERENCES.quality listen ||= CONFIG.default_user_preferences.listen.to_unsafe
related_videos ||= DEFAULT_USER_PREFERENCES.related_videos.to_unsafe local ||= CONFIG.default_user_preferences.local.to_unsafe
speed ||= DEFAULT_USER_PREFERENCES.speed preferred_captions ||= CONFIG.default_user_preferences.captions
video_loop ||= DEFAULT_USER_PREFERENCES.video_loop.to_unsafe quality ||= CONFIG.default_user_preferences.quality
volume ||= DEFAULT_USER_PREFERENCES.volume related_videos ||= CONFIG.default_user_preferences.related_videos.to_unsafe
speed ||= CONFIG.default_user_preferences.speed
video_loop ||= CONFIG.default_user_preferences.video_loop.to_unsafe
volume ||= CONFIG.default_user_preferences.volume
annotations = annotations == 1
autoplay = autoplay == 1 autoplay = autoplay == 1
continue = continue == 1 continue = continue == 1
continue_autoplay = continue_autoplay == 1
listen = listen == 1 listen = listen == 1
local = local == 1 local = local == 1
related_videos = related_videos == 1 related_videos = related_videos == 1
@ -859,25 +1305,28 @@ def process_video_params(query, preferences)
controls = query["controls"]?.try &.to_i? controls = query["controls"]?.try &.to_i?
controls ||= 1 controls ||= 1
controls = controls == 1 controls = controls >= 1
params = { params = VideoPreferences.new(
autoplay: autoplay, annotations: annotations,
continue: continue, autoplay: autoplay,
controls: controls, comments: comments,
listen: listen, continue: continue,
local: local, continue_autoplay: continue_autoplay,
controls: controls,
listen: listen,
local: local,
preferred_captions: preferred_captions, preferred_captions: preferred_captions,
quality: quality, quality: quality,
raw: raw, raw: raw,
region: region, region: region,
related_videos: related_videos, related_videos: related_videos,
speed: speed, speed: speed,
video_end: video_end, video_end: video_end,
video_loop: video_loop, video_loop: video_loop,
video_start: video_start, video_start: video_start,
volume: volume, volume: volume,
} )
return params return params
end end
@ -908,3 +1357,21 @@ def generate_thumbnails(json, id, config, kemal_config)
end end
end end
end end
def generate_storyboards(json, id, storyboards, config, kemal_config)
json.array do
storyboards.each do |storyboard|
json.object do
json.field "url", "/api/v1/storyboards/#{id}?width=#{storyboard[:width]}&height=#{storyboard[:height]}"
json.field "templateUrl", storyboard[:url]
json.field "width", storyboard[:width]
json.field "height", storyboard[:height]
json.field "count", storyboard[:count]
json.field "interval", storyboard[:interval]
json.field "storyboardWidth", storyboard[:storyboard_width]
json.field "storyboardHeight", storyboard[:storyboard_height]
json.field "storyboardCount", storyboard[:storyboard_count]
end
end
end
end