Search channels against handles and not names

pull/4300/head
Chocobozzz 2021-07-29 10:27:24 +02:00
parent fbd67e7f38
commit b033851fb5
No known key found for this signature in database
GPG Key ID: 583A612D890159BE
17 changed files with 87 additions and 34 deletions

View File

@ -100,7 +100,7 @@ async function searchVideoChannelsDB (query: VideoChannelsSearchQuery, res: expr
count: query.count, count: query.count,
sort: query.sort, sort: query.sort,
host: query.host, host: query.host,
names: query.names handles: query.handles
}, 'filter:api.search.video-channels.local.list.params') }, 'filter:api.search.video-channels.local.list.params')
const resultList = await Hooks.wrapPromiseFun( const resultList = await Hooks.wrapPromiseFun(

View File

@ -23,6 +23,10 @@ function isNotEmptyIntArray (value: any) {
return Array.isArray(value) && value.every(v => validator.isInt('' + v)) && value.length !== 0 return Array.isArray(value) && value.every(v => validator.isInt('' + v)) && value.length !== 0
} }
function isNotEmptyStringArray (value: any) {
return Array.isArray(value) && value.every(v => typeof v === 'string' && v.length !== 0) && value.length !== 0
}
function isArrayOf (value: any, validator: (value: any) => boolean) { function isArrayOf (value: any, validator: (value: any) => boolean) {
return isArray(value) && value.every(v => validator(v)) return isArray(value) && value.every(v => validator(v))
} }
@ -187,6 +191,7 @@ export {
isIntOrNull, isIntOrNull,
isIdValid, isIdValid,
isSafePath, isSafePath,
isNotEmptyStringArray,
isUUIDValid, isUUIDValid,
toCompleteUUIDs, toCompleteUUIDs,
toCompleteUUID, toCompleteUUID,

View File

@ -2,7 +2,7 @@ import * as express from 'express'
import { query } from 'express-validator' import { query } from 'express-validator'
import { isSearchTargetValid } from '@server/helpers/custom-validators/search' import { isSearchTargetValid } from '@server/helpers/custom-validators/search'
import { isHostValid } from '@server/helpers/custom-validators/servers' import { isHostValid } from '@server/helpers/custom-validators/servers'
import { areUUIDsValid, isDateValid, toCompleteUUIDs } from '../../helpers/custom-validators/misc' import { areUUIDsValid, isDateValid, isNotEmptyStringArray, toCompleteUUIDs } from '../../helpers/custom-validators/misc'
import { logger } from '../../helpers/logger' import { logger } from '../../helpers/logger'
import { areValidationErrors } from './shared' import { areValidationErrors } from './shared'
@ -64,9 +64,10 @@ const videoChannelsListSearchValidator = [
.optional() .optional()
.custom(isSearchTargetValid).withMessage('Should have a valid search target'), .custom(isSearchTargetValid).withMessage('Should have a valid search target'),
query('names') query('handles')
.optional() .optional()
.toArray(), .toArray()
.custom(isNotEmptyStringArray).withMessage('Should have valid handles'),
(req: express.Request, res: express.Response, next: express.NextFunction) => { (req: express.Request, res: express.Response, next: express.NextFunction) => {
logger.debug('Checking video channels search query', { parameters: req.query }) logger.debug('Checking video channels search query', { parameters: req.query })

View File

@ -18,7 +18,7 @@ import {
UpdatedAt UpdatedAt
} from 'sequelize-typescript' } from 'sequelize-typescript'
import { MAccountActor } from '@server/types/models' import { MAccountActor } from '@server/types/models'
import { AttributesOnly } from '@shared/core-utils' import { AttributesOnly, pick } from '@shared/core-utils'
import { ActivityPubActor } from '../../../shared/models/activitypub' import { ActivityPubActor } from '../../../shared/models/activitypub'
import { VideoChannel, VideoChannelSummary } from '../../../shared/models/videos' import { VideoChannel, VideoChannelSummary } from '../../../shared/models/videos'
import { import {
@ -59,7 +59,7 @@ type AvailableForListOptions = {
actorId: number actorId: number
search?: string search?: string
host?: string host?: string
names?: string[] handles?: string[]
} }
type AvailableWithStatsOptions = { type AvailableWithStatsOptions = {
@ -114,15 +114,33 @@ export type SummaryOptions = {
}) })
} }
if (options.names) { let rootWhere: WhereOptions
whereActorAnd.push({ if (options.handles) {
preferredUsername: { const or: WhereOptions[] = []
[Op.in]: options.names
for (const handle of options.handles || []) {
const [ preferredUsername, host ] = handle.split('@')
if (!host) {
or.push({
'$Actor.preferredUsername$': preferredUsername,
'$Actor.serverId$': null
})
} else {
or.push({
'$Actor.preferredUsername$': preferredUsername,
'$Actor.Server.host$': host
})
} }
}) }
rootWhere = {
[Op.or]: or
}
} }
return { return {
where: rootWhere,
include: [ include: [
{ {
attributes: { attributes: {
@ -473,7 +491,7 @@ ON "Account->Actor"."serverId" = "Account->Actor->Server"."id"`
sort: string sort: string
host?: string host?: string
names?: string[] handles?: string[]
}) { }) {
let attributesInclude: any[] = [ literal('0 as similarity') ] let attributesInclude: any[] = [ literal('0 as similarity') ]
let where: WhereOptions let where: WhereOptions
@ -507,7 +525,7 @@ ON "Account->Actor"."serverId" = "Account->Actor->Server"."id"`
return VideoChannelModel return VideoChannelModel
.scope({ .scope({
method: [ ScopeNames.FOR_API, { actorId: options.actorId, host: options.host, names: options.names } as AvailableForListOptions ] method: [ ScopeNames.FOR_API, pick(options, [ 'actorId', 'host', 'handles' ]) as AvailableForListOptions ]
}) })
.findAndCountAll(query) .findAndCountAll(query)
.then(({ rows, count }) => { .then(({ rows, count }) => {

View File

@ -216,6 +216,10 @@ describe('Test videos API validator', function () {
await makeGetRequest({ url: server.url, path, query: { ...query, host: '6565' }, expectedStatus: HttpStatusCode.BAD_REQUEST_400 }) await makeGetRequest({ url: server.url, path, query: { ...query, host: '6565' }, expectedStatus: HttpStatusCode.BAD_REQUEST_400 })
}) })
it('Should fail with invalid handles', async function () {
await makeGetRequest({ url: server.url, path, query: { ...query, handles: [ '' ] }, expectedStatus: HttpStatusCode.BAD_REQUEST_400 })
})
it('Should succeed with the correct parameters', async function () { it('Should succeed with the correct parameters', async function () {
await makeGetRequest({ url: server.url, path, query, expectedStatus: HttpStatusCode.OK_200 }) await makeGetRequest({ url: server.url, path, query, expectedStatus: HttpStatusCode.OK_200 })
}) })

View File

@ -122,18 +122,25 @@ describe('Test channels search', function () {
it('Should filter by names', async function () { it('Should filter by names', async function () {
{ {
const body = await command.advancedChannelSearch({ search: { names: [ 'squall_channel', 'zell_channel' ] } }) const body = await command.advancedChannelSearch({ search: { handles: [ 'squall_channel', 'zell_channel' ] } })
expect(body.total).to.equal(1)
expect(body.data).to.have.lengthOf(1)
expect(body.data[0].displayName).to.equal('Squall channel')
}
{
const body = await command.advancedChannelSearch({ search: { handles: [ 'chocobozzz_channel' ] } })
expect(body.total).to.equal(0)
expect(body.data).to.have.lengthOf(0)
}
{
const body = await command.advancedChannelSearch({ search: { handles: [ 'squall_channel', 'zell_channel@' + remoteServer.host ] } })
expect(body.total).to.equal(2) expect(body.total).to.equal(2)
expect(body.data).to.have.lengthOf(2) expect(body.data).to.have.lengthOf(2)
expect(body.data[0].displayName).to.equal('Squall channel') expect(body.data[0].displayName).to.equal('Squall channel')
expect(body.data[1].displayName).to.equal('Zell channel') expect(body.data[1].displayName).to.equal('Zell channel')
} }
{
const body = await command.advancedChannelSearch({ search: { names: [ 'chocobozzz_channel' ] } })
expect(body.total).to.equal(0)
expect(body.data).to.have.lengthOf(0)
}
}) })
after(async function () { after(async function () {

View File

@ -4,3 +4,4 @@ export * from './i18n'
export * from './plugins' export * from './plugins'
export * from './renderer' export * from './renderer'
export * from './users' export * from './users'
export * from './utils'

View File

@ -0,0 +1 @@
export * from './object'

View File

@ -0,0 +1,15 @@
function pick <T extends object> (object: T, keys: (keyof T)[]) {
const result: Partial<T> = {}
for (const key of keys) {
if (Object.prototype.hasOwnProperty.call(object, key)) {
result[key] = object[key]
}
}
return result
}
export {
pick
}

View File

@ -1,4 +1,4 @@
import { pick } from 'lodash' import { pick } from '@shared/core-utils'
import { import {
AbuseFilter, AbuseFilter,
AbuseMessage, AbuseMessage,
@ -81,7 +81,7 @@ export class AbusesCommand extends AbstractCommand {
searchVideo?: string searchVideo?: string
searchVideoChannel?: string searchVideoChannel?: string
} = {}) { } = {}) {
const toPick = [ const toPick: (keyof typeof options)[] = [
'count', 'count',
'filter', 'filter',
'id', 'id',
@ -121,7 +121,7 @@ export class AbusesCommand extends AbstractCommand {
search?: string search?: string
state?: AbuseState state?: AbuseState
}) { }) {
const toPick = [ const toPick: (keyof typeof options)[] = [
'id', 'id',
'search', 'search',
'state', 'state',

View File

@ -1,4 +1,4 @@
import { pick } from 'lodash' import { pick } from '@shared/core-utils'
import { ActivityPubActorType, ActorFollow, FollowState, HttpStatusCode, ResultList, ServerFollowCreate } from '@shared/models' import { ActivityPubActorType, ActorFollow, FollowState, HttpStatusCode, ResultList, ServerFollowCreate } from '@shared/models'
import { AbstractCommand, OverrideCommandOptions } from '../shared' import { AbstractCommand, OverrideCommandOptions } from '../shared'
import { PeerTubeServer } from './server' import { PeerTubeServer } from './server'
@ -15,8 +15,7 @@ export class FollowsCommand extends AbstractCommand {
}) { }) {
const path = '/api/v1/server/followers' const path = '/api/v1/server/followers'
const toPick = [ 'start', 'count', 'sort', 'search', 'state', 'actorType' ] const query = pick(options, [ 'start', 'count', 'sort', 'search', 'state', 'actorType' ])
const query = pick(options, toPick)
return this.getRequestBody<ResultList<ActorFollow>>({ return this.getRequestBody<ResultList<ActorFollow>>({
...options, ...options,
@ -38,8 +37,7 @@ export class FollowsCommand extends AbstractCommand {
} = {}) { } = {}) {
const path = '/api/v1/server/following' const path = '/api/v1/server/following'
const toPick = [ 'start', 'count', 'sort', 'search', 'state', 'actorType' ] const query = pick(options, [ 'start', 'count', 'sort', 'search', 'state', 'actorType' ])
const query = pick(options, toPick)
return this.getRequestBody<ResultList<ActorFollow>>({ return this.getRequestBody<ResultList<ActorFollow>>({
...options, ...options,

View File

@ -1,4 +1,4 @@
import { pick } from 'lodash' import { pick } from '@shared/core-utils'
import { HttpStatusCode } from '@shared/models' import { HttpStatusCode } from '@shared/models'
import { Job, JobState, JobType, ResultList } from '../../models' import { Job, JobState, JobType, ResultList } from '../../models'
import { AbstractCommand, OverrideCommandOptions } from '../shared' import { AbstractCommand, OverrideCommandOptions } from '../shared'

View File

@ -1,4 +1,5 @@
import { omit, pick } from 'lodash' import { omit } from 'lodash'
import { pick } from '@shared/core-utils'
import { import {
HttpStatusCode, HttpStatusCode,
MyUser, MyUser,

View File

@ -1,4 +1,4 @@
import { pick } from 'lodash' import { pick } from '@shared/core-utils'
import { HttpStatusCode, ResultList, VideoChannel, VideoChannelCreateResult } from '@shared/models' import { HttpStatusCode, ResultList, VideoChannel, VideoChannelCreateResult } from '@shared/models'
import { VideoChannelCreate } from '../../models/videos/channel/video-channel-create.model' import { VideoChannelCreate } from '../../models/videos/channel/video-channel-create.model'
import { VideoChannelUpdate } from '../../models/videos/channel/video-channel-update.model' import { VideoChannelUpdate } from '../../models/videos/channel/video-channel-update.model'

View File

@ -1,4 +1,5 @@
import { omit, pick } from 'lodash' import { omit } from 'lodash'
import { pick } from '@shared/core-utils'
import { import {
BooleanBothQuery, BooleanBothQuery,
HttpStatusCode, HttpStatusCode,

View File

@ -3,10 +3,11 @@
import { expect } from 'chai' import { expect } from 'chai'
import { createReadStream, stat } from 'fs-extra' import { createReadStream, stat } from 'fs-extra'
import got, { Response as GotResponse } from 'got' import got, { Response as GotResponse } from 'got'
import { omit, pick } from 'lodash' import { omit } from 'lodash'
import validator from 'validator' import validator from 'validator'
import { buildUUID } from '@server/helpers/uuid' import { buildUUID } from '@server/helpers/uuid'
import { loadLanguages } from '@server/initializers/constants' import { loadLanguages } from '@server/initializers/constants'
import { pick } from '@shared/core-utils'
import { import {
HttpStatusCode, HttpStatusCode,
ResultList, ResultList,

View File

@ -8,5 +8,5 @@ export interface VideoChannelsSearchQuery extends SearchTargetQuery {
sort?: string sort?: string
host?: string host?: string
names?: string[] handles?: string[]
} }