mirror of
https://github.com/fallenbagel/jellyseerr.git
synced 2025-12-24 02:39:18 -05:00
Compare commits
37 Commits
preview-te
...
e5c95e00b9
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e5c95e00b9 | ||
|
|
4b4272dc10 | ||
|
|
609082c7b3 | ||
|
|
39e6115467 | ||
|
|
85bca35f98 | ||
|
|
48bebaf727 | ||
|
|
91d202fcca | ||
|
|
6d3db3d596 | ||
|
|
d60b75adf4 | ||
|
|
af75e717f4 | ||
|
|
306582e87f | ||
|
|
7ff0a8c040 | ||
|
|
be5bdc9975 | ||
|
|
20d53a6a3e | ||
|
|
1fb296d64b | ||
|
|
9263f2f4b5 | ||
|
|
3d1083279c | ||
|
|
337882515f | ||
|
|
431fa6ba98 | ||
|
|
002b769f3f | ||
|
|
a8307e9118 | ||
|
|
0de2ed2086 | ||
|
|
7fabd0b1c0 | ||
|
|
37cc665706 | ||
|
|
2fb742e2a3 | ||
|
|
2822240d0f | ||
|
|
d79a91e556 | ||
|
|
bc5d441047 | ||
|
|
b560e50d85 | ||
|
|
381d82488e | ||
|
|
3f899f5e76 | ||
|
|
3ee69663dc | ||
|
|
539d49879d | ||
|
|
15356dfe49 | ||
|
|
1f04eeb040 | ||
|
|
e3028c21f2 | ||
|
|
9d8b343790 |
@@ -8,7 +8,7 @@
|
|||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="https://discord.gg/seerr"><img src="https://img.shields.io/discord/783137440809746482" alt="Discord"></a>
|
<a href="https://discord.gg/seerr"><img src="https://img.shields.io/discord/783137440809746482" alt="Discord"></a>
|
||||||
<a href="https://hub.docker.com/r/seerr/seerr"><img src="https://img.shields.io/docker/pulls/seerr/seerr" alt="Docker pulls"></a>
|
<a href="https://hub.docker.com/r/seerr/seerr"><img src="https://img.shields.io/docker/pulls/seerr/seerr" alt="Docker pulls"></a>
|
||||||
<a href="https://translate.seerr.dev/engage/seerr/"><img src="https://translate.seerr.dev/widget/seerr/seerr-frontend/svg-badge.svg" alt="Translation status" /></a>
|
<a href="https://translate.seerr.dev/engage/seerr/"><img src="https://translate.seerr.dev/widget/seerr/svg-badge.svg" alt="Translation status" /></a>
|
||||||
<a href="https://github.com/seerr-team/seerr/blob/develop/LICENSE"><img alt="GitHub" src="https://img.shields.io/github/license/seerr-team/seerr"></a>
|
<a href="https://github.com/seerr-team/seerr/blob/develop/LICENSE"><img alt="GitHub" src="https://img.shields.io/github/license/seerr-team/seerr"></a>
|
||||||
|
|
||||||
**Seerr** is a free and open source software application for managing requests for your media library. It integrates with the media server of your choice: [Jellyfin](https://jellyfin.org), [Plex](https://plex.tv), and [Emby](https://emby.media/). In addition, it integrates with your existing services, such as **[Sonarr](https://sonarr.tv/)**, **[Radarr](https://radarr.video/)**.
|
**Seerr** is a free and open source software application for managing requests for your media library. It integrates with the media server of your choice: [Jellyfin](https://jellyfin.org), [Plex](https://plex.tv), and [Emby](https://emby.media/). In addition, it integrates with your existing services, such as **[Sonarr](https://sonarr.tv/)**, **[Radarr](https://radarr.video/)**.
|
||||||
|
|||||||
@@ -174,4 +174,36 @@ This can happen if you have a new installation of Jellyfin/Emby or if you have c
|
|||||||
|
|
||||||
This process should restore your admin privileges while preserving your settings.
|
This process should restore your admin privileges while preserving your settings.
|
||||||
|
|
||||||
|
## Failed to enable web push notifications
|
||||||
|
|
||||||
|
### Option 1: You are using Pi-hole
|
||||||
|
|
||||||
|
When using Pi-hole, you need to whitelist the proper domains in order for the queries to not be intercepted and blocked by Pi-hole.
|
||||||
|
If you are using a chromium based browser (eg: Chrome, Brave, Edge...), the domain you need to whitelist is `fcm.googleapis.com`
|
||||||
|
If you are using Firefox, the domain you need to whitelist is `push.services.mozilla.com`
|
||||||
|
|
||||||
|
1. Log into your Pi-hole through the admin interface, then click on Domains situated under GROUP MANAGEMENT.
|
||||||
|
2. Add the domain corresponding to your browser in the `Domain to be added` field and then click on Add to allowed domains.
|
||||||
|
3. Now in order for those changes to be used you need to flush your current dns cache.
|
||||||
|
4. You can do so by using this command line in your Pi-hole terminal:
|
||||||
|
```bash
|
||||||
|
pihole restartdns
|
||||||
|
```
|
||||||
|
If this command fails (which is unlikely), use this equivalent:
|
||||||
|
```bash
|
||||||
|
pihole -f && pihole restartdns
|
||||||
|
```
|
||||||
|
5. Then restart your Seerr instance and try to enable the web push notifications again.
|
||||||
|
|
||||||
|
|
||||||
|
### Option 2: You are using Brave browser
|
||||||
|
|
||||||
|
Brave is a "De-Googled" browser. So by default or if you refused a prompt in the past, it cuts the access to the FCM (Firebase Cloud Messaging) service, which is mandatory for the web push notifications on Chromium based browsers.
|
||||||
|
|
||||||
|
1. Open Brave and paste this address in the url bar: `brave://settings/privacy`
|
||||||
|
2. Look for the option: "Use Google services for push messaging"
|
||||||
|
3. Activate this option
|
||||||
|
4. Relaunch Brave completely
|
||||||
|
5. You should now see the notifications prompt appearing instead of an error message.
|
||||||
|
|
||||||
If you still encounter issues, please reach out on our support channels.
|
If you still encounter issues, please reach out on our support channels.
|
||||||
|
|||||||
100
package.json
100
package.json
@@ -2,7 +2,7 @@
|
|||||||
"name": "seerr",
|
"name": "seerr",
|
||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"packageManager": "pnpm@10.17.1",
|
"packageManager": "pnpm@10.24.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"preinstall": "npx only-allow pnpm",
|
"preinstall": "npx only-allow pnpm",
|
||||||
"postinstall": "node postinstall-win.js",
|
"postinstall": "node postinstall-win.js",
|
||||||
@@ -33,38 +33,38 @@
|
|||||||
},
|
},
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@dr.pogodin/csurf": "^1.14.1",
|
"@dr.pogodin/csurf": "^1.16.6",
|
||||||
"@formatjs/intl-displaynames": "6.2.6",
|
"@formatjs/intl-displaynames": "6.8.13",
|
||||||
"@formatjs/intl-locale": "3.1.1",
|
"@formatjs/intl-locale": "3.1.1",
|
||||||
"@formatjs/intl-pluralrules": "5.1.10",
|
"@formatjs/intl-pluralrules": "5.4.6",
|
||||||
"@formatjs/intl-utils": "3.8.4",
|
"@formatjs/intl-utils": "3.8.4",
|
||||||
"@formatjs/swc-plugin-experimental": "^0.4.0",
|
"@formatjs/swc-plugin-experimental": "^0.4.0",
|
||||||
"@headlessui/react": "1.7.12",
|
"@headlessui/react": "1.7.12",
|
||||||
"@heroicons/react": "2.0.16",
|
"@heroicons/react": "2.2.0",
|
||||||
"@supercharge/request-ip": "1.2.0",
|
"@supercharge/request-ip": "1.2.0",
|
||||||
"@svgr/webpack": "6.5.1",
|
"@svgr/webpack": "6.5.1",
|
||||||
"@tanem/react-nprogress": "5.0.30",
|
"@tanem/react-nprogress": "5.0.56",
|
||||||
"@types/ua-parser-js": "^0.7.36",
|
"@types/ua-parser-js": "^0.7.36",
|
||||||
"@types/wink-jaro-distance": "^2.0.2",
|
"@types/wink-jaro-distance": "^2.0.2",
|
||||||
"ace-builds": "1.15.2",
|
"ace-builds": "1.43.4",
|
||||||
"axios": "1.10.0",
|
"axios": "1.13.2",
|
||||||
"axios-rate-limit": "1.3.0",
|
"axios-rate-limit": "1.4.0",
|
||||||
"bcrypt": "5.1.0",
|
"bcrypt": "5.1.0",
|
||||||
"bowser": "2.11.0",
|
"bowser": "2.13.1",
|
||||||
"connect-typeorm": "1.1.4",
|
"connect-typeorm": "1.1.4",
|
||||||
"cookie-parser": "1.4.7",
|
"cookie-parser": "1.4.7",
|
||||||
"copy-to-clipboard": "3.3.3",
|
"copy-to-clipboard": "3.3.3",
|
||||||
"country-flag-icons": "1.5.5",
|
"country-flag-icons": "1.6.4",
|
||||||
"cronstrue": "2.23.0",
|
"cronstrue": "2.23.0",
|
||||||
"date-fns": "2.29.3",
|
"date-fns": "2.29.3",
|
||||||
"dayjs": "1.11.7",
|
"dayjs": "1.11.19",
|
||||||
"dns-caching": "^0.2.7",
|
"dns-caching": "^0.2.7",
|
||||||
"email-templates": "12.0.1",
|
"email-templates": "12.0.3",
|
||||||
"express": "4.21.2",
|
"express": "4.21.2",
|
||||||
"express-openapi-validator": "4.13.8",
|
"express-openapi-validator": "4.13.8",
|
||||||
"express-rate-limit": "6.7.0",
|
"express-rate-limit": "6.7.0",
|
||||||
"express-session": "1.17.3",
|
"express-session": "1.18.2",
|
||||||
"formik": "^2.4.6",
|
"formik": "^2.4.9",
|
||||||
"gravatar-url": "3.1.0",
|
"gravatar-url": "3.1.0",
|
||||||
"http-proxy-agent": "^7.0.2",
|
"http-proxy-agent": "^7.0.2",
|
||||||
"https-proxy-agent": "^7.0.6",
|
"https-proxy-agent": "^7.0.6",
|
||||||
@@ -76,19 +76,19 @@
|
|||||||
"node-schedule": "2.1.1",
|
"node-schedule": "2.1.1",
|
||||||
"nodemailer": "6.10.0",
|
"nodemailer": "6.10.0",
|
||||||
"openpgp": "5.11.2",
|
"openpgp": "5.11.2",
|
||||||
"pg": "8.11.0",
|
"pg": "8.16.3",
|
||||||
"plex-api": "5.3.2",
|
"plex-api": "5.3.2",
|
||||||
"pug": "3.0.3",
|
"pug": "3.0.3",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-ace": "10.1.0",
|
"react-ace": "10.1.0",
|
||||||
"react-animate-height": "2.1.2",
|
"react-animate-height": "2.1.2",
|
||||||
"react-aria": "3.23.0",
|
"react-aria": "3.44.0",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
"react-intersection-observer": "9.4.3",
|
"react-intersection-observer": "9.4.3",
|
||||||
"react-intl": "^6.6.8",
|
"react-intl": "^6.6.8",
|
||||||
"react-markdown": "8.0.5",
|
"react-markdown": "8.0.5",
|
||||||
"react-popper-tooltip": "4.4.2",
|
"react-popper-tooltip": "4.4.2",
|
||||||
"react-select": "5.7.0",
|
"react-select": "5.10.2",
|
||||||
"react-spring": "9.7.1",
|
"react-spring": "9.7.1",
|
||||||
"react-tailwindcss-datepicker-sct": "1.3.4",
|
"react-tailwindcss-datepicker-sct": "1.3.4",
|
||||||
"react-toast-notifications": "2.5.1",
|
"react-toast-notifications": "2.5.1",
|
||||||
@@ -97,19 +97,19 @@
|
|||||||
"react-use-clipboard": "1.0.9",
|
"react-use-clipboard": "1.0.9",
|
||||||
"reflect-metadata": "0.1.13",
|
"reflect-metadata": "0.1.13",
|
||||||
"secure-random-password": "0.2.3",
|
"secure-random-password": "0.2.3",
|
||||||
"semver": "7.7.1",
|
"semver": "7.7.3",
|
||||||
"sharp": "^0.33.4",
|
"sharp": "^0.33.4",
|
||||||
"sqlite3": "5.1.7",
|
"sqlite3": "5.1.7",
|
||||||
"swagger-ui-express": "4.6.2",
|
"swagger-ui-express": "4.6.2",
|
||||||
"swr": "2.2.5",
|
"swr": "2.3.7",
|
||||||
"tailwind-merge": "^2.6.0",
|
"tailwind-merge": "^2.6.0",
|
||||||
"typeorm": "0.3.12",
|
"typeorm": "0.3.12",
|
||||||
"ua-parser-js": "^1.0.35",
|
"ua-parser-js": "^1.0.35",
|
||||||
"undici": "^7.3.0",
|
"undici": "^7.16.0",
|
||||||
"validator": "^13.15.15",
|
"validator": "^13.15.23",
|
||||||
"web-push": "3.5.0",
|
"web-push": "3.6.7",
|
||||||
"wink-jaro-distance": "^2.0.0",
|
"wink-jaro-distance": "^2.0.0",
|
||||||
"winston": "3.8.2",
|
"winston": "3.18.3",
|
||||||
"winston-daily-rotate-file": "4.7.1",
|
"winston-daily-rotate-file": "4.7.1",
|
||||||
"xml2js": "0.4.23",
|
"xml2js": "0.4.23",
|
||||||
"yamljs": "0.3.0",
|
"yamljs": "0.3.0",
|
||||||
@@ -123,32 +123,33 @@
|
|||||||
"@tailwindcss/forms": "0.5.10",
|
"@tailwindcss/forms": "0.5.10",
|
||||||
"@tailwindcss/typography": "0.5.16",
|
"@tailwindcss/typography": "0.5.16",
|
||||||
"@types/bcrypt": "5.0.0",
|
"@types/bcrypt": "5.0.0",
|
||||||
"@types/cookie-parser": "1.4.3",
|
"@types/cookie-parser": "1.4.10",
|
||||||
"@types/country-flag-icons": "1.2.0",
|
"@types/country-flag-icons": "1.2.2",
|
||||||
"@types/csurf": "1.11.2",
|
"@types/csurf": "1.11.5",
|
||||||
"@types/email-templates": "8.0.4",
|
"@types/email-templates": "8.0.4",
|
||||||
"@types/express": "4.17.17",
|
"@types/express": "4.17.17",
|
||||||
"@types/express-session": "1.17.6",
|
"@types/express-session": "1.18.2",
|
||||||
"@types/lodash": "4.14.191",
|
"@types/lodash": "4.17.21",
|
||||||
"@types/mime": "3",
|
"@types/mime": "3",
|
||||||
"@types/node": "22.10.5",
|
"@types/node": "22.10.5",
|
||||||
"@types/node-schedule": "2.1.0",
|
"@types/node-schedule": "2.1.8",
|
||||||
"@types/nodemailer": "6.4.7",
|
"@types/nodemailer": "6.4.7",
|
||||||
"@types/react": "^18.3.3",
|
"@types/react": "^18.3.3",
|
||||||
"@types/react-dom": "^18.3.0",
|
"@types/react-dom": "^18.3.0",
|
||||||
"@types/react-transition-group": "4.4.5",
|
"@types/react-transition-group": "4.4.12",
|
||||||
"@types/secure-random-password": "0.2.1",
|
"@types/secure-random-password": "0.2.1",
|
||||||
"@types/semver": "7.3.13",
|
"@types/semver": "7.7.1",
|
||||||
"@types/swagger-ui-express": "4.1.3",
|
"@types/swagger-ui-express": "4.1.8",
|
||||||
"@types/validator": "^13.15.3",
|
"@types/validator": "^13.15.10",
|
||||||
"@types/web-push": "3.3.2",
|
"@types/web-push": "3.6.4",
|
||||||
"@types/xml2js": "0.4.11",
|
"@types/xml2js": "0.4.11",
|
||||||
"@types/yamljs": "0.2.31",
|
"@types/yamljs": "0.2.31",
|
||||||
"@types/yup": "0.29.14",
|
"@types/yup": "0.29.14",
|
||||||
"@typescript-eslint/eslint-plugin": "5.54.0",
|
"@typescript-eslint/eslint-plugin": "5.54.0",
|
||||||
"@typescript-eslint/parser": "5.54.0",
|
"@typescript-eslint/parser": "5.54.0",
|
||||||
"autoprefixer": "10.4.13",
|
"autoprefixer": "10.4.22",
|
||||||
"commitizen": "4.3.0",
|
"baseline-browser-mapping": "^2.8.32",
|
||||||
|
"commitizen": "4.3.1",
|
||||||
"copyfiles": "2.4.1",
|
"copyfiles": "2.4.1",
|
||||||
"cy-mobile-commands": "0.3.0",
|
"cy-mobile-commands": "0.3.0",
|
||||||
"cypress": "14.1.0",
|
"cypress": "14.1.0",
|
||||||
@@ -157,22 +158,22 @@
|
|||||||
"eslint-config-next": "^14.2.4",
|
"eslint-config-next": "^14.2.4",
|
||||||
"eslint-config-prettier": "8.6.0",
|
"eslint-config-prettier": "8.6.0",
|
||||||
"eslint-plugin-formatjs": "4.9.0",
|
"eslint-plugin-formatjs": "4.9.0",
|
||||||
"eslint-plugin-jsx-a11y": "6.7.1",
|
"eslint-plugin-jsx-a11y": "6.10.2",
|
||||||
"eslint-plugin-no-relative-import-paths": "1.5.2",
|
"eslint-plugin-no-relative-import-paths": "1.6.1",
|
||||||
"eslint-plugin-prettier": "4.2.1",
|
"eslint-plugin-prettier": "4.2.1",
|
||||||
"eslint-plugin-react": "7.32.2",
|
"eslint-plugin-react": "7.37.5",
|
||||||
"eslint-plugin-react-hooks": "4.6.0",
|
"eslint-plugin-react-hooks": "4.6.0",
|
||||||
"husky": "8.0.3",
|
"husky": "8.0.3",
|
||||||
"lint-staged": "13.1.2",
|
"lint-staged": "13.1.2",
|
||||||
"nodemon": "3.1.9",
|
"nodemon": "3.1.11",
|
||||||
"postcss": "8.4.31",
|
"postcss": "8.5.6",
|
||||||
"prettier": "2.8.4",
|
"prettier": "2.8.4",
|
||||||
"prettier-plugin-organize-imports": "3.2.2",
|
"prettier-plugin-organize-imports": "3.2.2",
|
||||||
"prettier-plugin-tailwindcss": "0.2.3",
|
"prettier-plugin-tailwindcss": "0.2.3",
|
||||||
"tailwindcss": "3.2.7",
|
"tailwindcss": "3.2.7",
|
||||||
"ts-node": "10.9.1",
|
"ts-node": "10.9.2",
|
||||||
"tsc-alias": "1.8.2",
|
"tsc-alias": "1.8.16",
|
||||||
"tsconfig-paths": "4.1.2",
|
"tsconfig-paths": "4.2.0",
|
||||||
"typescript": "4.9.5"
|
"typescript": "4.9.5"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
@@ -181,7 +182,7 @@
|
|||||||
},
|
},
|
||||||
"overrides": {
|
"overrides": {
|
||||||
"sqlite3/node-gyp": "8.4.1",
|
"sqlite3/node-gyp": "8.4.1",
|
||||||
"@types/express-session": "1.17.6"
|
"@types/express-session": "1.18.2"
|
||||||
},
|
},
|
||||||
"config": {
|
"config": {
|
||||||
"commitizen": {
|
"commitizen": {
|
||||||
@@ -204,8 +205,11 @@
|
|||||||
},
|
},
|
||||||
"pnpm": {
|
"pnpm": {
|
||||||
"onlyBuiltDependencies": [
|
"onlyBuiltDependencies": [
|
||||||
"sqlite3",
|
"@swc/core",
|
||||||
"bcrypt"
|
"bcrypt",
|
||||||
|
"cypress",
|
||||||
|
"sharp",
|
||||||
|
"sqlite3"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
4120
pnpm-lock.yaml
generated
4120
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -112,6 +112,10 @@ export interface JellyfinLibraryItemExtended extends JellyfinLibraryItem {
|
|||||||
DateCreated?: string;
|
DateCreated?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type EpisodeReturn<T> = T extends { includeMediaInfo: true }
|
||||||
|
? JellyfinLibraryItemExtended[]
|
||||||
|
: JellyfinLibraryItem[];
|
||||||
|
|
||||||
export interface JellyfinItemsReponse {
|
export interface JellyfinItemsReponse {
|
||||||
Items: JellyfinLibraryItemExtended[];
|
Items: JellyfinLibraryItemExtended[];
|
||||||
TotalRecordCount: number;
|
TotalRecordCount: number;
|
||||||
@@ -415,13 +419,22 @@ class JellyfinAPI extends ExternalAPI {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getEpisodes(
|
public async getEpisodes<
|
||||||
|
T extends { includeMediaInfo?: boolean } | undefined = undefined
|
||||||
|
>(
|
||||||
seriesID: string,
|
seriesID: string,
|
||||||
seasonID: string
|
seasonID: string,
|
||||||
): Promise<JellyfinLibraryItem[]> {
|
options?: T
|
||||||
|
): Promise<EpisodeReturn<T>> {
|
||||||
try {
|
try {
|
||||||
const episodeResponse = await this.get<any>(
|
const episodeResponse = await this.get<any>(
|
||||||
`/Shows/${seriesID}/Episodes?seasonId=${seasonID}`
|
`/Shows/${seriesID}/Episodes`,
|
||||||
|
{
|
||||||
|
params: {
|
||||||
|
seasonId: seasonID,
|
||||||
|
...(options?.includeMediaInfo && { fields: 'MediaSources' }),
|
||||||
|
},
|
||||||
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
return episodeResponse.Items.filter(
|
return episodeResponse.Items.filter(
|
||||||
|
|||||||
@@ -1,8 +1,15 @@
|
|||||||
import { DbAwareColumn } from '@server/utils/DbColumnHelper';
|
import { DbAwareColumn } from '@server/utils/DbColumnHelper';
|
||||||
import { Column, Entity, ManyToOne, PrimaryGeneratedColumn } from 'typeorm';
|
import {
|
||||||
|
Column,
|
||||||
|
Entity,
|
||||||
|
ManyToOne,
|
||||||
|
PrimaryGeneratedColumn,
|
||||||
|
Unique,
|
||||||
|
} from 'typeorm';
|
||||||
import { User } from './User';
|
import { User } from './User';
|
||||||
|
|
||||||
@Entity()
|
@Entity()
|
||||||
|
@Unique(['endpoint', 'user'])
|
||||||
export class UserPushSubscription {
|
export class UserPushSubscription {
|
||||||
@PrimaryGeneratedColumn()
|
@PrimaryGeneratedColumn()
|
||||||
public id: number;
|
public id: number;
|
||||||
|
|||||||
@@ -24,6 +24,15 @@ interface PushNotificationPayload {
|
|||||||
isAdmin?: boolean;
|
isAdmin?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface WebPushError extends Error {
|
||||||
|
statusCode?: number;
|
||||||
|
status?: number;
|
||||||
|
body?: string | unknown;
|
||||||
|
response?: {
|
||||||
|
body?: string | unknown;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
class WebPushAgent
|
class WebPushAgent
|
||||||
extends BaseAgent<NotificationAgentConfig>
|
extends BaseAgent<NotificationAgentConfig>
|
||||||
implements NotificationAgent
|
implements NotificationAgent
|
||||||
@@ -188,19 +197,30 @@ class WebPushAgent
|
|||||||
notificationPayload
|
notificationPayload
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
const webPushError = e as WebPushError;
|
||||||
|
const statusCode = webPushError.statusCode || webPushError.status;
|
||||||
|
const errorMessage = webPushError.message || String(e);
|
||||||
|
|
||||||
|
// RFC 8030: 410/404 are permanent failures, others are transient
|
||||||
|
const isPermanentFailure = statusCode === 410 || statusCode === 404;
|
||||||
|
|
||||||
logger.error(
|
logger.error(
|
||||||
'Error sending web push notification; removing subscription',
|
isPermanentFailure
|
||||||
|
? 'Error sending web push notification; removing invalid subscription'
|
||||||
|
: 'Error sending web push notification (transient error, keeping subscription)',
|
||||||
{
|
{
|
||||||
label: 'Notifications',
|
label: 'Notifications',
|
||||||
recipient: pushSub.user.displayName,
|
recipient: pushSub.user.displayName,
|
||||||
type: Notification[type],
|
type: Notification[type],
|
||||||
subject: payload.subject,
|
subject: payload.subject,
|
||||||
errorMessage: e.message,
|
errorMessage,
|
||||||
|
statusCode: statusCode || 'unknown',
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
// Failed to send notification so we need to remove the subscription
|
if (isPermanentFailure) {
|
||||||
userPushSubRepository.remove(pushSub);
|
await userPushSubRepository.remove(pushSub);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -374,9 +374,10 @@ class JellyfinScanner {
|
|||||||
) ?? []
|
) ?? []
|
||||||
).length;
|
).length;
|
||||||
|
|
||||||
|
const jellyfinSeasons = await this.jfClient.getSeasons(Id);
|
||||||
|
|
||||||
for (const season of seasons) {
|
for (const season of seasons) {
|
||||||
const JellyfinSeasons = await this.jfClient.getSeasons(Id);
|
const matchedJellyfinSeason = jellyfinSeasons.find((md) => {
|
||||||
const matchedJellyfinSeason = JellyfinSeasons.find((md) => {
|
|
||||||
if (tvdbSeasonFromAnidb) {
|
if (tvdbSeasonFromAnidb) {
|
||||||
// In AniDB we don't have the concept of seasons,
|
// In AniDB we don't have the concept of seasons,
|
||||||
// we have multiple shows with only Season 1 (and sometimes a season with index 0 for specials).
|
// we have multiple shows with only Season 1 (and sometimes a season with index 0 for specials).
|
||||||
@@ -397,38 +398,52 @@ class JellyfinScanner {
|
|||||||
|
|
||||||
// Check if we found the matching season and it has all the available episodes
|
// Check if we found the matching season and it has all the available episodes
|
||||||
if (matchedJellyfinSeason) {
|
if (matchedJellyfinSeason) {
|
||||||
// If we have a matched Jellyfin season, get its children metadata so we can check details
|
|
||||||
const episodes = await this.jfClient.getEpisodes(
|
|
||||||
Id,
|
|
||||||
matchedJellyfinSeason.Id
|
|
||||||
);
|
|
||||||
|
|
||||||
//Get count of episodes that are HD and 4K
|
|
||||||
let totalStandard = 0;
|
let totalStandard = 0;
|
||||||
let total4k = 0;
|
let total4k = 0;
|
||||||
|
|
||||||
//use for loop to make sure this loop _completes_ in full
|
if (!this.enable4kShow) {
|
||||||
//before the next section
|
const episodes = await this.jfClient.getEpisodes(
|
||||||
for (const episode of episodes) {
|
Id,
|
||||||
let episodeCount = 1;
|
matchedJellyfinSeason.Id
|
||||||
|
);
|
||||||
|
|
||||||
// count number of combined episodes
|
for (const episode of episodes) {
|
||||||
if (
|
let episodeCount = 1;
|
||||||
episode.IndexNumber !== undefined &&
|
|
||||||
episode.IndexNumberEnd !== undefined
|
// count number of combined episodes
|
||||||
) {
|
if (
|
||||||
episodeCount =
|
episode.IndexNumber !== undefined &&
|
||||||
episode.IndexNumberEnd - episode.IndexNumber + 1;
|
episode.IndexNumberEnd !== undefined
|
||||||
}
|
) {
|
||||||
|
episodeCount =
|
||||||
|
episode.IndexNumberEnd - episode.IndexNumber + 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (!this.enable4kShow) {
|
|
||||||
totalStandard += episodeCount;
|
totalStandard += episodeCount;
|
||||||
} else {
|
}
|
||||||
const ExtendedEpisodeData = await this.jfClient.getItemData(
|
} else {
|
||||||
episode.Id
|
// 4K detection enabled - request media info to check resolution
|
||||||
);
|
const episodes = await this.jfClient.getEpisodes(
|
||||||
|
Id,
|
||||||
|
matchedJellyfinSeason.Id,
|
||||||
|
{ includeMediaInfo: true }
|
||||||
|
);
|
||||||
|
|
||||||
ExtendedEpisodeData?.MediaSources?.some((MediaSource) => {
|
for (const episode of episodes) {
|
||||||
|
let episodeCount = 1;
|
||||||
|
|
||||||
|
// count number of combined episodes
|
||||||
|
if (
|
||||||
|
episode.IndexNumber !== undefined &&
|
||||||
|
episode.IndexNumberEnd !== undefined
|
||||||
|
) {
|
||||||
|
episodeCount =
|
||||||
|
episode.IndexNumberEnd - episode.IndexNumber + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// MediaSources field is included in response when includeMediaInfo is true
|
||||||
|
// We iterate all MediaSources to detect if episode has both standard AND 4K versions
|
||||||
|
episode.MediaSources?.some((MediaSource) => {
|
||||||
return MediaSource.MediaStreams.some((MediaStream) => {
|
return MediaSource.MediaStreams.some((MediaStream) => {
|
||||||
if (MediaStream.Type === 'Video') {
|
if (MediaStream.Type === 'Video') {
|
||||||
if ((MediaStream.Width ?? 0) >= 2000) {
|
if ((MediaStream.Width ?? 0) >= 2000) {
|
||||||
|
|||||||
@@ -0,0 +1,19 @@
|
|||||||
|
import type { MigrationInterface, QueryRunner } from 'typeorm';
|
||||||
|
|
||||||
|
export class AddUniqueConstraintToPushSubscription1765233385034
|
||||||
|
implements MigrationInterface
|
||||||
|
{
|
||||||
|
name = 'AddUniqueConstraintToPushSubscription1765233385034';
|
||||||
|
|
||||||
|
public async up(queryRunner: QueryRunner): Promise<void> {
|
||||||
|
await queryRunner.query(
|
||||||
|
`ALTER TABLE "user_push_subscription" ADD CONSTRAINT "UQ_6427d07d9a171a3a1ab87480005" UNIQUE ("endpoint", "userId")`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async down(queryRunner: QueryRunner): Promise<void> {
|
||||||
|
await queryRunner.query(
|
||||||
|
`ALTER TABLE "user_push_subscription" DROP CONSTRAINT "UQ_6427d07d9a171a3a1ab87480005"`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import type { MigrationInterface, QueryRunner } from 'typeorm';
|
||||||
|
|
||||||
|
export class AddUniqueConstraintToPushSubscription1765233385034
|
||||||
|
implements MigrationInterface
|
||||||
|
{
|
||||||
|
name = 'AddUniqueConstraintToPushSubscription1765233385034';
|
||||||
|
|
||||||
|
public async up(queryRunner: QueryRunner): Promise<void> {
|
||||||
|
await queryRunner.query(
|
||||||
|
`CREATE UNIQUE INDEX "UQ_6427d07d9a171a3a1ab87480005" ON "user_push_subscription" ("endpoint", "userId")`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async down(queryRunner: QueryRunner): Promise<void> {
|
||||||
|
await queryRunner.query(`DROP INDEX "UQ_6427d07d9a171a3a1ab87480005"`);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -626,76 +626,6 @@ authRoutes.post('/local', async (req, res, next) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const mainUser = await userRepository.findOneOrFail({
|
|
||||||
select: { id: true, plexToken: true, plexId: true },
|
|
||||||
where: { id: 1 },
|
|
||||||
});
|
|
||||||
const mainPlexTv = new PlexTvAPI(mainUser.plexToken ?? '');
|
|
||||||
|
|
||||||
if (!user.plexId) {
|
|
||||||
try {
|
|
||||||
const plexUsersResponse = await mainPlexTv.getUsers();
|
|
||||||
const account = plexUsersResponse.MediaContainer.User.find(
|
|
||||||
(account) =>
|
|
||||||
account.$.email &&
|
|
||||||
account.$.email.toLowerCase() === user.email.toLowerCase()
|
|
||||||
)?.$;
|
|
||||||
|
|
||||||
if (
|
|
||||||
account &&
|
|
||||||
(await mainPlexTv.checkUserAccess(parseInt(account.id)))
|
|
||||||
) {
|
|
||||||
logger.info(
|
|
||||||
'Found matching Plex user; updating user with Plex data',
|
|
||||||
{
|
|
||||||
label: 'API',
|
|
||||||
ip: req.ip,
|
|
||||||
email: body.email,
|
|
||||||
userId: user.id,
|
|
||||||
plexId: account.id,
|
|
||||||
plexUsername: account.username,
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
user.plexId = parseInt(account.id);
|
|
||||||
user.avatar = account.thumb;
|
|
||||||
user.email = account.email;
|
|
||||||
user.plexUsername = account.username;
|
|
||||||
user.userType = UserType.PLEX;
|
|
||||||
|
|
||||||
await userRepository.save(user);
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
logger.error('Something went wrong fetching Plex users', {
|
|
||||||
label: 'API',
|
|
||||||
errorMessage: e.message,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
user.plexId &&
|
|
||||||
user.plexId !== mainUser.plexId &&
|
|
||||||
!(await mainPlexTv.checkUserAccess(user.plexId))
|
|
||||||
) {
|
|
||||||
logger.warn(
|
|
||||||
'Failed sign-in attempt from Plex user without access to the media server',
|
|
||||||
{
|
|
||||||
label: 'API',
|
|
||||||
account: {
|
|
||||||
ip: req.ip,
|
|
||||||
email: body.email,
|
|
||||||
userId: user.id,
|
|
||||||
plexId: user.plexId,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
);
|
|
||||||
return next({
|
|
||||||
status: 403,
|
|
||||||
message: 'Access denied.',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set logged in session
|
// Set logged in session
|
||||||
if (user && req.session) {
|
if (user && req.session) {
|
||||||
req.session.userId = user.id;
|
req.session.userId = user.id;
|
||||||
@@ -775,7 +705,7 @@ authRoutes.post('/logout', async (req, res, next) => {
|
|||||||
});
|
});
|
||||||
return next({ status: 500, message: 'Failed to destroy session.' });
|
return next({ status: 500, message: 'Failed to destroy session.' });
|
||||||
}
|
}
|
||||||
logger.info('Successfully logged out user', {
|
logger.debug('Successfully logged out user', {
|
||||||
label: 'Auth',
|
label: 'Auth',
|
||||||
userId,
|
userId,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import TautulliAPI from '@server/api/tautulli';
|
|||||||
import { MediaType } from '@server/constants/media';
|
import { MediaType } from '@server/constants/media';
|
||||||
import { MediaServerType } from '@server/constants/server';
|
import { MediaServerType } from '@server/constants/server';
|
||||||
import { UserType } from '@server/constants/user';
|
import { UserType } from '@server/constants/user';
|
||||||
import { getRepository } from '@server/datasource';
|
import dataSource, { getRepository } from '@server/datasource';
|
||||||
import Media from '@server/entity/Media';
|
import Media from '@server/entity/Media';
|
||||||
import { MediaRequest } from '@server/entity/MediaRequest';
|
import { MediaRequest } from '@server/entity/MediaRequest';
|
||||||
import { User } from '@server/entity/User';
|
import { User } from '@server/entity/User';
|
||||||
@@ -25,7 +25,8 @@ import { getHostname } from '@server/utils/getHostname';
|
|||||||
import { Router } from 'express';
|
import { Router } from 'express';
|
||||||
import gravatarUrl from 'gravatar-url';
|
import gravatarUrl from 'gravatar-url';
|
||||||
import { findIndex, sortBy } from 'lodash';
|
import { findIndex, sortBy } from 'lodash';
|
||||||
import { In } from 'typeorm';
|
import type { EntityManager } from 'typeorm';
|
||||||
|
import { In, Not } from 'typeorm';
|
||||||
import userSettingsRoutes from './usersettings';
|
import userSettingsRoutes from './usersettings';
|
||||||
|
|
||||||
const router = Router();
|
const router = Router();
|
||||||
@@ -188,30 +189,82 @@ router.post<
|
|||||||
}
|
}
|
||||||
>('/registerPushSubscription', async (req, res, next) => {
|
>('/registerPushSubscription', async (req, res, next) => {
|
||||||
try {
|
try {
|
||||||
const userPushSubRepository = getRepository(UserPushSubscription);
|
// This prevents race conditions where two requests both pass the checks
|
||||||
|
await dataSource.transaction(
|
||||||
|
async (transactionalEntityManager: EntityManager) => {
|
||||||
|
const transactionalRepo =
|
||||||
|
transactionalEntityManager.getRepository(UserPushSubscription);
|
||||||
|
|
||||||
const existingSubs = await userPushSubRepository.find({
|
// Check for existing subscription by auth or endpoint within transaction
|
||||||
relations: { user: true },
|
const existingSubscription = await transactionalRepo.findOne({
|
||||||
where: { auth: req.body.auth, user: { id: req.user?.id } },
|
relations: { user: true },
|
||||||
});
|
where: [
|
||||||
|
{ auth: req.body.auth, user: { id: req.user?.id } },
|
||||||
|
{ endpoint: req.body.endpoint, user: { id: req.user?.id } },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
if (existingSubs.length > 0) {
|
if (existingSubscription) {
|
||||||
logger.debug(
|
// If endpoint matches but auth is different, update with new keys (iOS refresh case)
|
||||||
'User push subscription already exists. Skipping registration.',
|
if (
|
||||||
{ label: 'API' }
|
existingSubscription.endpoint === req.body.endpoint &&
|
||||||
);
|
existingSubscription.auth !== req.body.auth
|
||||||
return res.status(204).send();
|
) {
|
||||||
}
|
existingSubscription.auth = req.body.auth;
|
||||||
|
existingSubscription.p256dh = req.body.p256dh;
|
||||||
|
existingSubscription.userAgent = req.body.userAgent;
|
||||||
|
|
||||||
const userPushSubscription = new UserPushSubscription({
|
await transactionalRepo.save(existingSubscription);
|
||||||
auth: req.body.auth,
|
|
||||||
endpoint: req.body.endpoint,
|
|
||||||
p256dh: req.body.p256dh,
|
|
||||||
userAgent: req.body.userAgent,
|
|
||||||
user: req.user,
|
|
||||||
});
|
|
||||||
|
|
||||||
userPushSubRepository.save(userPushSubscription);
|
logger.debug(
|
||||||
|
'Updated existing push subscription with new keys for same endpoint.',
|
||||||
|
{ label: 'API' }
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.debug(
|
||||||
|
'Duplicate subscription detected. Skipping registration.',
|
||||||
|
{ label: 'API' }
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clean up old subscriptions from the same device (userAgent) for this user
|
||||||
|
// iOS can silently refresh endpoints, leaving stale subscriptions in the database
|
||||||
|
// Only clean up if we're creating a new subscription (not updating an existing one)
|
||||||
|
if (req.body.userAgent) {
|
||||||
|
const staleSubscriptions = await transactionalRepo.find({
|
||||||
|
relations: { user: true },
|
||||||
|
where: {
|
||||||
|
userAgent: req.body.userAgent,
|
||||||
|
user: { id: req.user?.id },
|
||||||
|
// Only remove subscriptions with different endpoints (stale ones)
|
||||||
|
// Keep subscriptions that might be from different browsers/tabs
|
||||||
|
endpoint: Not(req.body.endpoint),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (staleSubscriptions.length > 0) {
|
||||||
|
await transactionalRepo.remove(staleSubscriptions);
|
||||||
|
logger.debug(
|
||||||
|
`Removed ${staleSubscriptions.length} stale push subscription(s) from same device.`,
|
||||||
|
{ label: 'API' }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const userPushSubscription = new UserPushSubscription({
|
||||||
|
auth: req.body.auth,
|
||||||
|
endpoint: req.body.endpoint,
|
||||||
|
p256dh: req.body.p256dh,
|
||||||
|
userAgent: req.body.userAgent,
|
||||||
|
user: req.user,
|
||||||
|
});
|
||||||
|
|
||||||
|
await transactionalRepo.save(userPushSubscription);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
return res.status(204).send();
|
return res.status(204).send();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ const LabeledCheckbox: React.FC<LabeledCheckboxProps> = ({
|
|||||||
<Field type="checkbox" id={id} name={id} onChange={onChange} />
|
<Field type="checkbox" id={id} name={id} onChange={onChange} />
|
||||||
</div>
|
</div>
|
||||||
<div className="ml-3 text-sm leading-6">
|
<div className="ml-3 text-sm leading-6">
|
||||||
<label htmlFor="localLogin" className="block">
|
<label htmlFor="localLogin" className="block" aria-label={label}>
|
||||||
<div className="flex flex-col">
|
<div className="flex flex-col">
|
||||||
<span className="font-medium text-white">{label}</span>
|
<span className="font-medium text-white">{label}</span>
|
||||||
<span className="font-normal text-gray-400">{description}</span>
|
<span className="font-normal text-gray-400">{description}</span>
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ const NotificationType = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="ml-3 text-sm leading-6">
|
<div className="ml-3 text-sm leading-6">
|
||||||
<label htmlFor={option.id} className="block">
|
<label htmlFor={option.id} className="block" aria-label={option.name}>
|
||||||
<div className="flex flex-col">
|
<div className="flex flex-col">
|
||||||
<span className="font-medium text-white">{option.name}</span>
|
<span className="font-medium text-white">{option.name}</span>
|
||||||
<span className="font-normal text-gray-400">
|
<span className="font-normal text-gray-400">
|
||||||
|
|||||||
@@ -123,7 +123,7 @@ const PermissionOption = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="ml-3 text-sm leading-6">
|
<div className="ml-3 text-sm leading-6">
|
||||||
<label htmlFor={option.id} className="block">
|
<label htmlFor={option.id} className="block" aria-label={option.name}>
|
||||||
<div className="flex flex-col">
|
<div className="flex flex-col">
|
||||||
<span className="font-medium text-white">{option.name}</span>
|
<span className="font-medium text-white">{option.name}</span>
|
||||||
<span className="font-normal text-gray-400">
|
<span className="font-normal text-gray-400">
|
||||||
|
|||||||
@@ -109,15 +109,28 @@ const UserWebPushSettings = () => {
|
|||||||
// Deletes/disables corresponding push subscription from database
|
// Deletes/disables corresponding push subscription from database
|
||||||
const disablePushNotifications = async (endpoint?: string) => {
|
const disablePushNotifications = async (endpoint?: string) => {
|
||||||
try {
|
try {
|
||||||
await unsubscribeToPushNotifications(user?.id, endpoint);
|
const unsubscribedEndpoint = await unsubscribeToPushNotifications(
|
||||||
|
user?.id,
|
||||||
// Delete from backend if endpoint is available
|
endpoint
|
||||||
if (subEndpoint) {
|
);
|
||||||
await deletePushSubscriptionFromBackend(subEndpoint);
|
|
||||||
}
|
|
||||||
|
|
||||||
localStorage.setItem('pushNotificationsEnabled', 'false');
|
localStorage.setItem('pushNotificationsEnabled', 'false');
|
||||||
setWebPushEnabled(false);
|
setWebPushEnabled(false);
|
||||||
|
|
||||||
|
// Only delete the current browser's subscription, not all devices
|
||||||
|
const endpointToDelete = unsubscribedEndpoint || subEndpoint || endpoint;
|
||||||
|
if (endpointToDelete) {
|
||||||
|
try {
|
||||||
|
await axios.delete(
|
||||||
|
`/api/v1/user/${user?.id}/pushSubscription/${encodeURIComponent(
|
||||||
|
endpointToDelete
|
||||||
|
)}`
|
||||||
|
);
|
||||||
|
} catch {
|
||||||
|
// Ignore deletion failures - backend cleanup is best effort
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
addToast(intl.formatMessage(messages.webpushhasbeendisabled), {
|
addToast(intl.formatMessage(messages.webpushhasbeendisabled), {
|
||||||
autoDismiss: true,
|
autoDismiss: true,
|
||||||
appearance: 'success',
|
appearance: 'success',
|
||||||
@@ -157,7 +170,33 @@ const UserWebPushSettings = () => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const verifyWebPush = async () => {
|
const verifyWebPush = async () => {
|
||||||
const enabled = await verifyPushSubscription(user?.id, currentSettings);
|
const enabled = await verifyPushSubscription(user?.id, currentSettings);
|
||||||
setWebPushEnabled(enabled);
|
let isEnabled = enabled;
|
||||||
|
|
||||||
|
if (!enabled && 'serviceWorker' in navigator) {
|
||||||
|
const { subscription } = await getPushSubscription();
|
||||||
|
if (subscription) {
|
||||||
|
isEnabled = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isEnabled && dataDevices && dataDevices.length > 0) {
|
||||||
|
const currentUserAgent = navigator.userAgent;
|
||||||
|
const hasMatchingDevice = dataDevices.some(
|
||||||
|
(device) => device.userAgent === currentUserAgent
|
||||||
|
);
|
||||||
|
|
||||||
|
if (hasMatchingDevice) {
|
||||||
|
isEnabled = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setWebPushEnabled(isEnabled);
|
||||||
|
if (localStorage.getItem('pushNotificationsEnabled') === null) {
|
||||||
|
localStorage.setItem(
|
||||||
|
'pushNotificationsEnabled',
|
||||||
|
isEnabled ? 'true' : 'false'
|
||||||
|
);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (user?.id) {
|
if (user?.id) {
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { UserType } from '@server/constants/user';
|
|||||||
import type { PermissionCheckOptions } from '@server/lib/permissions';
|
import type { PermissionCheckOptions } from '@server/lib/permissions';
|
||||||
import { hasPermission, Permission } from '@server/lib/permissions';
|
import { hasPermission, Permission } from '@server/lib/permissions';
|
||||||
import type { NotificationAgentKey } from '@server/lib/settings';
|
import type { NotificationAgentKey } from '@server/lib/settings';
|
||||||
|
import { useRouter } from 'next/router';
|
||||||
import type { MutatorCallback } from 'swr';
|
import type { MutatorCallback } from 'swr';
|
||||||
import useSWR from 'swr';
|
import useSWR from 'swr';
|
||||||
|
|
||||||
@@ -56,13 +57,21 @@ export const useUser = ({
|
|||||||
id,
|
id,
|
||||||
initialData,
|
initialData,
|
||||||
}: { id?: number; initialData?: User } = {}): UserHookResponse => {
|
}: { id?: number; initialData?: User } = {}): UserHookResponse => {
|
||||||
|
const router = useRouter();
|
||||||
|
const isAuthPage = /^\/(login|setup|resetpassword(?:\/|$))/.test(
|
||||||
|
router.pathname
|
||||||
|
);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
data,
|
data,
|
||||||
error,
|
error,
|
||||||
mutate: revalidate,
|
mutate: revalidate,
|
||||||
} = useSWR<User>(id ? `/api/v1/user/${id}` : `/api/v1/auth/me`, {
|
} = useSWR<User>(id ? `/api/v1/user/${id}` : `/api/v1/auth/me`, {
|
||||||
fallbackData: initialData,
|
fallbackData: initialData,
|
||||||
refreshInterval: 30000,
|
refreshInterval: !isAuthPage ? 30000 : 0,
|
||||||
|
revalidateOnFocus: !isAuthPage,
|
||||||
|
revalidateOnMount: !isAuthPage,
|
||||||
|
revalidateOnReconnect: !isAuthPage,
|
||||||
errorRetryInterval: 30000,
|
errorRetryInterval: 30000,
|
||||||
shouldRetryOnError: false,
|
shouldRetryOnError: false,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ class PlexOAuth {
|
|||||||
'X-Plex-Client-Identifier': clientId,
|
'X-Plex-Client-Identifier': clientId,
|
||||||
'X-Plex-Model': 'Plex OAuth',
|
'X-Plex-Model': 'Plex OAuth',
|
||||||
'X-Plex-Platform': browser.getBrowserName(),
|
'X-Plex-Platform': browser.getBrowserName(),
|
||||||
'X-Plex-Platform-Version': browser.getBrowserVersion(),
|
'X-Plex-Platform-Version': browser.getBrowserVersion() || 'Unknown',
|
||||||
'X-Plex-Device': browser.getOSName(),
|
'X-Plex-Device': browser.getOSName(),
|
||||||
'X-Plex-Device-Name': `${browser.getBrowserName()} (Seerr)`,
|
'X-Plex-Device-Name': `${browser.getBrowserName()} (Seerr)`,
|
||||||
'X-Plex-Device-Screen-Resolution':
|
'X-Plex-Device-Screen-Resolution':
|
||||||
|
|||||||
@@ -49,13 +49,17 @@ export const verifyPushSubscription = async (
|
|||||||
currentSettings.vapidPublic
|
currentSettings.vapidPublic
|
||||||
).toString();
|
).toString();
|
||||||
|
|
||||||
|
if (currentServerKey !== expectedServerKey) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
const endpoint = subscription.endpoint;
|
const endpoint = subscription.endpoint;
|
||||||
|
|
||||||
const { data } = await axios.get<UserPushSubscription>(
|
const { data } = await axios.get<UserPushSubscription>(
|
||||||
`/api/v1/user/${userId}/pushSubscription/${encodeURIComponent(endpoint)}`
|
`/api/v1/user/${userId}/pushSubscription/${encodeURIComponent(endpoint)}`
|
||||||
);
|
);
|
||||||
|
|
||||||
return expectedServerKey === currentServerKey && data.endpoint === endpoint;
|
return data.endpoint === endpoint;
|
||||||
} catch {
|
} catch {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -65,20 +69,39 @@ export const verifyAndResubscribePushSubscription = async (
|
|||||||
userId: number | undefined,
|
userId: number | undefined,
|
||||||
currentSettings: PublicSettingsResponse
|
currentSettings: PublicSettingsResponse
|
||||||
): Promise<boolean> => {
|
): Promise<boolean> => {
|
||||||
|
if (!userId) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { subscription } = await getPushSubscription();
|
||||||
const isValid = await verifyPushSubscription(userId, currentSettings);
|
const isValid = await verifyPushSubscription(userId, currentSettings);
|
||||||
|
|
||||||
if (isValid) {
|
if (isValid) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (subscription) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (currentSettings.enablePushRegistration) {
|
if (currentSettings.enablePushRegistration) {
|
||||||
try {
|
try {
|
||||||
// Unsubscribe from the backend to clear the existing push subscription (keys and endpoint)
|
const oldEndpoint = await unsubscribeToPushNotifications(userId);
|
||||||
await unsubscribeToPushNotifications(userId);
|
|
||||||
|
|
||||||
// Subscribe again to generate a fresh push subscription with updated keys and endpoint
|
|
||||||
await subscribeToPushNotifications(userId, currentSettings);
|
await subscribeToPushNotifications(userId, currentSettings);
|
||||||
|
|
||||||
|
if (oldEndpoint) {
|
||||||
|
try {
|
||||||
|
await axios.delete(
|
||||||
|
`/api/v1/user/${userId}/pushSubscription/${encodeURIComponent(
|
||||||
|
oldEndpoint
|
||||||
|
)}`
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
// Ignore errors when deleting old endpoint (it might not exist)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new Error(`[SW] Resubscribe failed: ${error.message}`);
|
throw new Error(`[SW] Resubscribe failed: ${error.message}`);
|
||||||
@@ -136,24 +159,26 @@ export const subscribeToPushNotifications = async (
|
|||||||
export const unsubscribeToPushNotifications = async (
|
export const unsubscribeToPushNotifications = async (
|
||||||
userId: number | undefined,
|
userId: number | undefined,
|
||||||
endpoint?: string
|
endpoint?: string
|
||||||
) => {
|
): Promise<string | null> => {
|
||||||
if (!('serviceWorker' in navigator) || !userId) {
|
if (!('serviceWorker' in navigator) || !userId) {
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { subscription } = await getPushSubscription();
|
const { subscription } = await getPushSubscription();
|
||||||
|
|
||||||
if (!subscription) {
|
if (!subscription) {
|
||||||
return false;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { endpoint: currentEndpoint } = subscription.toJSON();
|
const { endpoint: currentEndpoint } = subscription.toJSON();
|
||||||
|
|
||||||
if (!endpoint || endpoint === currentEndpoint) {
|
if (!endpoint || endpoint === currentEndpoint) {
|
||||||
await subscription.unsubscribe();
|
await subscription.unsubscribe();
|
||||||
return true;
|
return currentEndpoint ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
`Issue unsubscribing to push notifications: ${error.message}`
|
`Issue unsubscribing to push notifications: ${error.message}`
|
||||||
|
|||||||
Reference in New Issue
Block a user