using Bit.Core; using Bit.Core.AdminConsole.OrganizationFeatures.Groups.Interfaces; using Bit.Core.AdminConsole.Repositories; using Bit.Core.Enums; using Bit.Core.Exceptions; using Bit.Core.Repositories; using Bit.Core.Services; using Bit.Scim.Groups.Interfaces; using Bit.Scim.Models; using Bit.Scim.Utilities; using Microsoft.AspNetCore.Authorization; using Microsoft.AspNetCore.Mvc; namespace Bit.Scim.Controllers.v2; [Authorize("Scim")] [Route("v2/{organizationId}/groups")] [Produces("application/scim+json")] [ExceptionHandlerFilter] public class GroupsController : Controller { private readonly IGroupRepository _groupRepository; private readonly IOrganizationRepository _organizationRepository; private readonly IGetGroupsListQuery _getGroupsListQuery; private readonly IDeleteGroupCommand _deleteGroupCommand; private readonly IPatchGroupCommand _patchGroupCommand; private readonly IPatchGroupCommandvNext _patchGroupCommandvNext; private readonly IPostGroupCommand _postGroupCommand; private readonly IPutGroupCommand _putGroupCommand; private readonly IFeatureService _featureService; public GroupsController( IGroupRepository groupRepository, IOrganizationRepository organizationRepository, IGetGroupsListQuery getGroupsListQuery, IDeleteGroupCommand deleteGroupCommand, IPatchGroupCommand patchGroupCommand, IPatchGroupCommandvNext patchGroupCommandvNext, IPostGroupCommand postGroupCommand, IPutGroupCommand putGroupCommand, IFeatureService featureService ) { _groupRepository = groupRepository; _organizationRepository = organizationRepository; _getGroupsListQuery = getGroupsListQuery; _deleteGroupCommand = deleteGroupCommand; _patchGroupCommand = patchGroupCommand; _patchGroupCommandvNext = patchGroupCommandvNext; _postGroupCommand = postGroupCommand; _putGroupCommand = putGroupCommand; _featureService = featureService; } [HttpGet("{id}")] public async Task Get(Guid organizationId, Guid id) { var group = await _groupRepository.GetByIdAsync(id); if (group == null || group.OrganizationId != organizationId) { throw new NotFoundException("Group not found."); } return Ok(new ScimGroupResponseModel(group)); } [HttpGet("")] public async Task Get( Guid organizationId, [FromQuery] string filter, [FromQuery] int? count, [FromQuery] int? startIndex) { var groupsListQueryResult = await _getGroupsListQuery.GetGroupsListAsync(organizationId, filter, count, startIndex); var scimListResponseModel = new ScimListResponseModel { Resources = groupsListQueryResult.groupList.Select(g => new ScimGroupResponseModel(g)).ToList(), ItemsPerPage = count.GetValueOrDefault(groupsListQueryResult.groupList.Count()), TotalResults = groupsListQueryResult.totalResults, StartIndex = startIndex.GetValueOrDefault(1), }; return Ok(scimListResponseModel); } [HttpPost("")] public async Task Post(Guid organizationId, [FromBody] ScimGroupRequestModel model) { var organization = await _organizationRepository.GetByIdAsync(organizationId); var group = await _postGroupCommand.PostGroupAsync(organization, model); var scimGroupResponseModel = new ScimGroupResponseModel(group); return new CreatedResult(Url.Action(nameof(Get), new { group.OrganizationId, group.Id }), scimGroupResponseModel); } [HttpPut("{id}")] public async Task Put(Guid organizationId, Guid id, [FromBody] ScimGroupRequestModel model) { var organization = await _organizationRepository.GetByIdAsync(organizationId); var group = await _putGroupCommand.PutGroupAsync(organization, id, model); var response = new ScimGroupResponseModel(group); return Ok(response); } [HttpPatch("{id}")] public async Task Patch(Guid organizationId, Guid id, [FromBody] ScimPatchModel model) { if (_featureService.IsEnabled(FeatureFlagKeys.ShortcutDuplicatePatchRequests)) { var group = await _groupRepository.GetByIdAsync(id); if (group == null || group.OrganizationId != organizationId) { throw new NotFoundException("Group not found."); } await _patchGroupCommandvNext.PatchGroupAsync(group, model); return new NoContentResult(); } var organization = await _organizationRepository.GetByIdAsync(organizationId); await _patchGroupCommand.PatchGroupAsync(organization, id, model); return new NoContentResult(); } [HttpDelete("{id}")] public async Task Delete(Guid organizationId, Guid id) { await _deleteGroupCommand.DeleteGroupAsync(organizationId, id, EventSystemUser.SCIM); return new NoContentResult(); } }