furizon_webint/api.py

217 lines
7.7 KiB
Python
Raw Normal View History

2023-05-11 22:18:49 +00:00
from sanic import response
from sanic import Blueprint, exceptions
from ext import *
from config import *
import sqlite3
import smtplib
from email.mime.text import MIMEText
import random
import string
bp = Blueprint("api", url_prefix="/manage/api")
@bp.route("/members.json")
async def api_members(request):
ret = []
for o in sorted(request.app.ctx.om.cache.values(), key=lambda x: len(x.room_members), reverse=True):
if o.status in ['c', 'e']: continue
ret.append({
'code': o.code,
'sponsorship': o.sponsorship,
'is_fursuiter': o.is_fursuiter,
'name': o.name,
'has_early': o.has_early,
'has_late': o.has_late,
'propic': o.ans('propic'),
'propic_fursuiter': o.ans('propic_fursuiter'),
'staff_role': o.ans('staff_role'),
'country': o.country,
2023-05-23 19:02:02 +00:00
'is_checked_in': False,
'points': random.randint(0,50) if random.random() > 0.3 else 0
2023-05-11 22:18:49 +00:00
})
return response.json(ret)
2023-05-23 19:02:02 +00:00
@bp.route("/leaderboard.json")
async def api_leaderboard(request):
ret = []
for o in sorted(request.app.ctx.om.cache.values(), key=lambda x: len(x.room_members), reverse=True):
if o.status in ['c', 'e']: continue
ret.append({
'code': o.code,
'propic': o.ans('propic'),
'points': random.randint(0,50) if random.random() > 0.3 else '???'
})
return response.json(sorted(ret, key=lambda x: x['points'], reverse=True))
2023-05-11 22:18:49 +00:00
@bp.route("/events.json")
async def show_events(request):
with sqlite3.connect('data/event.db') as db:
db.row_factory = sqlite3.Row
events = db.execute('SELECT * FROM event ORDER BY start ASC')
2023-05-23 19:02:02 +00:00
r = response.json([dict(x) for x in events])
r.headers["Access-Control-Allow-Origin"] = "*"
2023-05-11 22:18:49 +00:00
2023-05-23 19:02:02 +00:00
return r
2023-05-11 22:18:49 +00:00
@bp.route("/achievements.json")
async def show_events(request):
2023-05-23 19:02:02 +00:00
if request.token:
user = await request.app.ctx.om.get_order(code=request.token[:5])
if not user or user.app_token != request.token[5:]:
return response.json({'ok': False, 'error': 'The token you have provided is not valid.'}, status=401)
2023-05-11 22:18:49 +00:00
with sqlite3.connect('data/achievement.db') as db:
db.row_factory = sqlite3.Row
2023-05-23 19:02:02 +00:00
events = db.execute('SELECT * FROM achievement ORDER BY points DESC')
return response.json([{'won_at': '2023-05-05T21:00Z' if request.token and random.random() < 0.2 else None, **dict(x), 'about': 'This is instructions on how to win the field.'} for x in events])
2023-05-11 22:18:49 +00:00
@bp.get("/logout")
async def logout(request):
if not request.token:
return response.json({'ok': False, 'error': 'You need to provide a token.'}, status=401)
user = await request.app.ctx.om.get_order(code=request.token[:5])
2023-05-23 19:02:02 +00:00
if not user or user.app_token != request.token[5:]:
2023-05-11 22:18:49 +00:00
return response.json({'ok': False, 'error': 'The token you have provided is not valid.'}, status=401)
2023-05-23 19:02:02 +00:00
await user.edit_answer('app_token', None)
2023-05-11 22:18:49 +00:00
await user.send_answers()
return response.json({'ok': True, 'message': 'You have been logged out and this token has been destroyed.'})
2023-05-23 19:02:02 +00:00
@bp.get("/test")
async def token_test(request):
if not request.token:
return response.json({'ok': False, 'error': 'You need to provide a token.'}, status=401)
user = await request.app.ctx.om.get_order(code=request.token[:5])
if not user or user.app_token != request.token[5:]:
return response.json({'ok': False, 'error': 'The token you have provided is not correct.'}, status=401)
return response.json({'ok': True, 'message': 'This token is valid :)'})
@bp.get("/welcome")
async def welcome_app(request):
if not request.token:
return response.json({'ok': False, 'error': 'You need to provide a token.'}, status=401)
o = await request.app.ctx.om.get_order(code=request.token[:5])
if not o or o.app_token != request.token[5:]:
return response.json({'ok': False, 'error': 'The token you have provided is not correct.'}, status=401)
return response.json({
'code': o.code,
'sponsorship': o.sponsorship,
'is_fursuiter': o.is_fursuiter,
'name': o.name,
'has_early': o.has_early,
'has_late': o.has_late,
'propic': o.ans('propic'),
'propic_fursuiter': o.ans('propic_fursuiter'),
'staff_role': o.ans('staff_role'),
'country': o.country,
'is_checked_in': False,
'points': random.randint(0,50) if random.random() > 0.3 else 0,
'can_scan_nfc': o.can_scan_nfc,
'actual_room_id': o.actual_room_id,
'phone': '+3901234567890'
})
@bp.get("/scan/<nfc_id>")
async def nfc_scan(request, nfc_id):
if not request.token:
return response.json({'ok': False, 'error': 'You need to provide a token.'}, status=401)
user = await request.app.ctx.om.get_order(code=request.token[:5])
if not user or user.app_token != request.token[5:]:
return response.json({'ok': False, 'error': 'The token you have provided is not correct.'}, status=401)
if not user.can_scan_nfc:
return response.json({'ok': False, 'error': 'You cannot scan NFC at this time.'}, status=401)
for o in request.app.ctx.om.cache:
if o.nfc_id == nfc_id:
return response.json({
'code': o.code,
'sponsorship': o.sponsorship,
'is_fursuiter': o.is_fursuiter,
'name': o.name,
'has_early': o.has_early,
'has_late': o.has_late,
'propic': o.ans('propic'),
'propic_fursuiter': o.ans('propic_fursuiter'),
'staff_role': o.ans('staff_role'),
'country': o.country,
'is_checked_in': False,
'points': random.randint(0,50) if random.random() > 0.3 else 0,
'comment': o.comment,
'actual_room_id': o.actual_room_id
})
return response.json({'ok': True, 'message': 'This NFC tag is not valid.'})
2023-05-11 22:18:49 +00:00
@bp.get("/get_token/<code>/<login_code>")
async def get_token_from_code(request, code, login_code):
if not code in request.app.ctx.login_codes:
2023-05-23 19:02:02 +00:00
print(request.app.ctx.login_codes)
2023-05-11 22:18:49 +00:00
return response.json({'ok': False, 'error': 'You need to reauthenticate. The code has expired.'}, status=401)
if request.app.ctx.login_codes[code][1] == 0:
del request.app.ctx.login_codes[code]
return response.json({'ok': False, 'error': 'Too many tries. Please reauthenticate again.'}, status=401)
if request.app.ctx.login_codes[code][0] != login_code:
request.app.ctx.login_codes[code][1] -= 1
return response.json({'ok': False, 'error': 'The login code is incorrect. Try again.'}, status=401)
user = await request.app.ctx.om.get_order(code=code)
token = ''.join(random.choice(string.ascii_letters) for _ in range(48))
2023-05-23 19:02:02 +00:00
await user.edit_answer('app_token', token)
2023-05-11 22:18:49 +00:00
await user.send_answers()
return response.json({'ok': True, 'token': code+token})
@bp.route("/get_token/<code>")
async def get_token(request, code):
user = await request.app.ctx.om.get_order(code=code)
if not user:
return response.json({'ok': False, 'error': 'The user you have requested does not exist.'}, status=404)
2023-05-23 19:02:02 +00:00
if user.status in ['expired', 'canceled']:
return response.json({'ok': False, 'error': 'This user is not allowed to login because the order has been canceled.'}, status=401)
2023-05-11 22:18:49 +00:00
if not user.email:
return response.json({'ok': False, 'error': 'This user has not provided their email.'}, status=401)
request.app.ctx.login_codes[code] = [''.join(random.choice(string.digits) for _ in range(6)), 3]
try:
msg = MIMEText(f"Hello {user.name}!\n\nWe have received a request to login in the app. If you didn't do this, please ignore this email. Somebody is probably playing with you.\n\nYour login code is: {request.app.ctx.login_codes[code][0]}\n\nPlease do not tell this to anybody!")
msg['Subject'] = '[Furizon] Your login code'
msg['From'] = 'Furizon <no-reply@furizon.net>'
msg['To'] = f"{user.name} <{user.email}>"
s = smtplib.SMTP_SSL(SMTP_HOST)
s.login(SMTP_USER, SMTP_PASSWORD)
s.sendmail(msg['From'], msg['to'], msg.as_string())
s.quit()
except:
return response.json({'ok': False, 'error': 'There has been an issue sending your e-mail. Please try again later or report to an admin.'}, status=500)
return response.json({'ok': True, 'message': 'A login code has been sent to your email.'})