Merge branch 'develop'
This commit is contained in:
1
.github/workflows/ci.yml
vendored
1
.github/workflows/ci.yml
vendored
@@ -63,6 +63,7 @@ jobs:
|
|||||||
discord:
|
discord:
|
||||||
name: Send Discord Notification
|
name: Send Discord Notification
|
||||||
needs: build_and_push
|
needs: build_and_push
|
||||||
|
if: always() && github.event_name != 'pull_request' && !contains(github.event.head_commit.message, '[skip ci]')
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
steps:
|
steps:
|
||||||
- name: Get Build Job Status
|
- name: Get Build Job Status
|
||||||
|
|||||||
53
.github/workflows/release.yml
vendored
53
.github/workflows/release.yml
vendored
@@ -7,6 +7,7 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
test:
|
test:
|
||||||
|
name: Lint & Test Build
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
container: node:12.18-alpine
|
container: node:12.18-alpine
|
||||||
steps:
|
steps:
|
||||||
@@ -41,9 +42,61 @@ jobs:
|
|||||||
DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }}
|
DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }}
|
||||||
DOCKER_PASSWORD: ${{ secrets.DOCKER_PASSWORD }}
|
DOCKER_PASSWORD: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
run: npx semantic-release
|
run: npx semantic-release
|
||||||
|
build-snap:
|
||||||
|
name: Build Snap Package (${{ matrix.architecture }})
|
||||||
|
needs: semantic-release
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
architecture:
|
||||||
|
- amd64
|
||||||
|
- arm64
|
||||||
|
- armhf
|
||||||
|
steps:
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Prepare
|
||||||
|
id: prepare
|
||||||
|
run: |
|
||||||
|
git fetch --prune --unshallow --tags
|
||||||
|
if [[ $GITHUB_REF == refs/tags/* || $GITHUB_REF == refs/heads/master ]]; then
|
||||||
|
echo ::set-output name=RELEASE::stable
|
||||||
|
else
|
||||||
|
echo ::set-output name=RELEASE::edge
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Set Up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v1
|
||||||
|
|
||||||
|
- name: Build Snap Package
|
||||||
|
uses: diddlesnaps/snapcraft-multiarch-action@v1
|
||||||
|
id: build
|
||||||
|
with:
|
||||||
|
architecture: ${{ matrix.architecture }}
|
||||||
|
|
||||||
|
- name: Upload Snap Package
|
||||||
|
uses: actions/upload-artifact@v2
|
||||||
|
with:
|
||||||
|
name: overseerr-snap-package-${{ matrix.architecture }}
|
||||||
|
path: ${{ steps.build.outputs.snap }}
|
||||||
|
|
||||||
|
- name: Review Snap Package
|
||||||
|
uses: diddlesnaps/snapcraft-review-tools-action@v1
|
||||||
|
with:
|
||||||
|
snap: ${{ steps.build.outputs.snap }}
|
||||||
|
|
||||||
|
- name: Publish Snap Package
|
||||||
|
uses: snapcore/action-publish@v1
|
||||||
|
with:
|
||||||
|
store_login: ${{ secrets.SNAP_LOGIN }}
|
||||||
|
snap: ${{ steps.build.outputs.snap }}
|
||||||
|
release: ${{ steps.prepare.outputs.RELEASE }}
|
||||||
discord:
|
discord:
|
||||||
name: Send Discord Notification
|
name: Send Discord Notification
|
||||||
needs: semantic-release
|
needs: semantic-release
|
||||||
|
if: always()
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
steps:
|
steps:
|
||||||
- name: Get Build Job Status
|
- name: Get Build Job Status
|
||||||
|
|||||||
4
.github/workflows/snap.yaml
vendored
4
.github/workflows/snap.yaml
vendored
@@ -3,8 +3,6 @@ name: Publish Snap
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: [develop]
|
branches: [develop]
|
||||||
tags: [v*]
|
|
||||||
pull_request: ~
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
test:
|
test:
|
||||||
@@ -74,10 +72,10 @@ jobs:
|
|||||||
store_login: ${{ secrets.SNAP_LOGIN }}
|
store_login: ${{ secrets.SNAP_LOGIN }}
|
||||||
snap: ${{ steps.build.outputs.snap }}
|
snap: ${{ steps.build.outputs.snap }}
|
||||||
release: ${{ steps.prepare.outputs.RELEASE }}
|
release: ${{ steps.prepare.outputs.RELEASE }}
|
||||||
|
|
||||||
discord:
|
discord:
|
||||||
name: Send Discord Notification
|
name: Send Discord Notification
|
||||||
needs: build-snap
|
needs: build-snap
|
||||||
|
if: always() && !contains(github.event.head_commit.message, '[skip ci]')
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
steps:
|
steps:
|
||||||
- name: Get Build Job Status
|
- name: Get Build Job Status
|
||||||
|
|||||||
@@ -151,6 +151,15 @@ export class MediaRequest {
|
|||||||
logger.error('No parent media!', { label: 'Media Request' });
|
logger.error('No parent media!', { label: 'Media Request' });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (media[this.is4k ? 'status4k' : 'status'] === MediaStatus.AVAILABLE) {
|
||||||
|
logger.warn(
|
||||||
|
'Media became available before request was approved. Approval notification will be skipped.',
|
||||||
|
{ label: 'Media Request' }
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const tmdb = new TheMovieDb();
|
const tmdb = new TheMovieDb();
|
||||||
if (this.media.mediaType === MediaType.MOVIE) {
|
if (this.media.mediaType === MediaType.MOVIE) {
|
||||||
const movie = await tmdb.getMovie({ movieId: this.media.tmdbId });
|
const movie = await tmdb.getMovie({ movieId: this.media.tmdbId });
|
||||||
@@ -205,7 +214,13 @@ export class MediaRequest {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const seasonRequestRepository = getRepository(SeasonRequest);
|
const seasonRequestRepository = getRepository(SeasonRequest);
|
||||||
if (this.status === MediaRequestStatus.APPROVED) {
|
if (
|
||||||
|
this.status === MediaRequestStatus.APPROVED &&
|
||||||
|
// Do not update the status if the item is already partially available or available
|
||||||
|
media[this.is4k ? 'status4k' : 'status'] !== MediaStatus.AVAILABLE &&
|
||||||
|
media[this.is4k ? 'status4k' : 'status'] !==
|
||||||
|
MediaStatus.PARTIALLY_AVAILABLE
|
||||||
|
) {
|
||||||
if (this.is4k) {
|
if (this.is4k) {
|
||||||
media.status4k = MediaStatus.PROCESSING;
|
media.status4k = MediaStatus.PROCESSING;
|
||||||
} else {
|
} else {
|
||||||
@@ -358,6 +373,21 @@ export class MediaRequest {
|
|||||||
});
|
});
|
||||||
const movie = await tmdb.getMovie({ movieId: this.media.tmdbId });
|
const movie = await tmdb.getMovie({ movieId: this.media.tmdbId });
|
||||||
|
|
||||||
|
const media = await mediaRepository.findOne({
|
||||||
|
where: { id: this.media.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!media) {
|
||||||
|
logger.error('Media not present');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
media[this.is4k ? 'status4k' : 'status'] === MediaStatus.AVAILABLE
|
||||||
|
) {
|
||||||
|
throw new Error('Media already available');
|
||||||
|
}
|
||||||
|
|
||||||
// Run this asynchronously so we don't wait for it on the UI side
|
// Run this asynchronously so we don't wait for it on the UI side
|
||||||
radarr
|
radarr
|
||||||
.addMovie({
|
.addMovie({
|
||||||
@@ -373,13 +403,6 @@ export class MediaRequest {
|
|||||||
})
|
})
|
||||||
.then(async (success) => {
|
.then(async (success) => {
|
||||||
if (!success) {
|
if (!success) {
|
||||||
const media = await mediaRepository.findOne({
|
|
||||||
where: { id: this.media.id },
|
|
||||||
});
|
|
||||||
if (!media) {
|
|
||||||
logger.error('Media not present');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
media.status = MediaStatus.UNKNOWN;
|
media.status = MediaStatus.UNKNOWN;
|
||||||
await mediaRepository.save(media);
|
await mediaRepository.save(media);
|
||||||
logger.warn(
|
logger.warn(
|
||||||
@@ -464,6 +487,12 @@ export class MediaRequest {
|
|||||||
throw new Error('Media data is missing');
|
throw new Error('Media data is missing');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
media[this.is4k ? 'status4k' : 'status'] === MediaStatus.AVAILABLE
|
||||||
|
) {
|
||||||
|
throw new Error('Media already available');
|
||||||
|
}
|
||||||
|
|
||||||
const tmdb = new TheMovieDb();
|
const tmdb = new TheMovieDb();
|
||||||
const sonarr = new SonarrAPI({
|
const sonarr = new SonarrAPI({
|
||||||
apiKey: sonarrSettings.apiKey,
|
apiKey: sonarrSettings.apiKey,
|
||||||
|
|||||||
@@ -160,7 +160,11 @@ requestRoutes.post(
|
|||||||
// (Unless there are no seasons, in which case we abort)
|
// (Unless there are no seasons, in which case we abort)
|
||||||
if (media.requests) {
|
if (media.requests) {
|
||||||
existingSeasons = media.requests
|
existingSeasons = media.requests
|
||||||
.filter((request) => request.is4k === req.body.is4k)
|
.filter(
|
||||||
|
(request) =>
|
||||||
|
request.is4k === req.body.is4k &&
|
||||||
|
request.status !== MediaRequestStatus.DECLINED
|
||||||
|
)
|
||||||
.reduce((seasons, request) => {
|
.reduce((seasons, request) => {
|
||||||
const combinedSeasons = request.seasons.map(
|
const combinedSeasons = request.seasons.map(
|
||||||
(season) => season.seasonNumber
|
(season) => season.seasonNumber
|
||||||
@@ -303,7 +307,12 @@ requestRoutes.put<{ requestId: string }>(
|
|||||||
|
|
||||||
// Get all requested seasons that are not part of this request we are editing
|
// Get all requested seasons that are not part of this request we are editing
|
||||||
const existingSeasons = media.requests
|
const existingSeasons = media.requests
|
||||||
.filter((r) => r.is4k === request.is4k && r.id !== request.id)
|
.filter(
|
||||||
|
(r) =>
|
||||||
|
r.is4k === request.is4k &&
|
||||||
|
r.id !== request.id &&
|
||||||
|
r.status !== MediaRequestStatus.DECLINED
|
||||||
|
)
|
||||||
.reduce((seasons, r) => {
|
.reduce((seasons, r) => {
|
||||||
const combinedSeasons = r.seasons.map(
|
const combinedSeasons = r.seasons.map(
|
||||||
(season) => season.seasonNumber
|
(season) => season.seasonNumber
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import {
|
|||||||
UpdateEvent,
|
UpdateEvent,
|
||||||
} from 'typeorm';
|
} from 'typeorm';
|
||||||
import TheMovieDb from '../api/themoviedb';
|
import TheMovieDb from '../api/themoviedb';
|
||||||
import { MediaStatus, MediaType } from '../constants/media';
|
import { MediaRequestStatus, MediaStatus, MediaType } from '../constants/media';
|
||||||
import Media from '../entity/Media';
|
import Media from '../entity/Media';
|
||||||
import { MediaRequest } from '../entity/MediaRequest';
|
import { MediaRequest } from '../entity/MediaRequest';
|
||||||
import Season from '../entity/Season';
|
import Season from '../entity/Season';
|
||||||
@@ -102,6 +102,24 @@ export class MediaSubscriber implements EntitySubscriberInterface {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async updateChildRequestStatus(event: Media, is4k: boolean) {
|
||||||
|
const requestRepository = getRepository(MediaRequest);
|
||||||
|
|
||||||
|
const requests = await requestRepository.find({
|
||||||
|
where: { media: event.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const request of requests) {
|
||||||
|
if (
|
||||||
|
request.is4k === is4k &&
|
||||||
|
request.status === MediaRequestStatus.PENDING
|
||||||
|
) {
|
||||||
|
request.status = MediaRequestStatus.APPROVED;
|
||||||
|
await requestRepository.save(request);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public beforeUpdate(event: UpdateEvent<Media>): void {
|
public beforeUpdate(event: UpdateEvent<Media>): void {
|
||||||
if (!event.entity) {
|
if (!event.entity) {
|
||||||
return;
|
return;
|
||||||
@@ -121,5 +139,19 @@ export class MediaSubscriber implements EntitySubscriberInterface {
|
|||||||
) {
|
) {
|
||||||
this.notifyAvailableSeries(event.entity, event.databaseEntity);
|
this.notifyAvailableSeries(event.entity, event.databaseEntity);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
event.entity.status === MediaStatus.AVAILABLE &&
|
||||||
|
event.databaseEntity.status === MediaStatus.PENDING
|
||||||
|
) {
|
||||||
|
this.updateChildRequestStatus(event.entity, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
event.entity.status4k === MediaStatus.AVAILABLE &&
|
||||||
|
event.databaseEntity.status4k === MediaStatus.PENDING
|
||||||
|
) {
|
||||||
|
this.updateChildRequestStatus(event.entity, true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -169,7 +169,11 @@ const TvRequestModal: React.FC<RequestModalProps> = ({
|
|||||||
|
|
||||||
const getAllRequestedSeasons = (): number[] => {
|
const getAllRequestedSeasons = (): number[] => {
|
||||||
const requestedSeasons = (data?.mediaInfo?.requests ?? [])
|
const requestedSeasons = (data?.mediaInfo?.requests ?? [])
|
||||||
.filter((request) => request.is4k === is4k)
|
.filter(
|
||||||
|
(request) =>
|
||||||
|
request.is4k === is4k &&
|
||||||
|
request.status !== MediaRequestStatus.DECLINED
|
||||||
|
)
|
||||||
.reduce((requestedSeasons, request) => {
|
.reduce((requestedSeasons, request) => {
|
||||||
return [
|
return [
|
||||||
...requestedSeasons,
|
...requestedSeasons,
|
||||||
@@ -255,8 +259,11 @@ const TvRequestModal: React.FC<RequestModalProps> = ({
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
data?.mediaInfo &&
|
data?.mediaInfo &&
|
||||||
(data.mediaInfo.requests || []).filter((request) => request.is4k === is4k)
|
(data.mediaInfo.requests || []).filter(
|
||||||
.length > 0
|
(request) =>
|
||||||
|
request.is4k === is4k &&
|
||||||
|
request.status !== MediaRequestStatus.DECLINED
|
||||||
|
).length > 0
|
||||||
) {
|
) {
|
||||||
data.mediaInfo.requests
|
data.mediaInfo.requests
|
||||||
.filter((request) => request.is4k === is4k)
|
.filter((request) => request.is4k === is4k)
|
||||||
|
|||||||
Reference in New Issue
Block a user