* Members and member types in the Management API * Add validation endpoints for members * Include validation result in service response + add unit tests * Regenerate OpenApi.json * Regenerate OpenApi.json after merge * Don't throw an exception when trying to set valid variation levels for member types * Added missing ProducesResponseType * Remove TODO, as that works * Allow creation of member with explicit key * Do not feature "parent" for member creation + add missing response type * Do not feature a "Folder" in create member type (folders are not supported) * Added missing build methods * Fixed issue with mapping --------- Co-authored-by: Bjarke Berg <mail@bergmania.dk>
304 lines
13 KiB
C#
304 lines
13 KiB
C#
using NUnit.Framework;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.Models;
|
|
using Umbraco.Cms.Core.Models.ContentEditing;
|
|
using Umbraco.Cms.Core.Models.Membership;
|
|
using Umbraco.Cms.Core.Security;
|
|
using Umbraco.Cms.Core.Services;
|
|
using Umbraco.Cms.Core.Services.OperationStatus;
|
|
using Umbraco.Cms.Tests.Common.Builders;
|
|
using Umbraco.Cms.Tests.Common.Testing;
|
|
using Umbraco.Cms.Tests.Integration.Testing;
|
|
|
|
namespace Umbraco.Cms.Tests.Integration.Umbraco.Infrastructure.Services;
|
|
|
|
[TestFixture]
|
|
[Category("Slow")]
|
|
[UmbracoTest(Database = UmbracoTestOptions.Database.NewSchemaPerTest)]
|
|
public class MemberEditingServiceTests : UmbracoIntegrationTest
|
|
{
|
|
private IMemberEditingService MemberEditingService => GetRequiredService<IMemberEditingService>();
|
|
|
|
private IMemberService MemberService => GetRequiredService<IMemberService>();
|
|
|
|
private IMemberTypeService MemberTypeService => GetRequiredService<IMemberTypeService>();
|
|
|
|
[Test]
|
|
public async Task Can_Create_Member()
|
|
{
|
|
var member = await CreateMemberAsync();
|
|
Assert.IsTrue(member.HasIdentity);
|
|
Assert.Greater(member.Id, 0);
|
|
|
|
member = await MemberEditingService.GetAsync(member.Key);
|
|
Assert.IsNotNull(member);
|
|
Assert.AreEqual("test@test.com", member.Email);
|
|
Assert.AreEqual("test", member.Username);
|
|
Assert.AreEqual("T. Est", member.Name);
|
|
Assert.IsTrue(member.IsApproved);
|
|
|
|
Assert.AreEqual("The title value", member.GetValue<string>("title"));
|
|
Assert.AreEqual("The author value", member.GetValue<string>("author"));
|
|
|
|
var memberManager = GetRequiredService<IMemberManager>();
|
|
var memberIdentityUser = await memberManager.FindByEmailAsync(member.Email);
|
|
Assert.IsNotNull(memberIdentityUser);
|
|
Assert.IsTrue(await memberManager.CheckPasswordAsync(memberIdentityUser, "SuperSecret123"));
|
|
|
|
var roles = MemberService.GetAllRoles(member.Id).ToArray();
|
|
Assert.AreEqual(1, roles.Length);
|
|
Assert.AreEqual("RoleOne", roles[0]);
|
|
}
|
|
|
|
[Test]
|
|
public async Task Can_Create_Member_With_Explicit_Key()
|
|
{
|
|
var key = Guid.NewGuid();
|
|
var member = await CreateMemberAsync(key);
|
|
Assert.IsTrue(member.HasIdentity);
|
|
Assert.Greater(member.Id, 0);
|
|
Assert.AreEqual(key, member.Key);
|
|
|
|
member = await MemberEditingService.GetAsync(member.Key);
|
|
Assert.IsNotNull(member);
|
|
}
|
|
|
|
[Test]
|
|
public async Task Can_Update_Member()
|
|
{
|
|
var member = await CreateMemberAsync();
|
|
|
|
var updateModel = new MemberUpdateModel
|
|
{
|
|
Email = "test-updated@test.com",
|
|
Username = "test-updated",
|
|
IsApproved = false,
|
|
InvariantName = "T. Est Updated",
|
|
InvariantProperties = new[]
|
|
{
|
|
new PropertyValueModel { Alias = "title", Value = "The updated title value" },
|
|
new PropertyValueModel { Alias = "author", Value = "The updated author value" }
|
|
}
|
|
};
|
|
|
|
var result = await MemberEditingService.UpdateAsync(member, updateModel, SuperUser());
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(ContentEditingOperationStatus.Success, result.Status.ContentEditingOperationStatus);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
|
|
member = result.Result.Content;
|
|
Assert.IsNotNull(member);
|
|
Assert.AreEqual("test-updated@test.com", member.Email);
|
|
Assert.AreEqual("test-updated", member.Username);
|
|
Assert.AreEqual("T. Est Updated", member.Name);
|
|
Assert.IsFalse(member.IsApproved);
|
|
|
|
Assert.AreEqual("The updated title value", member.GetValue<string>("title"));
|
|
Assert.AreEqual("The updated author value", member.GetValue<string>("author"));
|
|
}
|
|
|
|
[Test]
|
|
public async Task Can_Change_Member_Password()
|
|
{
|
|
var member = await CreateMemberAsync();
|
|
var memberManager = GetRequiredService<IMemberManager>();
|
|
|
|
var updateModel = new MemberUpdateModel
|
|
{
|
|
Email = member.Email,
|
|
Username = member.Username,
|
|
IsApproved = true,
|
|
InvariantName = member.Name,
|
|
NewPassword = "NewSuperSecret123"
|
|
};
|
|
|
|
var result = await MemberEditingService.UpdateAsync(member, updateModel, SuperUser());
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(ContentEditingOperationStatus.Success, result.Status.ContentEditingOperationStatus);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
|
|
var memberIdentityUser = await memberManager.FindByEmailAsync(member.Email);
|
|
Assert.IsNotNull(memberIdentityUser);
|
|
Assert.IsTrue(await memberManager.CheckPasswordAsync(memberIdentityUser, "NewSuperSecret123"));
|
|
}
|
|
|
|
[Test]
|
|
public async Task Can_Change_Member_Roles()
|
|
{
|
|
MemberService.AddRole("RoleTwo");
|
|
MemberService.AddRole("RoleThree");
|
|
|
|
var member = await CreateMemberAsync();
|
|
|
|
var updateModel = new MemberUpdateModel
|
|
{
|
|
Email = member.Email,
|
|
Username = member.Username,
|
|
IsApproved = true,
|
|
InvariantName = member.Name,
|
|
Roles = new [] { "RoleTwo", "RoleThree" }
|
|
};
|
|
|
|
var result = await MemberEditingService.UpdateAsync(member, updateModel, SuperUser());
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(ContentEditingOperationStatus.Success, result.Status.ContentEditingOperationStatus);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
|
|
var roles = MemberService.GetAllRoles(member.Id).ToArray();
|
|
Assert.AreEqual(2, roles.Length);
|
|
Assert.IsTrue(roles.Contains("RoleTwo"));
|
|
Assert.IsTrue(roles.Contains("RoleThree"));
|
|
}
|
|
|
|
[Test]
|
|
public async Task Can_Delete_Member()
|
|
{
|
|
var member = await CreateMemberAsync();
|
|
Assert.IsTrue(member.HasIdentity);
|
|
Assert.Greater(member.Id, 0);
|
|
|
|
member = await MemberEditingService.GetAsync(member.Key);
|
|
Assert.IsNotNull(member);
|
|
|
|
var result = await MemberEditingService.DeleteAsync(member.Key, Constants.Security.SuperUserKey);
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(ContentEditingOperationStatus.Success, result.Status.ContentEditingOperationStatus);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
|
|
member = await MemberEditingService.GetAsync(member.Key);
|
|
Assert.IsNull(member);
|
|
}
|
|
|
|
[TestCase(true)]
|
|
[TestCase(false)]
|
|
public async Task Can_Create_With_Property_Validation(bool addValidProperties)
|
|
{
|
|
IMemberType memberType = MemberTypeBuilder.CreateSimpleMemberType();
|
|
memberType.PropertyTypes.First(pt => pt.Alias == "title").Mandatory = true;
|
|
memberType.PropertyTypes.First(pt => pt.Alias == "author").ValidationRegExp = "^\\d*$";
|
|
MemberTypeService.Save(memberType);
|
|
|
|
var titleValue = addValidProperties ? "The title value" : null;
|
|
var authorValue = addValidProperties ? "12345" : "This is not a number";
|
|
|
|
var createModel = new MemberCreateModel
|
|
{
|
|
Email = "test@test.com",
|
|
Username = "test",
|
|
Password = "SuperSecret123",
|
|
IsApproved = true,
|
|
ContentTypeKey = memberType.Key,
|
|
InvariantName = "T. Est",
|
|
InvariantProperties = new[]
|
|
{
|
|
new PropertyValueModel { Alias = "title", Value = titleValue },
|
|
new PropertyValueModel { Alias = "author", Value = authorValue }
|
|
}
|
|
};
|
|
|
|
var result = await MemberEditingService.CreateAsync(createModel, SuperUser());
|
|
|
|
// success is expected regardless of property level validation - the validation error status is communicated in the attempt status (see below)
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
Assert.AreEqual(addValidProperties ? ContentEditingOperationStatus.Success : ContentEditingOperationStatus.PropertyValidationError, result.Status.ContentEditingOperationStatus);
|
|
Assert.IsNotNull(result.Result);
|
|
|
|
if (addValidProperties is false)
|
|
{
|
|
Assert.AreEqual(2, result.Result.ValidationResult.ValidationErrors.Count());
|
|
Assert.IsNotNull(result.Result.ValidationResult.ValidationErrors.FirstOrDefault(v => v.Alias == "title" && v.ErrorMessages.Length == 1));
|
|
Assert.IsNotNull(result.Result.ValidationResult.ValidationErrors.FirstOrDefault(v => v.Alias == "author" && v.ErrorMessages.Length == 1));
|
|
}
|
|
|
|
// NOTE: member creation must be successful, even if the mandatory property is missing
|
|
Assert.IsTrue(result.Result.Content!.HasIdentity);
|
|
Assert.AreEqual(titleValue, result.Result.Content!.GetValue<string>("title"));
|
|
Assert.AreEqual(authorValue, result.Result.Content!.GetValue<string>("author"));
|
|
}
|
|
|
|
[TestCase(true)]
|
|
[TestCase(false)]
|
|
public async Task Can_Update_With_Property_Validation(bool addValidProperties)
|
|
{
|
|
var member = await CreateMemberAsync();
|
|
var memberType = await MemberTypeService.GetAsync(member.ContentType.Key)!;
|
|
memberType.PropertyTypes.First(pt => pt.Alias == "title").Mandatory = true;
|
|
memberType.PropertyTypes.First(pt => pt.Alias == "author").ValidationRegExp = "^\\d*$";
|
|
await MemberTypeService.SaveAsync(memberType, Constants.Security.SuperUserKey);
|
|
|
|
var titleValue = addValidProperties ? "The title value" : null;
|
|
var authorValue = addValidProperties ? "12345" : "This is not a number";
|
|
|
|
var updateModel = new MemberUpdateModel
|
|
{
|
|
Email = member.Email,
|
|
Username = member.Username,
|
|
IsApproved = true,
|
|
InvariantName = member.Name,
|
|
InvariantProperties = new[]
|
|
{
|
|
new PropertyValueModel { Alias = "title", Value = titleValue },
|
|
new PropertyValueModel { Alias = "author", Value = authorValue }
|
|
}
|
|
};
|
|
|
|
var result = await MemberEditingService.UpdateAsync(member, updateModel, SuperUser());
|
|
|
|
// success is expected regardless of property level validation - the validation error status is communicated in the attempt status (see below)
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
Assert.AreEqual(addValidProperties ? ContentEditingOperationStatus.Success : ContentEditingOperationStatus.PropertyValidationError, result.Status.ContentEditingOperationStatus);
|
|
Assert.IsNotNull(result.Result);
|
|
|
|
if (addValidProperties is false)
|
|
{
|
|
Assert.AreEqual(2, result.Result.ValidationResult.ValidationErrors.Count());
|
|
Assert.IsNotNull(result.Result.ValidationResult.ValidationErrors.FirstOrDefault(v => v.Alias == "title" && v.ErrorMessages.Length == 1));
|
|
Assert.IsNotNull(result.Result.ValidationResult.ValidationErrors.FirstOrDefault(v => v.Alias == "author" && v.ErrorMessages.Length == 1));
|
|
}
|
|
|
|
// NOTE: member update must be successful, even if the mandatory property is missing
|
|
Assert.AreEqual(titleValue, result.Result.Content!.GetValue<string>("title"));
|
|
Assert.AreEqual(authorValue, result.Result.Content!.GetValue<string>("author"));
|
|
}
|
|
|
|
private IUser SuperUser() => GetRequiredService<IUserService>().GetAsync(Constants.Security.SuperUserKey).GetAwaiter().GetResult();
|
|
|
|
private async Task<IMember> CreateMemberAsync(Guid? key = null)
|
|
{
|
|
IMemberType memberType = MemberTypeBuilder.CreateSimpleMemberType();
|
|
MemberTypeService.Save(memberType);
|
|
MemberService.AddRole("RoleOne");
|
|
|
|
var createModel = new MemberCreateModel
|
|
{
|
|
Key = key,
|
|
Email = "test@test.com",
|
|
Username = "test",
|
|
Password = "SuperSecret123",
|
|
IsApproved = true,
|
|
ContentTypeKey = memberType.Key,
|
|
Roles = new [] { "RoleOne" },
|
|
InvariantName = "T. Est",
|
|
InvariantProperties = new[]
|
|
{
|
|
new PropertyValueModel { Alias = "title", Value = "The title value" },
|
|
new PropertyValueModel { Alias = "author", Value = "The author value" }
|
|
}
|
|
};
|
|
|
|
var result = await MemberEditingService.CreateAsync(createModel, SuperUser());
|
|
Assert.IsTrue(result.Success);
|
|
Assert.AreEqual(ContentEditingOperationStatus.Success, result.Status.ContentEditingOperationStatus);
|
|
Assert.AreEqual(MemberEditingOperationStatus.Success, result.Status.MemberEditingOperationStatus);
|
|
|
|
var member = result.Result.Content;
|
|
Assert.IsNotNull(member);
|
|
Assert.IsTrue(member.HasIdentity);
|
|
Assert.Greater(member.Id, 0);
|
|
|
|
return await MemberEditingService.GetAsync(member.Key) ?? throw new ApplicationException("Created member could not be retrieved");
|
|
}
|
|
}
|