Skip to content

Commit

Permalink
account: add IP rate limiting for account creation
Browse files Browse the repository at this point in the history
Add a configurable cooldown in seconds for which no further accounts
can be created from the same IP.
  • Loading branch information
CounterPillow committed Aug 26, 2019
1 parent 2680934 commit 7875901
Show file tree
Hide file tree
Showing 2 changed files with 23 additions and 3 deletions.
4 changes: 4 additions & 0 deletions config.example.py
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,10 @@
# Relies on USE_RECAPTCHA. Set to 0 to disable.
ACCOUNT_RECAPTCHA_AGE = 7 * 24 * 3600 # A week

# Seconds after which an IP is allowed to register another account
# (0 disables the limitation)
PER_IP_ACCOUNT_COOLDOWN = 24 * 3600

# Backup original .torrent uploads
BACKUP_TORRENT_FOLDER = 'torrents'

Expand Down
22 changes: 19 additions & 3 deletions nyaa/views/account.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import binascii
import time
from datetime import datetime
from datetime import datetime, timedelta
from ipaddress import ip_address

import flask
Expand Down Expand Up @@ -80,17 +80,33 @@ def logout():
return response


def _check_for_multi_account(ip, cooldown):
if not cooldown:
return False
cooldown_timestamp = datetime.utcnow() - timedelta(seconds=cooldown)
q = models.User.query.filter(ip == models.User.registration_ip,
models.User.created_time > cooldown_timestamp)
return db.session.query(q.exists()).scalar()


@bp.route('/register', methods=['GET', 'POST'])
def register():
if flask.g.user:
return flask.redirect(redirect_url())

form = forms.RegisterForm(flask.request.form)
if flask.request.method == 'POST' and form.validate():
ip = ip_address(flask.request.remote_addr).packed

if _check_for_multi_account(ip, app.config.get('PER_IP_ACCOUNT_COOLDOWN', 0)):
flask.flash('You or somebody else has already registered an account from this IP '
'recently. You cannot register another one.', 'danger')
return flask.render_template('register.html', form=form)

user = models.User(username=form.username.data.strip(),
email=form.email.data.strip(), password=form.password.data)
user.registration_ip = ip_address(flask.request.remote_addr).packed
user.last_login_ip = user.registration_ip
user.registration_ip = ip
user.last_login_ip = ip
db.session.add(user)
db.session.commit()

Expand Down

0 comments on commit 7875901

Please sign in to comment.