auth.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. from flask import Blueprint, render_template, redirect, url_for, request, current_app, flash, abort
  2. from flask_wtf import FlaskForm
  3. from wtforms import (EmailField,
  4. PasswordField,
  5. BooleanField,
  6. SubmitField,
  7. ValidationError,
  8. StringField)
  9. from wtforms.validators import DataRequired, Length, Regexp, EqualTo
  10. from flask_login import current_user, login_user, logout_user, login_required
  11. from urllib.parse import urljoin
  12. from sqlalchemy.exc import IntegrityError
  13. from .db import db, User, Role, Follow
  14. from .logger import Logger
  15. from .mail import send_msg
  16. from .login import role_required
  17. auth = Blueprint("auth", __name__)
  18. class AuthField(FlaskForm):
  19. @staticmethod
  20. def email_field(name: str, description: str):
  21. """ 提前定义 email 字段的生成函数,供下文调用 """
  22. return EmailField(name, description=description,
  23. validators=[
  24. DataRequired(f"必须填写{name}"),
  25. Length(1, 32, message=f"{name}长度1-32个字符"),
  26. Regexp(r"^[a-zA-Z0-9_\.\-]+@[a-zA-Z0-9_\-]+(\.[a-zA-Z0-9_\.]+)+$",
  27. message=f"{name}不满足正则表达式")])
  28. @staticmethod
  29. def passwd_field(name: str, description: str):
  30. """ 提前定义 passwd 字段的生成函数,供下文调用 """
  31. return PasswordField(name, description=description,
  32. validators=[
  33. DataRequired(f"必须填写{name}"),
  34. Length(8, 32, message=f"{name}长度为8-32位")])
  35. @staticmethod
  36. def passwd_again_field(name: str, description: str, passwd: str = "passwd"):
  37. """ 提前定义 passwd again 字段的生成函数,供下文调用 """
  38. return PasswordField(f"重复{name}", description=description,
  39. validators=[
  40. DataRequired(message=f"必须再次填写{name}"),
  41. EqualTo(passwd, message=f"两次输入的{name}不相同")])
  42. class EmailPasswd(AuthField):
  43. email = AuthField.email_field("邮箱", "用户邮箱")
  44. passwd = AuthField.passwd_field("密码", "用户密码")
  45. class PasswdLoginForm(EmailPasswd):
  46. remember = BooleanField("记住我")
  47. submit = SubmitField("登录")
  48. class EmailLoginForm(AuthField):
  49. email = AuthField.email_field("邮箱", "用户邮箱")
  50. remember = BooleanField("记住我")
  51. submit = SubmitField("登录")
  52. class RegisterForm(EmailPasswd):
  53. passwd_again = AuthField.passwd_again_field("密码", "用户密码")
  54. submit = SubmitField("注册")
  55. def validate_email(self, field):
  56. """ 检验email是否合法 """
  57. if User.query.filter_by(email=field.data).first():
  58. raise ValidationError("邮箱已被注册")
  59. class ChangePasswdForm(AuthField):
  60. old_passwd = AuthField.passwd_field("旧密码", "用户原密码")
  61. passwd = AuthField.passwd_field("新密码", "用户新密码")
  62. passwd_again = AuthField.passwd_again_field("新密码", "用户新密码")
  63. submit = SubmitField("修改密码")
  64. def validate_passwd(self, field):
  65. """ 检验新旧密码是否相同 """
  66. if field.data == self.old_passwd.data:
  67. raise ValidationError("新旧密码不能相同")
  68. class ChangeRoleForm(AuthField):
  69. email = AuthField.email_field("邮箱", "用户邮箱")
  70. role = StringField("角色", description="用户角色", validators=[DataRequired(message="必须指定用户角色")])
  71. submit = SubmitField("修改")
  72. def validate_role(self, field):
  73. if not Role.query.filter_by(name=field.data).first():
  74. raise ValidationError("角色不存在")
  75. def validate_email(self, field):
  76. if not User.query.filter_by(email=field.data).first():
  77. raise ValidationError("用户不存在")
  78. def __load_login_page(passwd_login_form=None, email_login_form=None, register_form=None,
  79. on_passwd_login=True, on_email_login=False, on_register=False):
  80. if not passwd_login_form:
  81. passwd_login_form = PasswdLoginForm()
  82. if not email_login_form:
  83. email_login_form = EmailLoginForm()
  84. if not register_form:
  85. register_form = RegisterForm()
  86. return render_template("auth/login.html",
  87. passwd_login_form=passwd_login_form,
  88. email_login_form=email_login_form,
  89. register_form=register_form,
  90. on_passwd_login=on_passwd_login,
  91. on_email_login=on_email_login,
  92. on_register=on_register)
  93. @auth.route("/")
  94. def auth_page():
  95. if current_user.is_authenticated: # 用户已经成功登陆
  96. return render_template("auth/yours.html")
  97. return __load_login_page()
  98. @auth.route('/login/passwd', methods=["GET", "POST"])
  99. def passwd_login_page():
  100. if current_user.is_authenticated: # 用户已经成功登陆
  101. Logger.print_user_not_allow_opt_log("passwd-login.txt")
  102. return redirect(url_for("auth.auth_page"))
  103. form = PasswdLoginForm()
  104. if form.validate_on_submit():
  105. user = User.query.filter_by(email=form.email.data).first()
  106. if user and user.check_passwd(form.passwd.data) and user.role.has_permission(Role.USABLE):
  107. login_user(user, form.remember.data)
  108. next_page = request.args.get("next")
  109. if next_page is None or not next_page.startswith('/'):
  110. next_page = url_for('base.index_page')
  111. flash("登陆成功")
  112. Logger.print_user_opt_success_log(f"passwd login.txt {form.email.data}")
  113. return redirect(next_page)
  114. flash("账号或密码错误")
  115. Logger.print_user_opt_fail_log(f"passwd login.txt {form.email.data}")
  116. return redirect(url_for("auth.passwd_login_page"))
  117. return __load_login_page(passwd_login_form=form, on_passwd_login=True)
  118. @auth.route('/login/email', methods=["GET", "POST"])
  119. def email_login_page():
  120. if current_user.is_authenticated: # 用户已经成功登陆
  121. Logger.print_user_not_allow_opt_log("email-login.txt")
  122. return redirect(url_for("auth.auth_page"))
  123. form = EmailLoginForm()
  124. if form.validate_on_submit():
  125. user = User.query.filter_by(email=form.email.data).first()
  126. if user and user.role.has_permission(Role.USABLE):
  127. token = user.login_creat_token(form.remember.data)
  128. login_url = urljoin(request.host_url, url_for("auth.email_login_confirm_page", token=token))
  129. send_msg("登录确认", user.email, "login", login_url=login_url)
  130. flash("登录确认邮件已发送至邮箱")
  131. Logger.print_user_opt_success_log(f"email login.txt {form.email.data}")
  132. return redirect(url_for("base.index_page"))
  133. flash("账号不存在")
  134. Logger.print_user_opt_fail_log(f"email login.txt {form.email.data}")
  135. return redirect(url_for("auth.passwd_login_page"))
  136. return __load_login_page(passwd_login_form=form, on_passwd_login=True)
  137. @auth.route('/register', methods=["GET", "POST"])
  138. def register_page():
  139. if current_user.is_authenticated:
  140. Logger.print_user_not_allow_opt_log("register.txt")
  141. return redirect(url_for("auth.auth_page"))
  142. form = RegisterForm()
  143. if form.validate_on_submit():
  144. token = User.register_creat_token(form.email.data, form.passwd.data)
  145. register_url = urljoin(request.host_url, url_for("auth.register_confirm_page", token=token))
  146. send_msg("注册确认", form.email.data, "register", register_url=register_url)
  147. flash("注册提交成功, 请进入邮箱点击确认注册链接")
  148. Logger.print_import_user_opt_success_log(f"register.txt {form.email.data}")
  149. return redirect(url_for("base.index_page"))
  150. return __load_login_page(register_form=form, on_register=True, on_passwd_login=False)
  151. @auth.route('/confirm/register')
  152. def register_confirm_page():
  153. token = request.args.get("token", None)
  154. if token is None:
  155. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  156. return abort(404)
  157. token = User.register_load_token(token)
  158. if token is None:
  159. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  160. return abort(404)
  161. if User.query.filter_by(email=token[0]).first():
  162. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  163. return abort(404)
  164. if User.query.limit(1).first(): # 不是第一个用户
  165. new_user = User(email=token[0], passwd_hash=User.get_passwd_hash(token[1]))
  166. else:
  167. admin = Role.query.filter_by(name="admin").first()
  168. if admin is None:
  169. Logger.print_user_opt_fail_log(f"Role admin not found")
  170. return abort(500)
  171. new_user = User(email=token[0], passwd_hash=User.get_passwd_hash(token[1]), role=admin)
  172. db.session.add(new_user)
  173. db.session.commit()
  174. Logger.print_import_user_opt_success_log(f"confirm {token[0]} success")
  175. flash(f"用户{token[0]}认证完成")
  176. return redirect(url_for("base.index_page"))
  177. @auth.route('/confirm/login')
  178. def email_login_confirm_page():
  179. token = request.args.get("token", None)
  180. if token is None:
  181. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  182. return abort(404)
  183. token = User.login_load_token(token)
  184. if token is None:
  185. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  186. return abort(404)
  187. user = User.query.filter_by(email=token[0]).first()
  188. if not user:
  189. Logger.print_user_opt_fail_log(f"Confirm (bad token)")
  190. return abort(404)
  191. login_user(user, token[1])
  192. flash("登陆成功")
  193. Logger.print_user_opt_success_log(f"passwd login.txt {user.email}")
  194. return redirect(url_for("base.index_page"))
  195. @auth.route('/set/passwd', methods=['GET', 'POST'])
  196. @login_required
  197. def change_passwd_page():
  198. form = ChangePasswdForm()
  199. if form.validate_on_submit():
  200. if not current_user.check_passwd(form.old_passwd.data):
  201. Logger.print_user_opt_error_log(f"change passwd")
  202. flash("旧密码错误")
  203. else:
  204. current_user.passwd = form.passwd.data
  205. db.session.commit()
  206. Logger.print_user_opt_success_log(f"change passwd")
  207. flash("密码修改成功")
  208. logout_user()
  209. return redirect(url_for("auth.passwd_login_page"))
  210. return redirect(url_for("auth.change_passwd_page"))
  211. Logger.print_load_page_log("user change passwd")
  212. return render_template("auth/change_passwd.html", form=form)
  213. @auth.route('/logout')
  214. @login_required
  215. def logout_page():
  216. Logger.print_import_user_opt_success_log(f"logout")
  217. logout_user()
  218. flash("退出登录成功")
  219. return redirect(url_for("base.index_page"))
  220. @auth.route("/user")
  221. def user_page():
  222. user_id = request.args.get("user", None, type=int)
  223. if not user_id:
  224. return abort(404)
  225. elif current_user.is_authenticated and current_user.id == user_id:
  226. return redirect(url_for("auth.auth_page"))
  227. user = User.query.filter_by(id=user_id).first()
  228. if not user:
  229. return abort(404)
  230. return render_template("auth/user.html", user=user)
  231. @auth.route("/follower/list")
  232. @login_required
  233. @role_required(Role.CHECK_FOLLOW)
  234. def follower_page():
  235. if current_user.follower_count == 0:
  236. return render_template("auth/no_follow.html", title="粉丝", msg="你暂时一个粉丝都没有哦。")
  237. page = request.args.get("page", 1, type=int)
  238. pagination = current_user.follower.paginate(page=page, per_page=8, error_out=False)
  239. return render_template("auth/follow.html",
  240. items=[i.follower for i in pagination.items],
  241. pagination=pagination,
  242. title="粉丝")
  243. @auth.route("/followed/list")
  244. @login_required
  245. @role_required(Role.CHECK_FOLLOW)
  246. def followed_page():
  247. if current_user.followed_count == 0:
  248. return render_template("auth/no_follow.html", title="关注", msg="你暂时未关注任何人。")
  249. page = request.args.get("page", 1, type=int)
  250. pagination = current_user.followed.paginate(page=page, per_page=8, error_out=False)
  251. return render_template("auth/follow.html",
  252. items=[i.followed for i in pagination.items],
  253. pagination=pagination,
  254. title="关注")
  255. @auth.route("/followed/follow")
  256. @login_required
  257. @role_required(Role.FOLLOW)
  258. def set_follow_page():
  259. user_id = request.args.get("user", None, type=int)
  260. if not user_id or user_id == current_user.id:
  261. return abort(404)
  262. user = User.query.filter_by(id=user_id).first()
  263. if not user:
  264. return abort(404)
  265. try:
  266. db.session.add(Follow(follower=current_user, followed=user))
  267. db.session.commit()
  268. except IntegrityError:
  269. flash("不能重复关注用户")
  270. else:
  271. flash("关注用户成功")
  272. return redirect(url_for("auth.user_page", user=user_id))
  273. @auth.route("/followed/unfollow")
  274. @login_required
  275. @role_required(Role.FOLLOW)
  276. def set_unfollow_page():
  277. user_id = request.args.get("user", None, type=int)
  278. if not user_id or user_id == current_user.id:
  279. return abort(404)
  280. user = User.query.filter_by(id=user_id).first()
  281. if not user:
  282. return abort(404)
  283. Follow.query.filter_by(follower=current_user, followed=user).delete()
  284. flash("取消关注用户成功")
  285. return redirect(url_for("auth.user_page", user=user_id))
  286. @auth.route("/block")
  287. @login_required
  288. @role_required(Role.BLOCK_USER)
  289. def set_block_page():
  290. user_id = request.args.get("user", None, type=int)
  291. if not user_id or user_id == current_user.id:
  292. return abort(404)
  293. user = User.query.filter_by(id=user_id).first()
  294. if not user:
  295. return abort(404)
  296. block = Role.query.filter_by(name="block").first()
  297. if not block:
  298. return abort(500)
  299. user.role = block
  300. db.session.commit()
  301. return redirect(url_for("auth.user_page", user=user_id))
  302. @auth.route('/role/user', methods=['GET', 'POST'])
  303. @login_required
  304. @role_required(Role.SYSTEM)
  305. def change_role_page():
  306. form = ChangeRoleForm()
  307. if form.validate_on_submit():
  308. user = User.query.filter_by(email=form.email.data).first()
  309. if not user:
  310. flash("用户不存在")
  311. return redirect(url_for("auth.change_role_page"))
  312. role = Role.query.filter_by(name=form.role.data).first()
  313. if not role:
  314. flash("角色不存在")
  315. return redirect(url_for("auth.change_role_page"))
  316. user.role = role
  317. db.session.commit()
  318. flash("用户分组修改成功")
  319. return redirect(url_for("auth.change_role_page"))
  320. return render_template("auth/change_role.html", form=form)