Skip to content

Commit cb7462b

Browse files
authored
Merge pull request #22792 from suhaib-mousa/features/tenant-name-validator
Introduce ITenantNameValidator for extensible tenant name validation
2 parents 991e54e + bb977c4 commit cb7462b

File tree

4 files changed

+91
-22
lines changed

4 files changed

+91
-22
lines changed
Lines changed: 26 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
using System.Threading.Tasks;
2+
using Volo.Abp.DependencyInjection;
3+
4+
namespace Volo.Abp.TenantManagement;
5+
6+
public class AbpTenantValidator : ITenantValidator, ITransientDependency
7+
{
8+
protected ITenantRepository TenantRepository { get; }
9+
10+
public AbpTenantValidator(ITenantRepository tenantRepository)
11+
{
12+
TenantRepository = tenantRepository;
13+
}
14+
15+
public virtual async Task ValidateAsync(Tenant tenant)
16+
{
17+
Check.NotNullOrWhiteSpace(tenant.Name, nameof(tenant.Name));
18+
Check.NotNullOrWhiteSpace(tenant.NormalizedName, nameof(tenant.NormalizedName));
19+
20+
var owner = await TenantRepository.FindByNameAsync(tenant.NormalizedName);
21+
if (owner != null && owner.Id != tenant.Id)
22+
{
23+
throw new BusinessException("Volo.Abp.TenantManagement:DuplicateTenantName").WithData("Name", tenant.NormalizedName);
24+
}
25+
}
26+
}
Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
using System.Threading.Tasks;
2+
3+
namespace Volo.Abp.TenantManagement;
4+
5+
public interface ITenantValidator
6+
{
7+
Task ValidateAsync(Tenant tenant);
8+
}
Lines changed: 10 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,4 @@
1-
using System;
2-
using System.Threading.Tasks;
3-
using Volo.Abp.Caching;
1+
using System.Threading.Tasks;
42
using Volo.Abp.Domain.Services;
53
using Volo.Abp.EventBus.Local;
64
using Volo.Abp.MultiTenancy;
@@ -9,16 +7,16 @@ namespace Volo.Abp.TenantManagement;
97

108
public class TenantManager : DomainService, ITenantManager
119
{
12-
protected ITenantRepository TenantRepository { get; }
10+
protected ITenantValidator TenantValidator { get; }
1311
protected ITenantNormalizer TenantNormalizer { get; }
1412
protected ILocalEventBus LocalEventBus { get; }
1513

1614
public TenantManager(
17-
ITenantRepository tenantRepository,
15+
ITenantValidator tenantValidator,
1816
ITenantNormalizer tenantNormalizer,
1917
ILocalEventBus localEventBus)
2018
{
21-
TenantRepository = tenantRepository;
19+
TenantValidator = tenantValidator;
2220
TenantNormalizer = tenantNormalizer;
2321
LocalEventBus = localEventBus;
2422
}
@@ -27,30 +25,20 @@ public virtual async Task<Tenant> CreateAsync(string name)
2725
{
2826
Check.NotNull(name, nameof(name));
2927

30-
var normalizedName = TenantNormalizer.NormalizeName(name);
31-
await ValidateNameAsync(normalizedName);
32-
return new Tenant(GuidGenerator.Create(), name, normalizedName);
28+
var tenant = new Tenant(GuidGenerator.Create(), name, TenantNormalizer.NormalizeName(name));
29+
await TenantValidator.ValidateAsync(tenant);
30+
return tenant;
3331
}
3432

3533
public virtual async Task ChangeNameAsync(Tenant tenant, string name)
3634
{
3735
Check.NotNull(tenant, nameof(tenant));
3836
Check.NotNull(name, nameof(name));
3937

40-
var normalizedName = TenantNormalizer.NormalizeName(name);
41-
42-
await ValidateNameAsync(normalizedName, tenant.Id);
4338
await LocalEventBus.PublishAsync(new TenantChangedEvent(tenant.Id, tenant.NormalizedName));
44-
tenant.SetName(name);
45-
tenant.SetNormalizedName(normalizedName);
46-
}
4739

48-
protected virtual async Task ValidateNameAsync(string normalizeName, Guid? expectedId = null)
49-
{
50-
var tenant = await TenantRepository.FindByNameAsync(normalizeName);
51-
if (tenant != null && tenant.Id != expectedId)
52-
{
53-
throw new BusinessException("Volo.Abp.TenantManagement:DuplicateTenantName").WithData("Name", normalizeName);
54-
}
40+
tenant.SetName(name);
41+
tenant.SetNormalizedName( TenantNormalizer.NormalizeName(name));
42+
await TenantValidator.ValidateAsync(tenant);
5543
}
5644
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
using System;
2+
using System.Threading.Tasks;
3+
using Shouldly;
4+
using Xunit;
5+
6+
namespace Volo.Abp.TenantManagement;
7+
8+
public class TenantValidator_Tests : AbpTenantManagementDomainTestBase
9+
{
10+
private readonly TenantManager _tenantManager;
11+
private readonly ITenantRepository _tenantRepository;
12+
13+
public TenantValidator_Tests()
14+
{
15+
_tenantManager = GetRequiredService<TenantManager>();
16+
_tenantRepository = GetRequiredService<ITenantRepository>();
17+
}
18+
19+
[Fact]
20+
public async Task Should_Throw_If_Name_Is_Null()
21+
{
22+
await Assert.ThrowsAsync<ArgumentException>(() => _tenantManager.CreateAsync(""));
23+
}
24+
25+
[Fact]
26+
public async Task Should_Throw_If_Duplicate_Name()
27+
{
28+
await Assert.ThrowsAsync<BusinessException>(() => _tenantManager.CreateAsync("VOLOSOFT"));
29+
30+
var tenant = await _tenantRepository.FindByNameAsync("ABP");
31+
await Assert.ThrowsAsync<BusinessException>(() => _tenantManager.ChangeNameAsync(tenant, "VOLOSOFT"));
32+
}
33+
34+
[Fact]
35+
public async Task Should_Not_Throw_For_Unique_Name()
36+
{
37+
var tenant = await _tenantManager.CreateAsync("VOLOSOFT2");
38+
await _tenantRepository.InsertAsync(tenant);
39+
40+
tenant = await _tenantRepository.FindByNameAsync("ABP");
41+
await _tenantManager.ChangeNameAsync(tenant, "VOLOSOFT3");
42+
await _tenantRepository.UpdateAsync(tenant);
43+
44+
tenant = await _tenantRepository.FindByNameAsync("VOLOSOFT3");
45+
tenant.ShouldNotBeNull();
46+
}
47+
}

0 commit comments

Comments
 (0)