Files
server/src/Core/Services/Implementations/OrganizationService.cs

202 lines
7.4 KiB
C#
Raw Normal View History

using System;
using System.Linq;
using System.Threading.Tasks;
using Bit.Core.Repositories;
using Bit.Core.Models.Business;
2017-03-08 21:45:08 -05:00
using Bit.Core.Models.Table;
using Bit.Core.Utilities;
using Bit.Core.Exceptions;
2017-03-09 23:58:43 -05:00
using System.Collections.Generic;
namespace Bit.Core.Services
{
public class OrganizationService : IOrganizationService
{
private readonly IOrganizationRepository _organizationRepository;
private readonly IOrganizationUserRepository _organizationUserRepository;
2017-03-09 23:58:43 -05:00
private readonly ISubvaultRepository _subvaultRepository;
private readonly ISubvaultUserRepository _subvaultUserRepository;
2017-03-04 21:28:41 -05:00
private readonly IUserRepository _userRepository;
public OrganizationService(
IOrganizationRepository organizationRepository,
2017-03-04 21:28:41 -05:00
IOrganizationUserRepository organizationUserRepository,
2017-03-09 23:58:43 -05:00
ISubvaultRepository subvaultRepository,
ISubvaultUserRepository subvaultUserRepository,
2017-03-04 21:28:41 -05:00
IUserRepository userRepository)
{
_organizationRepository = organizationRepository;
_organizationUserRepository = organizationUserRepository;
2017-03-09 23:58:43 -05:00
_subvaultRepository = subvaultRepository;
_subvaultUserRepository = subvaultUserRepository;
2017-03-04 21:28:41 -05:00
_userRepository = userRepository;
}
2017-03-04 21:28:41 -05:00
public async Task<Tuple<Organization, OrganizationUser>> SignUpAsync(OrganizationSignup signup)
{
2017-03-04 21:28:41 -05:00
var plan = StaticStore.Plans.FirstOrDefault(p => p.Type == signup.Plan);
if(plan == null)
{
throw new BadRequestException("Plan not found.");
}
var organization = new Organization
{
2017-03-04 21:28:41 -05:00
Name = signup.Name,
UserId = signup.Owner.Id,
PlanType = plan.Type,
MaxUsers = plan.MaxUsers,
PlanTrial = plan.Trial.HasValue,
PlanPrice = plan.Trial.HasValue ? 0 : plan.Price,
PlanRenewalPrice = plan.Price,
Plan = plan.ToString(),
CreationDate = DateTime.UtcNow,
RevisionDate = DateTime.UtcNow
};
if(plan.Trial.HasValue)
{
organization.PlanRenewalDate = DateTime.UtcNow.Add(plan.Trial.Value);
}
else if(plan.Cycle != null)
{
2017-03-08 22:33:31 -05:00
organization.PlanRenewalDate = DateTime.UtcNow.Add(plan.Cycle(DateTime.UtcNow));
}
await _organizationRepository.CreateAsync(organization);
try
{
var orgUser = new OrganizationUser
{
OrganizationId = organization.Id,
2017-03-04 21:28:41 -05:00
UserId = signup.Owner.Id,
Email = signup.Owner.Email,
Key = signup.OwnerKey,
Type = Enums.OrganizationUserType.Owner,
Status = Enums.OrganizationUserStatusType.Confirmed,
CreationDate = DateTime.UtcNow,
RevisionDate = DateTime.UtcNow
};
await _organizationUserRepository.CreateAsync(orgUser);
return new Tuple<Organization, OrganizationUser>(organization, orgUser);
}
catch
{
await _organizationRepository.DeleteAsync(organization);
throw;
}
}
2017-03-04 21:28:41 -05:00
2017-03-11 22:42:27 -05:00
public async Task<OrganizationUser> InviteUserAsync(Guid organizationId, string email,
IEnumerable<SubvaultUser> subvaults)
2017-03-04 21:28:41 -05:00
{
var orgUser = new OrganizationUser
{
OrganizationId = organizationId,
UserId = null,
Email = email,
Key = null,
Type = Enums.OrganizationUserType.User,
Status = Enums.OrganizationUserStatusType.Invited,
CreationDate = DateTime.UtcNow,
RevisionDate = DateTime.UtcNow
};
await _organizationUserRepository.CreateAsync(orgUser);
2017-03-11 22:42:27 -05:00
await SaveUserSubvaultsAsync(orgUser, subvaults, true);
2017-03-04 21:28:41 -05:00
// TODO: send email
return orgUser;
}
public async Task<OrganizationUser> AcceptUserAsync(Guid organizationUserId, User user, string token)
{
var orgUser = await _organizationUserRepository.GetByIdAsync(organizationUserId);
if(orgUser.Email != user.Email)
{
throw new BadRequestException("User invalid.");
}
// TODO: validate token
orgUser.Status = Enums.OrganizationUserStatusType.Accepted;
orgUser.UserId = orgUser.Id;
orgUser.Email = null;
await _organizationUserRepository.ReplaceAsync(orgUser);
// TODO: send email
return orgUser;
}
public async Task<OrganizationUser> ConfirmUserAsync(Guid organizationUserId, string key)
{
var orgUser = await _organizationUserRepository.GetByIdAsync(organizationUserId);
if(orgUser.Status != Enums.OrganizationUserStatusType.Accepted)
{
throw new BadRequestException("User not accepted.");
}
orgUser.Status = Enums.OrganizationUserStatusType.Confirmed;
orgUser.Key = key;
orgUser.Email = null;
await _organizationUserRepository.ReplaceAsync(orgUser);
// TODO: send email
return orgUser;
}
2017-03-09 23:58:43 -05:00
2017-03-11 22:42:27 -05:00
public async Task SaveUserAsync(OrganizationUser user, IEnumerable<SubvaultUser> subvaults)
2017-03-09 23:58:43 -05:00
{
if(user.Id.Equals(default(Guid)))
{
throw new BadRequestException("Invite the user first.");
}
await _organizationUserRepository.ReplaceAsync(user);
2017-03-11 22:42:27 -05:00
await SaveUserSubvaultsAsync(user, subvaults, false);
}
2017-03-09 23:58:43 -05:00
2017-03-11 22:42:27 -05:00
private async Task SaveUserSubvaultsAsync(OrganizationUser user, IEnumerable<SubvaultUser> subvaults, bool newUser)
{
2017-03-13 22:54:24 -04:00
if(subvaults == null)
{
subvaults = new List<SubvaultUser>();
}
2017-03-09 23:58:43 -05:00
var orgSubvaults = await _subvaultRepository.GetManyByOrganizationIdAsync(user.OrganizationId);
2017-03-11 22:42:27 -05:00
var currentUserSubvaults = newUser ? null : await _subvaultUserRepository.GetManyByOrganizationUserIdAsync(user.Id);
2017-03-09 23:58:43 -05:00
// Let's make sure all these belong to this user and organization.
2017-03-11 22:42:27 -05:00
var filteredSubvaults = subvaults.Where(s => orgSubvaults.Any(os => os.Id == s.SubvaultId));
2017-03-09 23:58:43 -05:00
foreach(var subvault in filteredSubvaults)
{
2017-03-13 22:54:24 -04:00
var existingSubvaultUser = currentUserSubvaults?.FirstOrDefault(cs => cs.SubvaultId == subvault.SubvaultId);
if(existingSubvaultUser != null)
{
subvault.Id = existingSubvaultUser.Id;
subvault.CreationDate = existingSubvaultUser.CreationDate;
}
2017-03-11 22:42:27 -05:00
subvault.OrganizationUserId = user.Id;
2017-03-09 23:58:43 -05:00
await _subvaultUserRepository.UpsertAsync(subvault);
}
2017-03-11 22:42:27 -05:00
if(!newUser)
2017-03-09 23:58:43 -05:00
{
2017-03-13 22:54:24 -04:00
var subvaultsToDelete = currentUserSubvaults.Where(cs =>
!filteredSubvaults.Any(s => s.SubvaultId == cs.SubvaultId));
2017-03-11 22:42:27 -05:00
foreach(var subvault in subvaultsToDelete)
{
await _subvaultUserRepository.DeleteAsync(subvault);
}
2017-03-09 23:58:43 -05:00
}
}
}
}