Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Channel sync #5135

Merged
merged 71 commits into from
Aug 10, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
71 commits
Select commit Hold shift + click to select a range
dafcc07
Add external channel URL for channel update / creation (#754)
fflorent Mar 28, 2022
3d9cdf2
Disallow synchronisation if user has no video quota (#754)
fflorent Mar 28, 2022
5295073
More constraints serverside (#754)
fflorent Mar 28, 2022
2899d4a
Disable sync if server configuration does not allow HTTP import (#754)
fflorent Apr 6, 2022
d78fcc2
Working version synchronizing videos with a job (#754)
fflorent Apr 8, 2022
bd0a4b3
More logs and try/catch (#754)
fflorent Apr 11, 2022
1b7448b
Fix eslint error (#754)
fflorent Apr 11, 2022
6ac7dfd
WIP: support synchronization time change (#754)
fflorent Apr 15, 2022
1cd8538
New frontend #754
fflorent May 15, 2022
60dd68a
WIP: Create sync front (#754)
fflorent Jul 5, 2022
38b568f
Enhance UI, sync creation form (#754)
fflorent Jul 6, 2022
18c5316
Warning message when HTTP upload is disallowed
fflorent Jul 6, 2022
9c5d8a3
More consistent names (#754)
fflorent Jul 6, 2022
7d1f63c
Binding Front with API (#754)
fflorent Jul 13, 2022
beb0bb1
Add a /me API (#754)
fflorent Jul 13, 2022
1a25f15
Improve list UI (#754)
fflorent Jul 13, 2022
7c07eea
Implement creation and deletion routes (#754)
fflorent Jul 14, 2022
f470a38
Lint (#754)
fflorent Jul 14, 2022
9fa21bf
Lint again (#754)
fflorent Jul 14, 2022
0b6d378
WIP: UI for triggering import existing videos (#754)
fflorent Jul 15, 2022
a65364f
Implement jobs for syncing and importing channels
fflorent Jul 17, 2022
54a8995
Don't sync videos before sync creation + avoid concurrency issue (#754)
fflorent Jul 17, 2022
22864f2
Cleanup (#754)
fflorent Jul 19, 2022
9d53de6
Cleanup: OpenAPI + API rework (#754)
fflorent Jul 20, 2022
1e29d34
Remove dead code (#754)
fflorent Jul 20, 2022
4c9ebbc
Eslint (#754)
fflorent Jul 20, 2022
df220b5
Revert the mess with whitespaces in constants.ts (#754)
fflorent Jul 25, 2022
80d7a88
Some fixes after rebase (#754)
fflorent Jul 25, 2022
3e11cf5
Several fixes after PR remarks (#754)
fflorent Jul 26, 2022
11e7b7b
Front + API: Rename video-channels-sync to video-channel-syncs (#754)
fflorent Jul 26, 2022
e250276
Allow enabling channel sync through UI (#754)
fflorent Jul 27, 2022
518f4e0
getChannelInfo (#754)
fflorent Jul 28, 2022
1d148d5
Minor fixes: openapi + model + sql (#754)
fflorent Jul 28, 2022
6c7a269
Simplified API validators (#754)
fflorent Jul 28, 2022
242f827
Rename MChannelSync to MChannelSyncChannel (#754)
fflorent Jul 29, 2022
1970660
Add command for VideoChannelSync (#754)
fflorent Jul 29, 2022
c89c40c
Use synchronization.enabled config (#754)
fflorent Aug 1, 2022
ec21cc3
Check parameters test + some fixes (#754)
fflorent Aug 1, 2022
a1cc50c
Fix conflict mistake (#754)
fflorent Aug 1, 2022
63664bf
Restrict access to video channel sync list API (#754)
fflorent Aug 1, 2022
f1e7276
Start adding unit test for synchronization (#754)
fflorent Aug 1, 2022
a14e312
Continue testing (#754)
fflorent Aug 2, 2022
720f865
Tests finished + convertion of job to scheduler (#754)
fflorent Aug 2, 2022
d5806ae
Add lastSyncAt field (#754)
fflorent Aug 2, 2022
c79ceb9
Fix externalRemoteUrl sort + creation date not well formatted (#754)
fflorent Aug 3, 2022
9e44753
Small fix (#754)
fflorent Aug 3, 2022
203c729
Factorize addYoutubeDLImport and buildVideo (#754)
fflorent Aug 3, 2022
1b5cb11
Check duplicates on channel not on users (#754)
fflorent Aug 3, 2022
b2f35a6
factorize thumbnail generation (#754)
fflorent Aug 4, 2022
0728c5d
Fetch error should return status 400 (#754)
fflorent Aug 4, 2022
4e1bff0
Separate video-channel-import and video-channel-sync-latest (#754)
fflorent Aug 4, 2022
bc405b0
Bump DB migration version after rebase (#754)
fflorent Aug 4, 2022
a9a3848
Prettier states in UI table (#754)
fflorent Aug 4, 2022
9bd1226
Add DefaultScope in VideoChannelSyncModel (#754)
fflorent Aug 4, 2022
7e9c769
Fix audit logs (#754)
fflorent Aug 4, 2022
926379d
Ensure user can upload when importing channel + minor fixes (#754)
fflorent Aug 4, 2022
4f023e6
Mark synchronization as failed on exception + typos (#754)
fflorent Aug 5, 2022
9dc5088
Change REST API for importing videos into channel (#754)
fflorent Aug 5, 2022
c8ab2bc
Add option for fully synchronize a chnanel (#754)
fflorent Aug 5, 2022
81e8e66
Return a whole sync object on creation to avoid tricks in Front (#754)
fflorent Aug 5, 2022
f76b3b1
Various remarks (#754)
fflorent Aug 6, 2022
019fdac
Single quotes by default (#754)
fflorent Aug 6, 2022
84ec5e6
Rename synchronization to video_channel_synchronization
fflorent Aug 6, 2022
b30054a
Add check.latest_videos_count and max_per_user options (#754)
fflorent Aug 6, 2022
29254d2
Better channel rendering in list #754
fflorent Aug 6, 2022
78cef0e
Allow sorting with channel name and state (#754)
fflorent Aug 8, 2022
47f4722
Add missing tests for channel imports (#754)
fflorent Aug 8, 2022
c203b31
Merge branch 'develop' into channel-sync
Chocobozzz Aug 9, 2022
555695a
Prefer using a parent job for channel sync
Chocobozzz Aug 9, 2022
75671b5
Styling
Chocobozzz Aug 9, 2022
770cf55
Client styling
Chocobozzz Aug 10, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Add check.latest_videos_count and max_per_user options (#754)
  • Loading branch information
fflorent committed Aug 8, 2022
commit b30054aa4bd0c9216f0e92e2d62a56e9d0528ffd
6 changes: 5 additions & 1 deletion config/default.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -541,7 +541,11 @@ import:
enabled: false
video_channel_synchronization:
enabled: false
check_interval: 1 hour
max_per_user: 10
check:
interval: 1 hour
# Number of latest published videos to check and to potentially import
latest_videos_count: 3

auto_blacklist:
# New videos automatically blacklisted so moderators can review before publishing
Expand Down
5 changes: 4 additions & 1 deletion config/dev.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,10 @@ import:
enabled: true
video_channel_synchronization:
enabled: true
check_interval: 5 minutes
max_per_user: 10
check:
interval: 5 minutes
latest_videos_count: 3 # Number of latest published videos to check and to potentially import

instance:
default_nsfw_policy: 'display'
Expand Down
11 changes: 11 additions & 0 deletions config/production.yaml.example
Original file line number Diff line number Diff line change
Expand Up @@ -549,6 +549,17 @@ import:
# We recommend to only enable magnet URI/torrent import if you trust your users
# See https://docs.joinpeertube.org/maintain-configuration?id=security for more information
enabled: false
# Add ability for your users to synchronize their channels with external channels, playlists, etc.
video_channel_synchronization:
# If you enable the channel synchronization, ensure to enable the video import through HTTP (import.videos.http.enabled)
enabled: false
# The number of synchronizations a user can create
max_per_user: 10
check:
# The check interval during which PeerTube imports the new videos published remotely
interval: 1 hour
# Number of latest published videos to check and to potentially import
latest_videos_count: 3

auto_blacklist:
# New videos automatically blacklisted so moderators can review before publishing
Expand Down
3 changes: 2 additions & 1 deletion server/controllers/api/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -273,7 +273,8 @@ function customConfig (): CustomConfig {
}
},
videoChannelSynchronization: {
enabled: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.ENABLED
enabled: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.ENABLED,
maxPerUser: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.MAX_PER_USER
}
},
trending: {
Expand Down
6 changes: 3 additions & 3 deletions server/helpers/youtube-dl/youtube-dl-cli.ts
Original file line number Diff line number Diff line change
Expand Up @@ -123,12 +123,12 @@ export class YoutubeDLCLI {

getChannelInfo (options: {
channelUrl: string
lastVideosCount?: number
latestVideosCount?: number
processOptions: execa.NodeOptions
}) {
const additionalYoutubeDLArgs = [ '--skip-download', '--playlist-reverse' ]
if (options.lastVideosCount !== undefined) {
additionalYoutubeDLArgs.push('--playlist-end', options.lastVideosCount.toString())
if (options.latestVideosCount !== undefined) {
additionalYoutubeDLArgs.push('--playlist-end', options.latestVideosCount.toString())
}
return this.getInfo({
url: options.channelUrl,
Expand Down
2 changes: 2 additions & 0 deletions server/initializers/checker-before-init.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ function checkMissedConfig () {
'transcoding.resolutions.480p', 'transcoding.resolutions.720p', 'transcoding.resolutions.1080p', 'transcoding.resolutions.1440p',
'transcoding.resolutions.2160p', 'video_studio.enabled',
'import.videos.http.enabled', 'import.videos.torrent.enabled', 'import.videos.concurrency', 'import.videos.timeout',
'import.video_channel_synchronization.enabled', 'import.video_channel_synchronization.max_per_user',
'import.video_channel_synchronization.check.interval', 'import.video_channel_synchronization.check.latest_videos_count',
'auto_blacklist.videos.of_users.enabled', 'trending.videos.interval_days',
'client.videos.miniature.display_author_avatar',
'client.videos.miniature.prefer_author_display_name', 'client.menu.login.redirect_on_single_external_auth',
Expand Down
8 changes: 7 additions & 1 deletion server/initializers/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -398,7 +398,13 @@ const CONFIG = {
},
VIDEO_CHANNEL_SYNCHRONIZATION: {
get ENABLED () { return config.get<boolean>('import.video_channel_synchronization.enabled') },
get CHECK_INTERVAL () { return parseDurationToMs(config.get<string>('import.video_channel_synchronization.check_interval')) }
get MAX_PER_USER () { return config.get<number>('import.video_channel_synchronization.max_per_user') },
CHECK: {
get INTERVAL () { return parseDurationToMs(config.get<string>('import.video_channel_synchronization.check.interval')) },
get LATEST_VIDEOS_COUNT () {
return parseDurationToMs(config.get<number>('import.video_channel_synchronization.check.latest_videos_count'))
}
}
}
},
AUTO_BLACKLIST: {
Expand Down
5 changes: 1 addition & 4 deletions server/initializers/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -246,7 +246,7 @@ const SCHEDULER_INTERVALS_MS = {
REMOVE_OLD_HISTORY: 60000 * 60 * 24, // 1 day
UPDATE_INBOX_STATS: 1000 * 60, // 1 minute
REMOVE_DANGLING_RESUMABLE_UPLOADS: 60000 * 60, // 1 hour
CHANNEL_SYNC_CHECK_INTERVAL: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.CHECK_INTERVAL
CHANNEL_SYNC_CHECK_INTERVAL: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.CHECK.INTERVAL
}

// ---------------------------------------------------------------------------
Expand Down Expand Up @@ -461,8 +461,6 @@ const VIDEO_PRIVACIES: { [ id in VideoPrivacy ]: string } = {
[VideoPrivacy.INTERNAL]: 'Internal'
}

const VIDEO_CHANNEL_MAX_SYNC = 3

const VIDEO_STATES: { [ id in VideoState ]: string } = {
[VideoState.PUBLISHED]: 'Published',
[VideoState.TO_TRANSCODE]: 'To transcode',
Expand Down Expand Up @@ -993,7 +991,6 @@ export {
VIDEO_LANGUAGES,
VIDEO_PRIVACIES,
VIDEO_LICENCES,
VIDEO_CHANNEL_MAX_SYNC,
VIDEO_STATES,
WORKER_THREADS,
VIDEO_RATE_TYPES,
Expand Down
4 changes: 2 additions & 2 deletions server/lib/schedulers/video-channel-sync-latest-scheduler.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import { CONFIG } from '@server/initializers/config'
import { VideoChannelSyncModel } from '@server/models/video/video-channel-sync'
import { MChannelSyncChannel } from '@server/types/models'
import { VideoChannelSyncState } from '@shared/models'
import { SCHEDULER_INTERVALS_MS, VIDEO_CHANNEL_MAX_SYNC } from '../../initializers/constants'
import { SCHEDULER_INTERVALS_MS } from '../../initializers/constants'
import { synchronizeChannel } from '../video-import-channel'
import { AbstractScheduler } from './abstract-scheduler'

Expand Down Expand Up @@ -35,7 +35,7 @@ export class VideoChannelSyncLatestScheduler extends AbstractScheduler {
const { errors, successes, alreadyImported } = await synchronizeChannel(sync.VideoChannel, sync.externalChannelUrl, {
youtubeDL,
secondsToWait: 5,
lastVideosCount: VIDEO_CHANNEL_MAX_SYNC,
latestVideosCount: CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.CHECK.LATEST_VIDEOS_COUNT,
onlyAfter: syncCreationDate
})
if (errors > 0) {
Expand Down
7 changes: 3 additions & 4 deletions server/lib/video-import-channel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ const processOptions = {
export type SynchronizeChannelOptions = {
youtubeDL: YoutubeDLCLI
secondsToWait: number
lastVideosCount?: number
latestVideosCount?: number
onlyAfter?: Date
}

Expand All @@ -32,11 +32,11 @@ function formatDateForYoutubeDl (date: Date) {
export async function synchronizeChannel (
channel: MChannelAccountDefault,
externalChannelUrl: string,
{ youtubeDL, secondsToWait, lastVideosCount, onlyAfter }: SynchronizeChannelOptions
{ youtubeDL, secondsToWait, latestVideosCount, onlyAfter }: SynchronizeChannelOptions
): Promise<ChannelSyncInfo> {
const user = await UserModel.loadByChannelActorId(channel.actorId)
const channelInfo = await youtubeDL.getChannelInfo({
lastVideosCount,
latestVideosCount,
channelUrl: externalChannelUrl,
processOptions
})
Expand All @@ -62,7 +62,6 @@ export async function synchronizeChannel (

for (const targetUrl of targetUrls) {
try {
// TODO retry pour l'import d'une chaîne ?
if (!await VideoImportModel.urlAlreadyImported(channel.id, targetUrl)) {
const { job } = await addYoutubeDLImport({
user,
Expand Down
8 changes: 8 additions & 0 deletions server/middlewares/validators/videos/video-channel-sync.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,14 @@ export const videoChannelSyncValidator = [
return
}

const count = await VideoChannelSyncModel.countByAccount(res.locals.videoChannel.accountId)
if (count >= CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.MAX_PER_USER) {
res.fail({
message: `You cannot create more than ${CONFIG.IMPORT.VIDEO_CHANNEL_SYNCHRONIZATION.MAX_PER_USER} channel synchronizations`
})
return false
}

return next()
}
]
Expand Down
16 changes: 16 additions & 0 deletions server/models/video/video-channel-sync.ts
Original file line number Diff line number Diff line change
Expand Up @@ -107,6 +107,22 @@ export class VideoChannelSyncModel extends Model<Partial<AttributesOnly<VideoCha
]).then(([ total, data ]) => ({ total, data }))
}

static countByAccount (accountId: number) {
const query = {
include: [
{
model: VideoChannelModel.unscoped(),
required: true,
where: {
accountId
}
}
]
}

return VideoChannelSyncModel.unscoped().count(query)
}

static loadWithAccount (id: number) {
return this.unscoped().findByPk(id, {
include: [ {
Expand Down
24 changes: 8 additions & 16 deletions server/tests/api/check-params/config.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/* eslint-disable @typescript-eslint/no-unused-expressions,@typescript-eslint/require-await */

import 'mocha'
import { omit } from 'lodash'
import { chain, omit } from 'lodash'
import {
cleanupTests,
createSingleServer,
Expand Down Expand Up @@ -162,7 +162,8 @@ describe('Test config API validators', function () {
}
},
videoChannelSynchronization: {
enabled: false
enabled: false,
maxPerUser: 10
}
},
trending: {
Expand Down Expand Up @@ -348,20 +349,11 @@ describe('Test config API validators', function () {
})

it('Should fail with a disabled http upload & enabled sync', async function () {
const newUpdateParams: CustomConfig = {
...updateParams,
import: {
videos: {
...updateParams.import.videos,
http: {
enabled: false
}
},
videoChannelSynchronization: {
enabled: true
}
}
}
const newUpdateParams: CustomConfig = chain(updateParams)
.cloneDeep()
.update('import.videos.http.enabled', () => false)
.update('import.videoChannelSynchronization.enabled', () => true)
.value()
await makePutBodyRequest({
url: server.url,
path,
Expand Down
29 changes: 29 additions & 0 deletions server/tests/api/check-params/video-channel-syncs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import {
setAccessTokensToServers,
setDefaultVideoChannel
} from '@shared/server-commands'
import { chain } from 'lodash'
import 'mocha'

describe('Test video channel sync API validator', () => {
Expand All @@ -33,6 +34,21 @@ describe('Test video channel sync API validator', () => {
}
}

async function withMaxSyncsPerUser<T> (maxSync: number, callback: () => Promise<T>): Promise<void> {
const origConfig = await server.config.getCustomConfig()
const newConfig = chain(origConfig).cloneDeep().update('import.videoChannelSynchronization.maxPerUser', () => 1).value()
await server.config.updateCustomConfig({
newCustomConfig: newConfig
})
try {
await callback()
} finally {
await server.config.updateCustomConfig({
newCustomConfig: origConfig
})
}
}

before(async function () {
this.timeout(30_000)

Expand Down Expand Up @@ -159,6 +175,19 @@ describe('Test video channel sync API validator', () => {
})
rootChannelSyncId = res.id
})

it('Should fail when the user exceeds allowed number of synchronizations', async function () {
await withMaxSyncsPerUser(1, async () => {
await command.create({
token: server.accessToken,
attributes: {
...baseCorrectParams,
videoChannelId: userInfo.channelId
},
expectedStatus: HttpStatusCode.BAD_REQUEST_400
})
})
})
})

describe('When listing my video imports', function () {
Expand Down
3 changes: 2 additions & 1 deletion server/tests/api/server/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -364,7 +364,8 @@ const newCustomConfig: CustomConfig = {
}
},
videoChannelSynchronization: {
enabled: false
enabled: false,
maxPerUser: 10
}
},
trending: {
Expand Down
1 change: 1 addition & 0 deletions shared/models/server/custom-config.model.ts
Original file line number Diff line number Diff line change
Expand Up @@ -164,6 +164,7 @@ export interface CustomConfig {
}
videoChannelSynchronization: {
enabled: boolean
maxPerUser: number
}
}

Expand Down
3 changes: 2 additions & 1 deletion shared/server-commands/server/config-command.ts
Original file line number Diff line number Diff line change
Expand Up @@ -384,7 +384,8 @@ export class ConfigCommand extends AbstractCommand {
}
},
videoChannelSynchronization: {
enabled: false
enabled: false,
maxPerUser: 10
}
},
trending: {
Expand Down