port: 9000
rates_limit:
+ api:
+ # 50 attempts in 10 seconds
+ window: 10 seconds
+ max: 50
login:
# 15 attempts in 5 min
window: 5 minutes
max: 15
+ signup:
+ # 2 attempts in 5 min (only succeeded attempts are taken into account)
+ window: 5 minutes
+ max: 2
ask_send_email:
# 3 attempts in 5 min
window: 5 minutes
port: 443
rates_limit:
+ api:
+ # 50 attempts in 10 seconds
+ window: 10 seconds
+ max: 50
login:
# 15 attempts in 5 min
window: 5 minutes
max: 15
+ signup:
+ # 2 attempts in 5 min (only succeeded attempts are taken into account)
+ window: 5 minutes
+ max: 2
ask_send_email:
# 3 attempts in 5 min
window: 5 minutes
webserver:
https: false
+rates_limit:
+ signup:
+ window: 10 minutes
+ max: 50
+ login:
+ window: 5 minutes
+ max: 20
+
database:
hostname: 'localhost'
port: 5432
import { checkMissedConfig, checkFFmpeg } from './server/initializers/checker-before-init'
// Do not use barrels because we don't want to load all modules here (we need to initialize database first)
-import { logger } from './server/helpers/logger'
-import { API_VERSION, FILES_CACHE, WEBSERVER, loadLanguages } from './server/initializers/constants'
import { CONFIG } from './server/initializers/config'
+import { API_VERSION, FILES_CACHE, WEBSERVER, loadLanguages } from './server/initializers/constants'
+import { logger } from './server/helpers/logger'
const missed = checkMissedConfig()
if (missed.length !== 0) {
import * as express from 'express'
+import * as RateLimit from 'express-rate-limit'
import { configRouter } from './config'
import { jobsRouter } from './jobs'
import { oauthClientsRouter } from './oauth-clients'
import { searchRouter } from './search'
import { overviewsRouter } from './overviews'
import { videoPlaylistRouter } from './video-playlist'
+import { CONFIG } from '../../initializers/config'
const apiRouter = express.Router()
credentials: true
}))
+// FIXME: https://github.com/nfriedly/express-rate-limit/issues/138
+// @ts-ignore
+const apiRateLimiter = RateLimit({
+ windowMs: CONFIG.RATES_LIMIT.API.WINDOW_MS,
+ max: CONFIG.RATES_LIMIT.API.MAX
+})
+apiRouter.use(apiRateLimiter)
+
apiRouter.use('/server', serverRouter)
apiRouter.use('/oauth-clients', oauthClientsRouter)
apiRouter.use('/config', configRouter)
import { UserCreate, UserRight, UserRole, UserUpdate } from '../../../../shared'
import { logger } from '../../../helpers/logger'
import { getFormattedObjects } from '../../../helpers/utils'
-import { RATES_LIMIT, WEBSERVER } from '../../../initializers/constants'
+import { WEBSERVER } from '../../../initializers/constants'
import { Emailer } from '../../../lib/emailer'
import { Redis } from '../../../lib/redis'
import { createUserAccountAndChannelAndPlaylist, sendVerifyUserEmail } from '../../../lib/user'
// FIXME: https://github.com/nfriedly/express-rate-limit/issues/138
// @ts-ignore
const loginRateLimiter = RateLimit({
- windowMs: RATES_LIMIT.LOGIN.WINDOW_MS,
- max: RATES_LIMIT.LOGIN.MAX
+ windowMs: CONFIG.RATES_LIMIT.LOGIN.WINDOW_MS,
+ max: CONFIG.RATES_LIMIT.LOGIN.MAX
+})
+
+// @ts-ignore
+const signupRateLimiter = RateLimit({
+ windowMs: CONFIG.RATES_LIMIT.SIGNUP.WINDOW_MS,
+ max: CONFIG.RATES_LIMIT.SIGNUP.MAX,
+ skipFailedRequests: true
})
// @ts-ignore
const askSendEmailLimiter = new RateLimit({
- windowMs: RATES_LIMIT.ASK_SEND_EMAIL.WINDOW_MS,
- max: RATES_LIMIT.ASK_SEND_EMAIL.MAX
+ windowMs: CONFIG.RATES_LIMIT.ASK_SEND_EMAIL.WINDOW_MS,
+ max: CONFIG.RATES_LIMIT.ASK_SEND_EMAIL.MAX
})
const usersRouter = express.Router()
)
usersRouter.post('/register',
+ signupRateLimiter,
asyncMiddleware(ensureUserRegistrationAllowed),
ensureUserRegistrationAllowedForIP,
asyncMiddleware(usersRegisterValidator),
PORT: config.get<number>('webserver.port')
},
RATES_LIMIT: {
+ API: {
+ WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.api.window')),
+ MAX: config.get<number>('rates_limit.api.max')
+ },
+ SIGNUP: {
+ WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.signup.window')),
+ MAX: config.get<number>('rates_limit.signup.max')
+ },
LOGIN: {
WINDOW_MS: parseDurationToMs(config.get<string>('rates_limit.login.window')),
MAX: config.get<number>('rates_limit.login.max')
}
}
-const RATES_LIMIT = {
- LOGIN: {
- WINDOW_MS: CONFIG.RATES_LIMIT.LOGIN.WINDOW_MS,
- MAX: CONFIG.RATES_LIMIT.LOGIN.MAX
- },
- ASK_SEND_EMAIL: {
- WINDOW_MS: CONFIG.RATES_LIMIT.ASK_SEND_EMAIL.WINDOW_MS,
- MAX: CONFIG.RATES_LIMIT.ASK_SEND_EMAIL.MAX
- }
-}
-
let VIDEO_VIEW_LIFETIME = 60000 * 60 // 1 hour
let CONTACT_FORM_LIFETIME = 60000 * 60 // 1 hour
FILES_CACHE.VIDEO_CAPTIONS.MAX_AGE = 3000
MEMOIZE_TTL.OVERVIEWS_SAMPLE = 1
ROUTE_CACHE_LIFETIME.OVERVIEWS.VIDEOS = '0ms'
-
- RATES_LIMIT.LOGIN.MAX = 20
}
updateWebserverUrls()
SCHEDULER_INTERVALS_MS,
REPEAT_JOBS,
STATIC_DOWNLOAD_PATHS,
- RATES_LIMIT,
MIMETYPES,
CRAWL_REQUEST_CONCURRENCY,
DEFAULT_AUDIO_RESOLUTION,
import 'mocha'
import * as chai from 'chai'
-import { cleanupTests, getVideo, uploadVideo, userLogin, viewVideo, wait } from '../../../../shared/extra-utils'
+import { cleanupTests, getVideo, registerUser, uploadVideo, userLogin, viewVideo, wait } from '../../../../shared/extra-utils'
import { flushAndRunServer, setAccessTokensToServers } from '../../../../shared/extra-utils/index'
const expect = chai.expect
before(async function () {
this.timeout(30000)
- server = await flushAndRunServer(1)
+
+ const config = {
+ rates_limit: {
+ api: {
+ max: 50,
+ window: 5000
+ },
+ signup: {
+ max: 3,
+ window: 5000
+ },
+ login: {
+ max: 20
+ }
+ },
+ signup: {
+ limit: 20
+ }
+ }
+
+ server = await flushAndRunServer(1, config)
await setAccessTokensToServers([ server ])
const { body } = await uploadVideo(server.url, server.accessToken, {})
await userLogin(server, user, 429)
})
+ it('Should rate limit signup', async function () {
+ for (let i = 0; i < 3; i++) {
+ await registerUser(server.url, 'test' + i, 'password')
+ }
+
+ await registerUser(server.url, 'test42', 'password', 429)
+ })
+
+ it('Should not rate limit failed signup', async function () {
+ this.timeout(30000)
+
+ await wait(7000)
+
+ for (let i = 0; i < 3; i++) {
+ await registerUser(server.url, 'test' + i, 'password', 409)
+ }
+
+ await registerUser(server.url, 'test43', 'password', 204)
+
+ })
+
+ it('Should rate limit API calls', async function () {
+ this.timeout(30000)
+
+ await wait(7000)
+
+ for (let i = 0; i < 50; i++) {
+ await getVideo(server.url, videoId)
+ }
+
+ await getVideo(server.url, videoId, 429)
+ })
+
after(async function () {
await cleanupTests([ server ])
})