AppRegisterController.php 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. <?php
  2. namespace App\Http\Controllers;
  3. use App\Mail\InAppRegisterEmailVerify;
  4. use App\Models\AppRegister;
  5. use App\User;
  6. use App\Util\Lexer\RestrictedNames;
  7. use Illuminate\Http\Request;
  8. use Illuminate\Support\Facades\DB;
  9. use Illuminate\Support\Facades\Hash;
  10. use Illuminate\Support\Facades\Mail;
  11. use Purify;
  12. class AppRegisterController extends Controller
  13. {
  14. public function index(Request $request)
  15. {
  16. abort_unless(config('auth.iar') == true, 404);
  17. $open = (bool) config_cache('pixelfed.open_registration');
  18. if (! $open || $request->user()) {
  19. return redirect('/');
  20. }
  21. return view('auth.iar');
  22. }
  23. public function store(Request $request)
  24. {
  25. abort_unless(config('auth.iar') == true, 404);
  26. $open = (bool) config_cache('pixelfed.open_registration');
  27. if (! $open || $request->user()) {
  28. return redirect('/');
  29. }
  30. $rules = [
  31. 'email' => 'required|email:rfc,dns,spoof,strict|unique:users,email|unique:app_registers,email',
  32. ];
  33. if ((bool) config_cache('captcha.enabled') && (bool) config_cache('captcha.active.register')) {
  34. $rules['h-captcha-response'] = 'required|captcha';
  35. }
  36. $this->validate($request, $rules);
  37. $email = $request->input('email');
  38. $code = str_pad(random_int(0, 999999), 6, '0', STR_PAD_LEFT);
  39. $exists = AppRegister::whereEmail($email)->where('created_at', '>', now()->subHours(24))->count();
  40. if ($exists && $exists > 3) {
  41. $errorParams = http_build_query([
  42. 'status' => 'error',
  43. 'message' => 'Too many attempts, please try again later.',
  44. ]);
  45. return redirect()->away("pixelfed://verifyEmail?{$errorParams}");
  46. }
  47. DB::beginTransaction();
  48. $registration = AppRegister::create([
  49. 'email' => $email,
  50. 'verify_code' => $code,
  51. 'email_delivered_at' => now(),
  52. ]);
  53. try {
  54. Mail::to($email)->send(new InAppRegisterEmailVerify($code));
  55. } catch (\Exception $e) {
  56. DB::rollBack();
  57. $errorParams = http_build_query([
  58. 'status' => 'error',
  59. 'message' => 'Failed to send verification code',
  60. ]);
  61. return redirect()->away("pixelfed://verifyEmail?{$errorParams}");
  62. }
  63. DB::commit();
  64. $queryParams = http_build_query([
  65. 'email' => $request->email,
  66. 'expires_in' => 3600,
  67. 'status' => 'success',
  68. ]);
  69. return redirect()->away("pixelfed://verifyEmail?{$queryParams}");
  70. }
  71. public function verifyCode(Request $request)
  72. {
  73. abort_unless(config('auth.iar') == true, 404);
  74. $open = (bool) config_cache('pixelfed.open_registration');
  75. if (! $open || $request->user()) {
  76. return redirect('/');
  77. }
  78. $this->validate($request, [
  79. 'email' => 'required|email:rfc,dns,spoof,strict|unique:users,email',
  80. 'verify_code' => ['required', 'digits:6', 'numeric'],
  81. ]);
  82. $email = $request->input('email');
  83. $code = $request->input('verify_code');
  84. $exists = AppRegister::whereEmail($email)
  85. ->whereVerifyCode($code)
  86. ->where('created_at', '>', now()->subMinutes(60))
  87. ->exists();
  88. return response()->json([
  89. 'status' => $exists ? 'success' : 'error',
  90. ]);
  91. }
  92. public function onboarding(Request $request)
  93. {
  94. abort_unless(config('auth.iar') == true, 404);
  95. $open = (bool) config_cache('pixelfed.open_registration');
  96. if (! $open || $request->user()) {
  97. return redirect('/');
  98. }
  99. $this->validate($request, [
  100. 'email' => 'required|email:rfc,dns,spoof,strict|unique:users,email',
  101. 'verify_code' => ['required', 'digits:6', 'numeric'],
  102. 'username' => $this->validateUsernameRule(),
  103. 'name' => 'nullable|string|max:'.config('pixelfed.max_name_length'),
  104. 'password' => 'required|string|min:'.config('pixelfed.min_password_length'),
  105. ]);
  106. $email = $request->input('email');
  107. $code = $request->input('verify_code');
  108. $username = $request->input('username');
  109. $name = $request->input('name');
  110. $password = $request->input('password');
  111. $exists = AppRegister::whereEmail($email)
  112. ->whereVerifyCode($code)
  113. ->where('created_at', '>', now()->subMinutes(60))
  114. ->exists();
  115. if (! $exists) {
  116. return response()->json([
  117. 'status' => 'error',
  118. 'message' => 'Invalid verification code, please try again later.',
  119. ]);
  120. }
  121. $user = User::create([
  122. 'name' => Purify::clean($name),
  123. 'username' => $username,
  124. 'email' => $email,
  125. 'password' => Hash::make($password),
  126. 'app_register_ip' => request()->ip(),
  127. 'register_source' => 'app',
  128. ]);
  129. sleep(10);
  130. return response()->json([
  131. 'status' => 'success',
  132. 'auth_token' => $user->createToken('Pixelfed App')->accessToken,
  133. ]);
  134. }
  135. protected function validateUsernameRule()
  136. {
  137. return [
  138. 'required',
  139. 'min:2',
  140. 'max:30',
  141. 'unique:users',
  142. function ($attribute, $value, $fail) {
  143. $dash = substr_count($value, '-');
  144. $underscore = substr_count($value, '_');
  145. $period = substr_count($value, '.');
  146. if (ends_with($value, ['.php', '.js', '.css'])) {
  147. return $fail('Username is invalid.');
  148. }
  149. if (($dash + $underscore + $period) > 1) {
  150. return $fail('Username is invalid. Can only contain one dash (-), period (.) or underscore (_).');
  151. }
  152. if (! ctype_alnum($value[0])) {
  153. return $fail('Username is invalid. Must start with a letter or number.');
  154. }
  155. if (! ctype_alnum($value[strlen($value) - 1])) {
  156. return $fail('Username is invalid. Must end with a letter or number.');
  157. }
  158. $val = str_replace(['_', '.', '-'], '', $value);
  159. if (! ctype_alnum($val)) {
  160. return $fail('Username is invalid. Username must be alpha-numeric and may contain dashes (-), periods (.) and underscores (_).');
  161. }
  162. if (! preg_match('/[a-zA-Z]/', $value)) {
  163. return $fail('Username is invalid. Must contain at least one alphabetical character.');
  164. }
  165. $restricted = RestrictedNames::get();
  166. if (in_array(strtolower($value), array_map('strtolower', $restricted))) {
  167. return $fail('Username cannot be used.');
  168. }
  169. },
  170. ];
  171. }
  172. }