mirror of https://github.com/Chocobozzz/PeerTube
Search channels against handles and not names
parent
fbd67e7f38
commit
b033851fb5
|
@ -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(
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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 })
|
||||||
|
|
|
@ -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 }) => {
|
||||||
|
|
|
@ -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 })
|
||||||
})
|
})
|
||||||
|
|
|
@ -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 () {
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
export * from './object'
|
|
@ -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
|
||||||
|
}
|
|
@ -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',
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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'
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -8,5 +8,5 @@ export interface VideoChannelsSearchQuery extends SearchTargetQuery {
|
||||||
sort?: string
|
sort?: string
|
||||||
|
|
||||||
host?: string
|
host?: string
|
||||||
names?: string[]
|
handles?: string[]
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue