Files
Umbraco-CMS/tests/Umbraco.Tests.Integration/Umbraco.Infrastructure/Services/MemberEditingServiceTests.cs
Kenn Jacobsen 71b3076de9 Members and member types in the Management API (#15662)
* 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>
2024-02-05 06:42:07 +01:00

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");
}
}