I’ve been searching the internet and have yet to find a solution to the following problem…
We currently have a website developed using Laravel which the user table is a remote Microsoft SQL database. The driver in config/auth.php has been set to “database”. All is working fine except for the password reset functionality, which we get the following error:
UnexpectedValueException in PasswordBroker.php line 238: User must implement CanResetPassword interface.
From my limited understanding of Laravel (this is my first experiance with Laravel), the Eloquent driver has support for the CanResetPassword functionality, however, this has not been implemented in the Database User Provider by Laravel, hence the error.
So my question is thus, has anyone had a configuration where they have the driver to “Database” and implemented a reset password functionality? All the examples I have seen to date relate to using the Eloquent model, which from my understanding of Laravel is not an option since during the initial development we had to change the driver from Eloquent to database to get the remote Microsoft SQL server working in the first place. Moving the Microsoft SQL database to a local database is not an option I’m afraid.
Alternatively, if anyone has implemented another method of a user resetting their password using an email address I would be open to suggestions.
Advertisement
Answer
To write your own password reset logic, you can still use the default migration that comes out of the box or simply create yours. The most important part is the token. Because you are making your own password reset, you have a couple of decisions to make:
- Will the token expire?
- Can a user use the same token multiple times?
You will need 2 pages, 4 different routes and 4 different functions in the same controller. The ‘I forgot my password’ page and the ‘Reset password’ page. In the first page, display a form where you take the user email. And post to the following controller.
//to be added on top as use statements 
use DB;
use Auth;
use Hash;
use Carbon;
use AppUser;
public function sendPasswordResetToken(Request $request)
{
    $user = User::where ('email', $request->email)-first();
    if ( !$user ) return redirect()->back()->withErrors(['error' => '404']);
    //create a new token to be sent to the user. 
    DB::table('password_resets')->insert([
        'email' => $request->email,
        'token' => str_random(60), //change 60 to any length you want
        'created_at' => Carbon::now()
    ]);
    $tokenData = DB::table('password_resets')
    ->where('email', $request->email)->first();
   $token = $tokenData->token;
   $email = $request->email; // or $email = $tokenData->email;
   /**
    * Send email to the email above with a link to your password reset
    * something like url('password-reset/' . $token)
    * Sending email varies according to your Laravel version. Very easy to implement
    */
}
Second part, when the user clicks on the link
/**
 * Assuming the URL looks like this 
 * http://localhost/password-reset/random-string-here
 * You check if the user and the token exist and display a page
 */
 public function showPasswordResetForm($token)
 {
     $tokenData = DB::table('password_resets')
     ->where('token', $token)->first();
     if ( !$tokenData ) return redirect()->to('home'); //redirect them anywhere you want if the token does not exist.
     return view('passwords.show');
 }
Display a page with a form containing 2 inputs
– New password password or whateveer you want 
– New password confirmation password_confirm or whatever you want
The form should post to the same URL mapped to the following controller. Why? because we still need to use the token to find the actual user. 
 public function resetPassword(Request $request, $token)
 {
     //some validation
     ...
     $password = $request->password;
     $tokenData = DB::table('password_resets')
     ->where('token', $token)->first();
     $user = User::where('email', $tokenData->email)->first();
     if ( !$user ) return redirect()->to('home'); //or wherever you want
     $user->password = Hash::make($password);
     $user->update(); //or $user->save();
     //do we log the user directly or let them login and try their password for the first time ? if yes 
     Auth::login($user);
    // If the user shouldn't reuse the token later, delete the token 
    DB::table('password_resets')->where('email', $user->email')->delete();
    //redirect where we want according to whether they are logged in or not.
 }
Don’t forget to add routes
Route::get('password-reset', 'PasswordController@showForm'); //I did not create this controller. it simply displays a view with a form to take the email
Route::post('password-reset', 'PasswordController@sendPasswordResetToken');
Route::get('reset-password/{token}', 'PasswordController@showPasswordResetForm');
Route::post('reset-password/{token}', 'PasswordController@resetPassword');
Note: There might be typos or syntax errors because I did not test this and wrote it here directly from the top of my head. If you see an error/exception, don’t panick, read the error and search google.