Truncate correctly video descriptions

pull/429/head
Chocobozzz 2018-03-28 11:00:02 +02:00
parent 63f4b13281
commit c73e83da28
No known key found for this signature in database
GPG Key ID: 583A612D890159BE
4 changed files with 36 additions and 17 deletions

View File

@ -12,6 +12,7 @@ import { isAbsolute, join } from 'path'
import * as pem from 'pem' import * as pem from 'pem'
import * as rimraf from 'rimraf' import * as rimraf from 'rimraf'
import { URL } from 'url' import { URL } from 'url'
import { truncate } from 'lodash'
function sanitizeUrl (url: string) { function sanitizeUrl (url: string) {
const urlObject = new URL(url) const urlObject = new URL(url)
@ -78,6 +79,22 @@ function buildPath (path: string) {
return join(root(), path) return join(root(), path)
} }
// Consistent with .length, lodash truncate function is not
function peertubeTruncate (str: string, maxLength: number) {
const options = {
length: maxLength
}
const truncatedStr = truncate(str, options)
// The truncated string is okay, we can return it
if (truncatedStr.length <= maxLength) return truncatedStr
// Lodash takes into account all UTF characters, whereas String.prototype.length does not: some characters have a length of 2
// We always use the .length so we need to truncate more if needed
options.length -= truncatedStr.length - maxLength
return truncate(str, options)
}
function promisify0<A> (func: (cb: (err: any, result: A) => void) => void): () => Promise<A> { function promisify0<A> (func: (cb: (err: any, result: A) => void) => void): () => Promise<A> {
return function promisified (): Promise<A> { return function promisified (): Promise<A> {
return new Promise<A>((resolve: (arg: A) => void, reject: (err: any) => void) => { return new Promise<A>((resolve: (arg: A) => void, reject: (err: any) => void) => {
@ -145,6 +162,7 @@ export {
sanitizeUrl, sanitizeUrl,
sanitizeHost, sanitizeHost,
buildPath, buildPath,
peertubeTruncate,
promisify0, promisify0,
promisify1, promisify1,

View File

@ -1,5 +1,6 @@
import * as validator from 'validator' import * as validator from 'validator'
import { ACTIVITY_PUB } from '../../../initializers' import { ACTIVITY_PUB, CONSTRAINTS_FIELDS } from '../../../initializers'
import { peertubeTruncate } from '../../core-utils'
import { exists, isBooleanValid, isDateValid, isUUIDValid } from '../misc' import { exists, isBooleanValid, isDateValid, isUUIDValid } from '../misc'
import { import {
isVideoAbuseReasonValid, isVideoAbuseReasonValid,
@ -56,6 +57,7 @@ function isVideoTorrentObjectValid (video: any) {
isBooleanValid(video.commentsEnabled) && isBooleanValid(video.commentsEnabled) &&
isDateValid(video.published) && isDateValid(video.published) &&
isDateValid(video.updated) && isDateValid(video.updated) &&
setTruncatedContent(video) &&
(!video.content || isRemoteVideoContentValid(video.mediaType, video.content)) && (!video.content || isRemoteVideoContentValid(video.mediaType, video.content)) &&
isRemoteVideoIconValid(video.icon) && isRemoteVideoIconValid(video.icon) &&
setValidRemoteVideoUrls(video) && setValidRemoteVideoUrls(video) &&
@ -111,6 +113,14 @@ function setValidRemoteVideoUrls (video: any) {
return true return true
} }
function setTruncatedContent (video: any) {
if (video.content) {
video.content = peertubeTruncate(video.content, CONSTRAINTS_FIELDS.VIDEOS.TRUNCATED_DESCRIPTION.max)
}
return true
}
function isRemoteVideoUrlValid (url: any) { function isRemoteVideoUrlValid (url: any) {
return url.type === 'Link' && return url.type === 'Link' &&
( (

View File

@ -43,7 +43,7 @@ async function processActivities (activities: Activity[], signatureActor?: Actor
try { try {
await activityProcessor(activity, inboxActor) await activityProcessor(activity, inboxActor)
} catch (err) { } catch (err) {
logger.warn('Cannot process activity %s.', activity.type, { error: err.stack }) logger.warn('Cannot process activity %s.', activity.type, { err })
} }
} }
} }

View File

@ -1,5 +1,5 @@
import * as Bluebird from 'bluebird' import * as Bluebird from 'bluebird'
import { map, maxBy, truncate } from 'lodash' import { map, maxBy } from 'lodash'
import * as magnetUtil from 'magnet-uri' import * as magnetUtil from 'magnet-uri'
import * as parseTorrent from 'parse-torrent' import * as parseTorrent from 'parse-torrent'
import { join } from 'path' import { join } from 'path'
@ -31,7 +31,10 @@ import { VideoTorrentObject } from '../../../shared/models/activitypub/objects'
import { Video, VideoDetails } from '../../../shared/models/videos' import { Video, VideoDetails } from '../../../shared/models/videos'
import { VideoFilter } from '../../../shared/models/videos/video-query.type' import { VideoFilter } from '../../../shared/models/videos/video-query.type'
import { activityPubCollection } from '../../helpers/activitypub' import { activityPubCollection } from '../../helpers/activitypub'
import { createTorrentPromise, renamePromise, statPromise, unlinkPromise, writeFilePromise } from '../../helpers/core-utils' import {
createTorrentPromise, peertubeTruncate, renamePromise, statPromise, unlinkPromise,
writeFilePromise
} from '../../helpers/core-utils'
import { isActivityPubUrlValid } from '../../helpers/custom-validators/activitypub/misc' import { isActivityPubUrlValid } from '../../helpers/custom-validators/activitypub/misc'
import { isBooleanValid } from '../../helpers/custom-validators/misc' import { isBooleanValid } from '../../helpers/custom-validators/misc'
import { import {
@ -1191,19 +1194,7 @@ export class VideoModel extends Model<VideoModel> {
if (!this.description) return null if (!this.description) return null
const maxLength = CONSTRAINTS_FIELDS.VIDEOS.TRUNCATED_DESCRIPTION.max const maxLength = CONSTRAINTS_FIELDS.VIDEOS.TRUNCATED_DESCRIPTION.max
return peertubeTruncate(this.description, maxLength)
const options = {
length: maxLength
}
const truncatedDescription = truncate(this.description, options)
// The truncated string is okay, we can return it
if (truncatedDescription.length <= maxLength) return truncatedDescription
// Lodash takes into account all UTF characters, whereas String.prototype.length does not: some characters have a length of 2
// We always use the .length so we need to truncate more if needed
options.length -= maxLength - truncatedDescription.length
return truncate(this.description, options)
} }
optimizeOriginalVideofile = async function () { optimizeOriginalVideofile = async function () {