Fetch outbox to grab old activities tests

This commit is contained in:
Chocobozzz 2017-11-22 11:27:40 +01:00
parent c986175d68
commit c46edbc2f6
No known key found for this signature in database
GPG Key ID: 583A612D890159BE
11 changed files with 81 additions and 30 deletions

View File

@ -2,7 +2,6 @@ import { createReadStream } from 'fs'
import { join } from 'path' import { join } from 'path'
import { createInterface } from 'readline' import { createInterface } from 'readline'
import * as winston from 'winston' import * as winston from 'winston'
import { readFileBufferPromise } from '../server/helpers/core-utils'
import { CONFIG } from '../server/initializers/constants' import { CONFIG } from '../server/initializers/constants'
const label = CONFIG.WEBSERVER.HOSTNAME + ':' + CONFIG.WEBSERVER.PORT const label = CONFIG.WEBSERVER.HOSTNAME + ':' + CONFIG.WEBSERVER.PORT
@ -16,7 +15,8 @@ const logger = new winston.Logger({
humanReadableUnhandledException: true, humanReadableUnhandledException: true,
json: false, json: false,
colorize: true, colorize: true,
prettyPrint: true prettyPrint: true,
stderrLevels: []
}) })
], ],
exitOnError: true exitOnError: true

View File

@ -56,7 +56,7 @@ async function accountController (req: express.Request, res: express.Response, n
async function accountFollowersController (req: express.Request, res: express.Response, next: express.NextFunction) { async function accountFollowersController (req: express.Request, res: express.Response, next: express.NextFunction) {
const account: AccountInstance = res.locals.account const account: AccountInstance = res.locals.account
const page = req.params.page || 1 const page = req.query.page || 1
const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE)
const result = await db.AccountFollow.listAcceptedFollowerUrlsForApi([ account.id ], start, count) const result = await db.AccountFollow.listAcceptedFollowerUrlsForApi([ account.id ], start, count)
@ -68,7 +68,7 @@ async function accountFollowersController (req: express.Request, res: express.Re
async function accountFollowingController (req: express.Request, res: express.Response, next: express.NextFunction) { async function accountFollowingController (req: express.Request, res: express.Response, next: express.NextFunction) {
const account: AccountInstance = res.locals.account const account: AccountInstance = res.locals.account
const page = req.params.page || 1 const page = req.query.page || 1
const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE)
const result = await db.AccountFollow.listAcceptedFollowingUrlsForApi([ account.id ], start, count) const result = await db.AccountFollow.listAcceptedFollowingUrlsForApi([ account.id ], start, count)

View File

@ -28,7 +28,7 @@ export {
async function outboxController (req: express.Request, res: express.Response, next: express.NextFunction) { async function outboxController (req: express.Request, res: express.Response, next: express.NextFunction) {
const account: AccountInstance = res.locals.account const account: AccountInstance = res.locals.account
const page = req.params.page || 1 const page = req.query.page || 1
const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE)
const data = await db.Video.listAllAndSharedByAccountForOutbox(account.id, start, count) const data = await db.Video.listAllAndSharedByAccountForOutbox(account.id, start, count)

View File

@ -1,11 +1,15 @@
import * as express from 'express' import * as express from 'express'
import { UserRight } from '../../../../shared/models/users/user-right.enum' import { UserRight } from '../../../../shared/models/users/user-right.enum'
import { getFormattedObjects } from '../../../helpers' import { getFormattedObjects } from '../../../helpers'
import { retryTransactionWrapper } from '../../../helpers/database-utils'
import { logger } from '../../../helpers/logger' import { logger } from '../../../helpers/logger'
import { getServerAccount } from '../../../helpers/utils' import { getServerAccount } from '../../../helpers/utils'
import { getAccountFromWebfinger } from '../../../helpers/webfinger' import { getAccountFromWebfinger } from '../../../helpers/webfinger'
import { SERVER_ACCOUNT_NAME } from '../../../initializers/constants' import { SERVER_ACCOUNT_NAME } from '../../../initializers/constants'
import { database as db } from '../../../initializers/database' import { database as db } from '../../../initializers/database'
import { saveAccountAndServerIfNotExist } from '../../../lib/activitypub/account'
import { sendUndoFollow } from '../../../lib/activitypub/send/send-undo'
import { sendFollow } from '../../../lib/index'
import { asyncMiddleware, paginationValidator, removeFollowingValidator, setFollowersSort, setPagination } from '../../../middlewares' import { asyncMiddleware, paginationValidator, removeFollowingValidator, setFollowersSort, setPagination } from '../../../middlewares'
import { authenticate } from '../../../middlewares/oauth' import { authenticate } from '../../../middlewares/oauth'
import { setBodyHostsPort } from '../../../middlewares/servers' import { setBodyHostsPort } from '../../../middlewares/servers'
@ -13,13 +17,8 @@ import { setFollowingSort } from '../../../middlewares/sort'
import { ensureUserHasRight } from '../../../middlewares/user-right' import { ensureUserHasRight } from '../../../middlewares/user-right'
import { followValidator } from '../../../middlewares/validators/follows' import { followValidator } from '../../../middlewares/validators/follows'
import { followersSortValidator, followingSortValidator } from '../../../middlewares/validators/sort' import { followersSortValidator, followingSortValidator } from '../../../middlewares/validators/sort'
import { AccountFollowInstance } from '../../../models/index'
import { sendFollow } from '../../../lib/index'
import { sendUndoFollow } from '../../../lib/activitypub/send/send-undo'
import { AccountInstance } from '../../../models/account/account-interface' import { AccountInstance } from '../../../models/account/account-interface'
import { retryTransactionWrapper } from '../../../helpers/database-utils' import { AccountFollowInstance } from '../../../models/index'
import { saveAccountAndServerIfNotExist } from '../../../lib/activitypub/account'
import { addFetchOutboxJob } from '../../../lib/activitypub/fetch'
const serverFollowsRouter = express.Router() const serverFollowsRouter = express.Router()
@ -137,8 +136,6 @@ async function follow (fromAccount: AccountInstance, targetAccount: AccountInsta
if (accountFollow.state === 'pending') { if (accountFollow.state === 'pending') {
await sendFollow(accountFollow, t) await sendFollow(accountFollow, t)
} }
await addFetchOutboxJob(targetAccount, t)
}) })
} catch (err) { } catch (err) {
// Reset target account // Reset target account

View File

@ -24,12 +24,15 @@ function activityPubContextify <T> (data: T) {
}) })
} }
function activityPubCollectionPagination (url: string, page: number, result: ResultList<any>) { function activityPubCollectionPagination (url: string, page: any, result: ResultList<any>) {
let next: string let next: string
let prev: string let prev: string
// Assert page is a number
page = parseInt(page, 10)
// There are more results // There are more results
if (result.total > ((page + 1) * ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE)) { if (result.total > page * ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) {
next = url + '?page=' + (page + 1) next = url + '?page=' + (page + 1)
} }
@ -53,6 +56,8 @@ function activityPubCollectionPagination (url: string, page: number, result: Res
totalItems: result.total, totalItems: result.total,
first: orderedCollectionPagination first: orderedCollectionPagination
}) })
} else {
orderedCollectionPagination['totalItems'] = result.total
} }
return orderedCollectionPagination return orderedCollectionPagination

View File

@ -328,6 +328,7 @@ if (isTestInstance() === true) {
REMOTE_SCHEME.HTTP = 'http' REMOTE_SCHEME.HTTP = 'http'
REMOTE_SCHEME.WS = 'ws' REMOTE_SCHEME.WS = 'ws'
STATIC_MAX_AGE = '0' STATIC_MAX_AGE = '0'
ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE = 2
} }
// --------------------------------------------------------------------------- // ---------------------------------------------------------------------------

View File

@ -1,6 +1,7 @@
import { ActivityAccept } from '../../../../shared/models/activitypub/activity' import { ActivityAccept } from '../../../../shared/models/activitypub/activity'
import { database as db } from '../../../initializers' import { database as db } from '../../../initializers'
import { AccountInstance } from '../../../models/account/account-interface' import { AccountInstance } from '../../../models/account/account-interface'
import { addFetchOutboxJob } from '../fetch'
async function processAcceptActivity (activity: ActivityAccept, inboxAccount?: AccountInstance) { async function processAcceptActivity (activity: ActivityAccept, inboxAccount?: AccountInstance) {
if (inboxAccount === undefined) throw new Error('Need to accept on explicit inbox.') if (inboxAccount === undefined) throw new Error('Need to accept on explicit inbox.')
@ -24,4 +25,5 @@ async function processAccept (account: AccountInstance, targetAccount: AccountIn
follow.set('state', 'accepted') follow.set('state', 'accepted')
await follow.save() await follow.save()
await addFetchOutboxJob(targetAccount, undefined)
} }

View File

@ -48,7 +48,7 @@ function addRemoteVideo (account: AccountInstance,
activity: ActivityAdd, activity: ActivityAdd,
videoChannel: VideoChannelInstance, videoChannel: VideoChannelInstance,
videoToCreateData: VideoTorrentObject) { videoToCreateData: VideoTorrentObject) {
logger.debug('Adding remote video %s.', videoToCreateData.url) logger.debug('Adding remote video %s.', videoToCreateData.id)
return db.sequelize.transaction(async t => { return db.sequelize.transaction(async t => {
const sequelizeOptions = { const sequelizeOptions = {

View File

@ -25,7 +25,7 @@ async function process (payload: ActivityPubHttpPayload, jobId: number) {
if (firstBody.first && Array.isArray(firstBody.first.orderedItems)) { if (firstBody.first && Array.isArray(firstBody.first.orderedItems)) {
const activities = firstBody.first.orderedItems const activities = firstBody.first.orderedItems
logger.info('Processing %i items ActivityPub fetcher for %s.', activities.length, uri) logger.info('Processing %i items ActivityPub fetcher for %s.', activities.length, options.uri)
await processActivities(activities) await processActivities(activities)
} }
@ -37,12 +37,12 @@ async function process (payload: ActivityPubHttpPayload, jobId: number) {
options.uri = nextLink options.uri = nextLink
const { body } = await doRequest(options) const { body } = await doRequest(options)
nextLink = body.nextLink nextLink = body.next
i++ i++
if (Array.isArray(body.orderedItems)) { if (Array.isArray(body.orderedItems)) {
const activities = body.orderedItems const activities = body.orderedItems
logger.info('Processing %i items ActivityPub fetcher for %s.', activities.length, uri) logger.info('Processing %i items ActivityPub fetcher for %s.', activities.length, options.uri)
await processActivities(activities) await processActivities(activities)
} }

View File

@ -46,6 +46,7 @@ import { TagInstance } from './tag-interface'
import { VideoFileInstance, VideoFileModel } from './video-file-interface' import { VideoFileInstance, VideoFileModel } from './video-file-interface'
import { VideoAttributes, VideoInstance, VideoMethods } from './video-interface' import { VideoAttributes, VideoInstance, VideoMethods } from './video-interface'
import { sendDeleteVideo } from '../../lib/index' import { sendDeleteVideo } from '../../lib/index'
import * as Bluebird from 'bluebird'
const Buffer = safeBuffer.Buffer const Buffer = safeBuffer.Buffer
@ -786,14 +787,21 @@ list = function () {
} }
listAllAndSharedByAccountForOutbox = function (accountId: number, start: number, count: number) { listAllAndSharedByAccountForOutbox = function (accountId: number, start: number, count: number) {
const queryVideo = 'SELECT "Video"."id" FROM "Videos" AS "Video" ' + function getRawQuery (select: string) {
const queryVideo = 'SELECT ' + select + ' FROM "Videos" AS "Video" ' +
'INNER JOIN "VideoChannels" AS "VideoChannel" ON "VideoChannel"."id" = "Video"."channelId" ' + 'INNER JOIN "VideoChannels" AS "VideoChannel" ON "VideoChannel"."id" = "Video"."channelId" ' +
'WHERE "VideoChannel"."accountId" = ' + accountId 'WHERE "VideoChannel"."accountId" = ' + accountId
const queryVideoShare = 'SELECT "Video"."id" FROM "VideoShares" AS "VideoShare" ' + const queryVideoShare = 'SELECT ' + select + ' FROM "VideoShares" AS "VideoShare" ' +
'INNER JOIN "Videos" AS "Video" ON "Video"."id" = "VideoShare"."videoId" ' + 'INNER JOIN "Videos" AS "Video" ON "Video"."id" = "VideoShare"."videoId" ' +
'INNER JOIN "VideoChannels" AS "VideoChannel" ON "VideoChannel"."id" = "Video"."channelId" ' +
'WHERE "VideoShare"."accountId" = ' + accountId 'WHERE "VideoShare"."accountId" = ' + accountId
const rawQuery = `(${queryVideo}) UNION (${queryVideoShare}) LIMIT ${count} OFFSET ${start}`
let rawQuery = `(${queryVideo}) UNION (${queryVideoShare})`
return rawQuery
}
const rawQuery = getRawQuery('"Video"."id"')
const rawCountQuery = getRawQuery('COUNT("Video"."id") as "total"')
const query = { const query = {
distinct: true, distinct: true,
@ -825,10 +833,20 @@ listAllAndSharedByAccountForOutbox = function (accountId: number, start: number,
] ]
} }
return Video.findAndCountAll(query).then(({ rows, count }) => { return Bluebird.all([
Video.findAll(query),
Video['sequelize'].query(rawCountQuery, { type: Sequelize.QueryTypes.SELECT })
]).then(([ rows, totals ]) => {
// totals: totalVideos + totalVideoShares
let totalVideos = 0
let totalVideoShares = 0
if (totals[0]) totalVideos = parseInt(totals[0].total, 10)
if (totals[1]) totalVideoShares = parseInt(totals[1].total, 10)
const total = totalVideos + totalVideoShares
return { return {
data: rows, data: rows,
total: count total: total
} }
}) })
} }

View File

@ -22,7 +22,7 @@ describe('Test follows', function () {
let server3Id: number let server3Id: number
before(async function () { before(async function () {
this.timeout(120000) this.timeout(20000)
servers = await flushAndRunMultipleServers(3) servers = await flushAndRunMultipleServers(3)
@ -163,6 +163,34 @@ describe('Test follows', function () {
expect(res.body.data[0].name).to.equal('server3') expect(res.body.data[0].name).to.equal('server3')
}) })
it('Should propagate previous uploaded videos on a new following', async function () {
this.timeout(20000)
await uploadVideo(servers[2].url, servers[2].accessToken, { name: 'server3-2' })
await uploadVideo(servers[2].url, servers[2].accessToken, { name: 'server3-3' })
await uploadVideo(servers[2].url, servers[2].accessToken, { name: 'server3-4' })
await uploadVideo(servers[2].url, servers[2].accessToken, { name: 'server3-5' })
await uploadVideo(servers[2].url, servers[2].accessToken, { name: 'server3-6' })
await wait(5000)
// Server 1 follows server 3
await follow(servers[0].url, [ servers[2].url ], servers[0].accessToken)
await wait(7000)
let res = await getVideosList(servers[0].url)
expect(res.body.total).to.equal(7)
const video2 = res.body.data.find(v => v.name === 'server3-2')
const video4 = res.body.data.find(v => v.name === 'server3-4')
const video6 = res.body.data.find(v => v.name === 'server3-6')
expect(video2).to.not.be.undefined
expect(video4).to.not.be.undefined
expect(video6).to.not.be.undefined
})
after(async function () { after(async function () {
killallServers(servers) killallServers(servers)