Account Takeover [It Looked Secure at First]
In a recent pentest for a client, I was going through the password reset flow. You know:
1. Forgot password => Enter email => Receive email with link:
Click on this link to reset your password.The link expires in...
2. The reset link was something like this:/passreset/<token>
The token looks like some salted bcrypt, but it will really not matter in the end…
3. Clicking on the link takes you to the /passreset/<token>
page, where you have 2 inputs: New Password
and Confirm New Password
.
4. Filling in the 2 fields, sending and intercepting the request takes you into a whole different game:
What is going on here?
- There is an
id
parameter, which, as you may guess, holds value for theid
of the user whose password is being reset. - This
id
is 10 characters long. Looking at other user IDs, it appears the format is alpha-numeric only. Thus, this is an easy job for bruteforcing, especially in the context of lacking rate limits. - Thus, to take over any account, just input a valid
id
.
Going further, this app was far from secure because a low-privileged user could enumerate all the users with their details (an admin
feature). Thus, another way to collect IDs for account takeover.
And even more critical, a low-privileged user could elevate their privileges to admin
. I went a step further and also found that the role updating operation could be performed unauthenticated (Yes, you heard that right!):
You may wonder, how can there be apps like this working in 2024?
Well, it’s not an exception, but rather an encounter more frequent than you might think. And sadly, for this company, this was their production environment.
Their focus was to implement frontend limitations, potentially thinking that may be sufficient for keeping away privy eyes from peeking into unauthorized areas. I keep seeing this behavior frequently in pentests. Bad way to go about it.
When it comes to the password reset feature, their token thing has not been implemented all the way to the end. It has been lost somewhere in translation. The token should be time-sensitive and bound to the user. Moreover, the token must actually be used in the request responsible for the password reset, instead of the user ID.
With these (and other) issues uncovered, they’ll very likely start implementing a stronger security posture. Hopefully.