Activities of "woodyarray"

In Blazor Server, I added the following in blazor-global-styles.css and both the logo and the logo icon were replaced when logged in.

:root {
    --lpx-logo: url('/images/logo/logo.svg') !important;
    --lpx-logo-icon: url('/images/logo/logo-icon.svg') !important;
}

I also added it to global-styles.css in order for it to affect the login screen as well.

Answer

You can refer to the ANZ documentation which is largely applicable for ABP

https://aspnetboilerplate.com/Pages/Documents/Zero/Organization-Units#common-use-cases

https://aspnetboilerplate.com/Pages/Documents/Articles%5CHow-To%5Cadd-custom-data-filter-ef-core

The second link covers the case when a User can be assigned to only one Org Unit. If your use case covers M:M assignments it gets a bit more complicate. Here is sample code (I'm a noob so FWIW):

Create a Customer UserClaimsPrincipalFactory and in it put

 public async override Task<ClaimsPrincipal> CreateAsync(IdentityUser user) 
        { 
            var principal = await base.CreateAsync(user); 
            var organizationUnits = await _identityUserManager.GetOrganizationUnitsAsync(user); 
            foreach (OrganizationUnit org in organizationUnits) 
            { 
                principal.Identities.First().AddClaim(new Claim("OU", org.Code)); 
            } 
            return principal; 
        } 

In your DBContext

   protected override Expression<Func<TEntity, bool>> CreateFilterExpression<TEntity>() 
    { 
        var expression = base.CreateFilterExpression<TEntity>(); 
 
        if (typeof(IMayHaveOrganizationUnit).IsAssignableFrom(typeof(TEntity))) 
        { 
            var predicate = PredicateBuilder.New<OrganizationUnit>(); 
            if (OrgUnitCodes == null || !OrgUnitCodes.Any()) 
            { 
 
                // If No organizations assigned to User, return no entities 
                predicate = predicate.And(p => false); 
            } 
            else 
            { 
                foreach (var code in OrgUnitCodes) 
                { 
                    predicate = predicate.Or(ou => ou.Code.StartsWith(code)); 
                } 
            } 
            Expression<Func<TEntity, bool>> mayHaveOuFilter = e => !IsOrganizationUnitFilterEnabled 
             || OrganizationUnits 
             .Where(predicate) 
             .Select(s => s.Id).Contains(((IMayHaveOrganizationUnit)e).OrganizationUnitId.Value) 
             == IsOrganizationUnitFilterEnabled; 
            expression = expression == null ? mayHaveOuFilter : CombineExpressions(expression, mayHaveOuFilter); 
        } 
 
        return expression; 
    } 

In my example, I was attempting to implement user -> roles, user -> organization based Data Filters in DbContext.

If the filter requires a "join query", it will not work because the predicate seems to run once and get fixed - so even if user changes the filter remains the same.

I think this is a related post: https://stackoverflow.com/questions/67820361/entity-framework-core-global-dynamic-query-filter/68318214#68318214

Using https://docs.microsoft.com/en-us/ef/core/modeling/dynamic-model might be a solution, but I feel like both ABP and EF don't want you to use Data Filter, EF Global Filters for anything other than simple comparisons that do not require a join (e.g., isActive == true).

Answer

sorry shobhit my code does not work

The filter expression is created once (first time entity is queried) and does not change regardless of current user.

Answer

You can refer to the ANZ documentation which is largely applicable for ABP

https://aspnetboilerplate.com/Pages/Documents/Zero/Organization-Units#common-use-cases

https://aspnetboilerplate.com/Pages/Documents/Articles%5CHow-To%5Cadd-custom-data-filter-ef-core

The second link covers the case when a User can be assigned to only one Org Unit. If your use case covers M:M assignments it gets a bit more complicate. Here is sample code (I'm a noob so FWIW):

Create a Customer UserClaimsPrincipalFactory and in it put

 public async override Task<ClaimsPrincipal> CreateAsync(IdentityUser user)
        {
            var principal = await base.CreateAsync(user);
            var organizationUnits = await _identityUserManager.GetOrganizationUnitsAsync(user);
            foreach (OrganizationUnit org in organizationUnits)
            {
                principal.Identities.First().AddClaim(new Claim("OU", org.Code));
            }
            return principal;
        }

In your DBContext

   protected override Expression<Func<TEntity, bool>> CreateFilterExpression<TEntity>()
    {
        var expression = base.CreateFilterExpression<TEntity>();

        if (typeof(IMayHaveOrganizationUnit).IsAssignableFrom(typeof(TEntity)))
        {
            var predicate = PredicateBuilder.New<OrganizationUnit>();
            if (OrgUnitCodes == null || !OrgUnitCodes.Any())
            {

                // If No organizations assigned to User, return no entities
                predicate = predicate.And(p => false);
            }
            else
            {
                foreach (var code in OrgUnitCodes)
                {
                    predicate = predicate.Or(ou => ou.Code.StartsWith(code));
                }
            }
            Expression<Func<TEntity, bool>> mayHaveOuFilter = e => !IsOrganizationUnitFilterEnabled
             || OrganizationUnits
             .Where(predicate)
             .Select(s => s.Id).Contains(((IMayHaveOrganizationUnit)e).OrganizationUnitId.Value)
             == IsOrganizationUnitFilterEnabled;
            expression = expression == null ? mayHaveOuFilter : CombineExpressions(expression, mayHaveOuFilter);
        }

        return expression;
    }
Showing 1 to 4 of 4 entries
Made with ❤️ on ABP v9.1.0-rc.1. Updated on January 17, 2025, 14:13