Hi @ismcagdas,
What is your product version? v12.4.0
What is your product type (Angular or MVC)? Angular
What is product framework type (.net framework or .net core)? .net 7
The Session Hijacking attack compromises the session token by stealing or predicting a valid session token to gain unauthorized access to the Web Server. Using the Cookie, attacker can gain access to the application as authorized user and can modify the data, and he can pose as legitimate users, gain information, and take actions under the assumed identity. Please see the details in the following screenshots.
When copying an admin's Abp.AuthToken cookie to a non-admin user, WMS Pro shows an error message: "Request data failed, Session.UserId is null! Probably, user is not logged in.".
Reload the page, the user is logged out. This is normal. Please investigate why sometimes the user is not logged out. Refresh the page, the non-admin user is shown as an admin.
Related:
-
My question at https://support.aspnetzero.com/QA/Questions/10654/Login-Issue-w-Antiforgery-Tokens
-
Why do we have to comment out [ValidateAntiForgeryToken] in the following places:
\webapi\src\umsplus.Web.Core\Controllers\TenantCustomizationController.cs
//[ValidateAntiForgeryToken]
public async Task<JsonResult> UploadLightLogo()
...
//[ValidateAntiForgeryToken]
public async Task<JsonResult> UploadCustomCss()
...
\webapi\src\umsplus.Web.Core\Controllers\UsersControllerBase.cs
//[ValidateAntiForgeryToken]
public async Task<JsonResult> ImportFromExcel()
...
... -
\webapi\src\umsplus.Web.Host\Startup\Startup.cs
services.AddControllersWithViews(options =>
{
options.Filters.Add(new AbpAutoValidateAntiforgeryTokenAttribute());
It should be done for dynamic web api automatically. Does this mean, with this set, we don't need to set the attribute [ValidateAntiForgeryToken] in the places?
14 Answer(s)
-
0
Hi,
You shouldn't disable
ValidateAntiForgeryToken
attribute. Did you face any issues while using it ? If not, you can enable it back.
It seems like your HTTPS certificate is not valid as far as I can see from the screnshot. I suggest using a valid certificate. This should solve the problem. -
0
I have reinstated [ValidateAntiForgeryToken] attributes. But it didn't help solve the reported session hijacking described at the top pf this post.
The example of session hijacking: I copied the AbP.AuthToken from an admin user to a non-admin user. The non-admin user now has the admin rights, which is an authentication and authorization issue.
I think the [ValidateAntiForgeryToken] attribute won't directly address the session hijacking. It is not directly applicable to preventing token impersonation. Is this an authentication issue?
Please note that this issue can also be reproduced with Zero app 12.4.2.
-
0
Hi,
Are you hosting Angular app with the Host app together under the same domain ?
-
0
Hi @ismcagdas,
Yes, we host Angular app with the Host app together under the same domain. -
0
Thanks,
In that case, instead of directly serving Angular app (and its index.html file), you can host this file via an ASP.NET Core Controller and cshtml file. You can follow an approach like this https://github.com/angular/angular-cli/issues/2960#issuecomment-263639803
In that case, Anti-Forgery will work.
I assume you already enabled HTTP-Only cookies. In that case, client side can't access auth token at all.
-
0
Hi @ismcagdas,
Yes, our app has HTTP-Only cookies enabled. The HttpOnly flag prevents client-side scripts (JavaScript) from accessing the cookie containing the token. This helps mitigate Cross-site Scripting (XSS) attacks, where an attacker injects malicious code into a web page and steals the token using JavaScript.
However, it doesn't prevent access from the developer tools provided by your browser. Anyone with access to your browser, including yourself, can view and copy the token using these tools.
Please try Zero app 12.4.2: copy the Abp.AuthToken from an admin user to a non-admin user. The non-admin user now has the admin rights.
-
0
Hi,
In that case, as I stated above (if you also want to enable Anti-Forgery validation), you need to follow;
In that case, instead of directly serving Angular app (and its index.html file), you can host this file via an ASP.NET Core Controller and cshtml file. You can follow an approach like this https://github.com/angular/angular-cli/issues/2960#issuecomment-263639803
This way, you can force anti-forgery validaiton for your Angular client.
-
0
Hi @ismcagdas,
Not sure whether forcing anti-forgery validation will fix the session hijacking issue. Put anti-forgery validation aside, have you considered the session hijacking issue in the original Zero app. Please note that I can reproduce it in ASP.NET Zero 12.4.2. -
0
Hi,
Yes, we did but it didn't come out in our tests.
Please take a look at the following artice https://www.globalsign.com/en/blog/session-hijacking-and-how-to-prevent-it.Normally tools like you are using offers a solution to their findings. Do you have such an information ? We can evaluate to implement it.
-
0
Hi @ismcagdas,
I've added SessionValidationMiddleware, seesionID claim in access and refresh tokens, and UserSessionCache with seesionID and userId. See the code below.
If an admin's token is copied to a non-admin user. jwtUserIdClaimed is the admin userId, but userId is also changed to admin userId / jwtUserIdClaimed?
Does this mean that the userId in _abpSession is always the same as jwtUserIdClaimed. Is the principal synced with the access token by ABP?
Please reproduce this session hijacking issue with ASP.NET Zero 13.1.0 code. We use ASP.NET Zero 13.1.0.
This is a critical security issue. A hacker can copy an admin user's access token through developer tool and make him gain admin privileges. Please investigate and resolve the issue.
` public class SessionValidationMiddleware
{
private readonly RequestDelegate _next;
private readonly ICacheManager _cacheManager;
private readonly IAbpSession _abpSession;public SessionValidationMiddleware(RequestDelegate next, ICacheManager cacheManager, IAbpSession abpSession) { _next = next; _cacheManager = cacheManager; _abpSession = abpSession; } public async Task Invoke(HttpContext context) { var userClaims = context.User.Claims; var sessionIdClaim = userClaims.FirstOrDefault(c => c.Type == "SessionId"); var userIdClaim = userClaims.FirstOrDefault(c => c.Type == "user_identifier"); if (sessionIdClaim != null && userIdClaim != null) { if (userIdClaim.Value != _abpSession.ToUserIdentifier().ToUserIdentifierString()) { //never reached here. They should be the same. throw new WmsproException((int)HttpStatusCode.Unauthorized, "Invalid user identifier"); } //user_identifier: UserId + "@" + TenantId var parts = userIdClaim.Value.Split('@'); if (parts.Length == 2) { var userId = parts[0]; //var tenantId = parts[1]; if (userId != _abpSession.ToUserIdentifier().UserId.ToString()) { //never reached here. They should be the same. throw new WmsproException((int)HttpStatusCode.Unauthorized, "Invalid user identifier"); } var sessionId = sessionIdClaim.Value; var token = context.Request.Headers["Authorization"].ToString().Replace("Bearer ", ""); if (!string.IsNullOrEmpty(token)) { var handler = new JwtSecurityTokenHandler(); var jwtToken = handler.ReadToken(token) as JwtSecurityToken; if (jwtToken != null) { var jwtUserIdClaim = jwtToken.Claims.FirstOrDefault(claim => claim.Type == JwtRegisteredClaimNames.Sub); var jwtSessionIdClaim = jwtToken.Claims.FirstOrDefault(claim => claim.Type == "SessionId"); if (jwtUserIdClaim != null && jwtSessionIdClaim != null) { var jwtUserIdClaimed = jwtUserIdClaim.Value; //seems _abpSession always gets the userId from jwt access token rather than from the principal, or the principal is synced with the token by ABP. //var userId = _abpSession.UserId.Value.ToString(); var jwtSessionIdClaimed = jwtSessionIdClaim.Value; // Retrieve the cached session ID using the correct method signature var cache = _cacheManager.GetCache("UserSessionCache"); var cachedSessionId = await cache.GetAsync(userId, async (key) => null); // Assuming a factory function returning null if not found // Retrieve the current session user ID (assumes session is available) //var sessionUserId = context.Session.GetString("UserId"); //If an admin's token is copied to a non-admin. jwtUserIdClaimed is the admin userId, but userId is also changed to admin userId / jwtUserIdClaimed? //This means that the userId in _abpSession is always the same as jwtUserIdClaimed. Is the principal synced with the access token by ABP? if ((cachedSessionId != null && (cachedSessionId != sessionId || cachedSessionId != jwtSessionIdClaimed)) || (jwtSessionIdClaimed != sessionId || jwtUserIdClaimed != userId)) { throw new WmsproException((int)HttpStatusCode.Unauthorized, "Invalid session"); } } else { //context.Response.StatusCode = (int)HttpStatusCode.Unauthorized; //await context.Response.WriteAsync("Token is missing required claims."); //return; } } } } } await _next(context); } }`
In Startup.cs, I have added
app.UseMiddleware<SessionValidationMiddleware>();
The changes in TokenAuthController.cs:
InternalAuthenticate():
` var sessionId = Guid.NewGuid().ToString();
await _cacheManager.GetCache("UserSessionCache").SetAsync(loginResult.User.Id.ToString(), sessionId, TimeSpan.FromMinutes(30));var refreshToken = CreateRefreshToken(await CreateJwtClaims(loginResult.Identity, loginResult.User, tokenType: TokenType.RefreshToken, sessionId: sessionId)); var accessToken = CreateAccessToken(await CreateJwtClaims(loginResult.Identity, loginResult.User, refreshTokenKey: refreshToken.key, sessionId: sessionId));`
` private async Task<IEnumerable> CreateJwtClaims(
ClaimsIdentity identity, User user,
TimeSpan? expiration = null,
TokenType tokenType = TokenType.AccessToken,
string refreshTokenKey = null, string sessionId = null)
{
var tokenValidityKey = Guid.NewGuid().ToString();
var claims = identity.Claims.ToList();
var nameIdClaim = claims.First(c => c.Type == _identityOptions.ClaimsIdentity.UserIdClaimType);if (_identityOptions.ClaimsIdentity.UserIdClaimType != JwtRegisteredClaimNames.Sub) { claims.Add(new Claim(JwtRegisteredClaimNames.Sub, nameIdClaim.Value)); } if (!string.IsNullOrEmpty(sessionId)) { claims.AddRange(new[] { new Claim("SessionId", sessionId), }); } claims.AddRange(new[] { new Claim(JwtRegisteredClaimNames.Jti, Guid.NewGuid().ToString()), new Claim(JwtRegisteredClaimNames.Iat, DateTimeOffset.Now.ToUnixTimeSeconds().ToString(), ClaimValueTypes.Integer64), new Claim(AppConsts.TokenValidityKey, tokenValidityKey), new Claim(AppConsts.UserIdentifier, user.ToUserIdentifier().ToUserIdentifierString()), new Claim(AppConsts.TokenType, tokenType.To().ToString()) }); `
-
0
Hi @hongbing.wang
Does your implementation fix the problem ?
-
0
Hi @ismcagdas,
It doesn't.
So, I want to use ASP.NET Core Session instead of AbpSession.
I aim to use
HttpContext.Session.SetString("SId", sessionId);
HttpContext.Session.SetString("UId", loginResult.User.Id.ToString());In Startup.cs, I added
app.UseSession()But it throws an exception at
app.UseAbp(options =>
{
options.UseAbpRequestLocalization = false; //used below: UseAbpRequestLocalization
});
The exception is
Component Abp.Domain.Uow.UnitOfWorkDefaultOptions could not be registered. There is already a component with that name. Did you want to modify the existing component instead? If not, make sure you specify a unique name.How do I resolve the error? Can I use ASP.NET Core Session anyway?
-
0
I've figured out the issue with using use ASP.NET Core Session. In startup.cs, I also need to add the following service registration.
services.AddSession(options => { options.IdleTimeout = TimeSpan.FromSeconds(30); options.Cookie.HttpOnly = true; options.Cookie.IsEssential = true; });
When logging in, the HttpContext.Session has the two key-value pairs. However in Invoke(HttpContext context) in the validator, context.Session lost the key-value pairs. So it is still not working.
Developer tool, Application tab > Under Storage, select Cookies and then the site (http://localhost:4200), there is not any cookie named .AspNetCore.Session or something similar.
-
0
Hi,
Replied to your email. We can continue via email adn update the ticket when the issue is resolved.