2017-06-10 15:15:25 -05:00
|
|
|
import * as Sequelize from 'sequelize'
|
|
|
|
|
2017-05-22 13:58:25 -05:00
|
|
|
import { database as db } from '../../initializers/database'
|
2017-07-10 12:43:21 -05:00
|
|
|
import { AbstractRequestScheduler, RequestsObjects } from './abstract-request-scheduler'
|
2017-05-15 15:22:03 -05:00
|
|
|
import { logger } from '../../helpers'
|
|
|
|
import {
|
|
|
|
REQUESTS_VIDEO_QADU_LIMIT_PODS,
|
|
|
|
REQUESTS_VIDEO_QADU_LIMIT_PER_POD,
|
|
|
|
REQUEST_VIDEO_QADU_ENDPOINT,
|
|
|
|
REQUEST_VIDEO_QADU_TYPES
|
|
|
|
} from '../../initializers'
|
2017-07-10 12:43:21 -05:00
|
|
|
import { RequestsVideoQaduGrouped, PodInstance } from '../../models'
|
|
|
|
import { RemoteQaduVideoRequest, RequestVideoQaduType } from '../../../shared'
|
|
|
|
|
|
|
|
// We create a custom interface because we need "videos" attribute for our computations
|
|
|
|
interface RequestsObjectsCustom<U> extends RequestsObjects<U> {
|
|
|
|
[ id: string ]: {
|
|
|
|
toPod: PodInstance
|
|
|
|
endpoint: string
|
|
|
|
ids: number[] // ids
|
|
|
|
datas: U[]
|
|
|
|
|
|
|
|
videos: {
|
|
|
|
[ id: string ]: {
|
|
|
|
remoteId: string
|
|
|
|
likes?: number
|
|
|
|
dislikes?: number
|
|
|
|
views?: number
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-15 15:22:03 -05:00
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
export type RequestVideoQaduSchedulerOptions = {
|
2017-06-16 03:36:18 -05:00
|
|
|
type: RequestVideoQaduType
|
2017-06-10 15:15:25 -05:00
|
|
|
videoId: string
|
|
|
|
transaction?: Sequelize.Transaction
|
|
|
|
}
|
|
|
|
|
2017-07-05 06:26:25 -05:00
|
|
|
class RequestVideoQaduScheduler extends AbstractRequestScheduler<RequestsVideoQaduGrouped> {
|
2017-02-21 14:35:59 -06:00
|
|
|
constructor () {
|
|
|
|
super()
|
|
|
|
|
|
|
|
// We limit the size of the requests
|
2017-05-15 15:22:03 -05:00
|
|
|
this.limitPods = REQUESTS_VIDEO_QADU_LIMIT_PODS
|
|
|
|
this.limitPerPod = REQUESTS_VIDEO_QADU_LIMIT_PER_POD
|
2017-02-21 14:35:59 -06:00
|
|
|
|
|
|
|
this.description = 'video QADU requests'
|
|
|
|
}
|
|
|
|
|
|
|
|
getRequestModel () {
|
|
|
|
return db.RequestVideoQadu
|
|
|
|
}
|
|
|
|
|
|
|
|
getRequestToPodModel () {
|
|
|
|
return db.RequestVideoQadu
|
|
|
|
}
|
|
|
|
|
2017-07-10 12:43:21 -05:00
|
|
|
buildRequestsObjects (requests: RequestsVideoQaduGrouped) {
|
|
|
|
const requestsToMakeGrouped: RequestsObjectsCustom<RemoteQaduVideoRequest> = {}
|
2017-02-21 14:35:59 -06:00
|
|
|
|
|
|
|
Object.keys(requests).forEach(toPodId => {
|
|
|
|
requests[toPodId].forEach(data => {
|
|
|
|
const request = data.request
|
|
|
|
const video = data.video
|
|
|
|
const pod = data.pod
|
|
|
|
const hashKey = toPodId
|
|
|
|
|
|
|
|
if (!requestsToMakeGrouped[hashKey]) {
|
|
|
|
requestsToMakeGrouped[hashKey] = {
|
|
|
|
toPod: pod,
|
2017-05-15 15:22:03 -05:00
|
|
|
endpoint: REQUEST_VIDEO_QADU_ENDPOINT,
|
2017-02-21 14:35:59 -06:00
|
|
|
ids: [], // request ids, to delete them from the DB in the future
|
|
|
|
datas: [], // requests data
|
|
|
|
videos: {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-08 14:35:43 -06:00
|
|
|
// Maybe another attribute was filled for this video
|
|
|
|
let videoData = requestsToMakeGrouped[hashKey].videos[video.id]
|
2017-07-10 12:43:21 -05:00
|
|
|
if (!videoData) videoData = { remoteId: null }
|
2017-03-08 14:35:43 -06:00
|
|
|
|
2017-02-21 14:35:59 -06:00
|
|
|
switch (request.type) {
|
2017-05-15 15:22:03 -05:00
|
|
|
case REQUEST_VIDEO_QADU_TYPES.LIKES:
|
2017-02-21 14:35:59 -06:00
|
|
|
videoData.likes = video.likes
|
|
|
|
break
|
|
|
|
|
2017-05-15 15:22:03 -05:00
|
|
|
case REQUEST_VIDEO_QADU_TYPES.DISLIKES:
|
2017-03-08 14:35:43 -06:00
|
|
|
videoData.dislikes = video.dislikes
|
2017-02-21 14:35:59 -06:00
|
|
|
break
|
|
|
|
|
2017-05-15 15:22:03 -05:00
|
|
|
case REQUEST_VIDEO_QADU_TYPES.VIEWS:
|
2017-02-21 14:35:59 -06:00
|
|
|
videoData.views = video.views
|
|
|
|
break
|
|
|
|
|
|
|
|
default:
|
|
|
|
logger.error('Unknown request video QADU type %s.', request.type)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do not forget the remoteId so the remote pod can identify the video
|
|
|
|
videoData.remoteId = video.id
|
|
|
|
requestsToMakeGrouped[hashKey].ids.push(request.id)
|
2017-02-26 12:27:08 -06:00
|
|
|
|
|
|
|
// Maybe there are multiple quick and dirty update for the same video
|
|
|
|
// We use this hashmap to dedupe them
|
2017-02-21 14:35:59 -06:00
|
|
|
requestsToMakeGrouped[hashKey].videos[video.id] = videoData
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2017-02-26 12:27:08 -06:00
|
|
|
// Now we deduped similar quick and dirty updates, we can build our requests datas
|
2017-02-21 14:35:59 -06:00
|
|
|
Object.keys(requestsToMakeGrouped).forEach(hashKey => {
|
|
|
|
Object.keys(requestsToMakeGrouped[hashKey].videos).forEach(videoId => {
|
|
|
|
const videoData = requestsToMakeGrouped[hashKey].videos[videoId]
|
|
|
|
|
|
|
|
requestsToMakeGrouped[hashKey].datas.push({
|
|
|
|
data: videoData
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
// We don't need it anymore, it was just to build our datas array
|
|
|
|
delete requestsToMakeGrouped[hashKey].videos
|
|
|
|
})
|
|
|
|
|
|
|
|
return requestsToMakeGrouped
|
|
|
|
}
|
|
|
|
|
2017-07-05 06:26:25 -05:00
|
|
|
createRequest ({ type, videoId, transaction }: RequestVideoQaduSchedulerOptions) {
|
2017-06-10 15:15:25 -05:00
|
|
|
const dbRequestOptions: Sequelize.BulkCreateOptions = {}
|
2017-02-21 14:35:59 -06:00
|
|
|
if (transaction) dbRequestOptions.transaction = transaction
|
|
|
|
|
|
|
|
// Send the update to all our friends
|
2017-07-05 06:26:25 -05:00
|
|
|
return db.Pod.listAllIds(transaction).then(podIds => {
|
2017-02-21 14:35:59 -06:00
|
|
|
const queries = []
|
|
|
|
podIds.forEach(podId => {
|
|
|
|
queries.push({ type, videoId, podId })
|
|
|
|
})
|
|
|
|
|
2017-07-05 06:26:25 -05:00
|
|
|
return db.RequestVideoQadu.bulkCreate(queries, dbRequestOptions)
|
2017-02-21 14:35:59 -06:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2017-05-15 15:22:03 -05:00
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
RequestVideoQaduScheduler
|
|
|
|
}
|