auth.py 14 KB

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