diff --git a/AsbCloudInfrastructure/Services/DrillingProgram/DrillingProgramService.cs b/AsbCloudInfrastructure/Services/DrillingProgram/DrillingProgramService.cs index 3771ad34..d53797cb 100644 --- a/AsbCloudInfrastructure/Services/DrillingProgram/DrillingProgramService.cs +++ b/AsbCloudInfrastructure/Services/DrillingProgram/DrillingProgramService.cs @@ -24,6 +24,8 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram private readonly IWellService wellService; private readonly IConfiguration configuration; private readonly IBackgroundWorkerService backgroundWorker; + //email + private readonly IEmailService emailService; private readonly string connectionString; private const int idFileCategoryDrillingProgram = 1000; @@ -52,8 +54,9 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram IUserService userService, IWellService wellService, IConfiguration configuration, - IBackgroundWorkerService backgroundWorker) - { + IBackgroundWorkerService backgroundWorker, + IEmailService emailService) + { this.context = context; this.fileService = fileService; this.userService = userService; @@ -61,6 +64,7 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram this.configuration = configuration; this.backgroundWorker = backgroundWorker; this.connectionString = configuration.GetConnectionString("DefaultConnection"); + this.emailService = emailService; } public async Task> GetAvailableUsers(int idWell, CancellationToken token = default) @@ -154,7 +158,8 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram await TryEnqueueMakeProgramAsync(idWell, state, token); return state; } - + + public async Task AddFile(int idWell, int idFileCategory, int idUser, string fileFullName, System.IO.Stream fileStream, CancellationToken token = default) { var part = await context.DrillingProgramParts @@ -176,6 +181,35 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram token); await RemoveDrillingProgramAsync(part.IdWell, token); + var well = await context.Wells + .FirstOrDefaultAsync(x => x.Id == idWell, token); + var cluster = await context.Clusters + .FirstOrDefaultAsync(x => x.Wells == well, token); + + var deposit = await context.Deposits + .FirstOrDefaultAsync(x => x.Clusters == cluster, token); + var partApprovers = await context.RelationDrillingProgramPartUsers + .Where(y => y.IdDrillingProgramPart == part.Id & y.IdUserRole == idUserRoleApprover).ToListAsync(token); + foreach (var partApprover in partApprovers) + { + var approver = await context.Users + .FirstOrDefaultAsync(x => x.Id == partApprover.IdUser, token); + + var bodyHtml = new MailCoordinating() + { + idWell=idWell, + idDocument = result.Id, + documentName = fileFullName, + wellName = well.Caption, + clusterName = cluster.Caption, + fieldName = deposit.Caption, + userName = $"{approver.Name} {approver.Surname}" + + + }; + emailService.EnqueueSend(new List {approver.Email}, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + //emailService.EnqueueSend(new List { "79827873134@yandex.ru" }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + } return result.Id; } @@ -242,10 +276,35 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram IdDrillingProgramPart = part.Id, IdUserRole = idUserRole, }; - + var well = await context.Wells + .FirstOrDefaultAsync(x => x.Id == idWell, token); + var cluster = await context.Clusters + .FirstOrDefaultAsync(x => x.Wells == well, token); + context.RelationDrillingProgramPartUsers.Add(newRelation); + var deposit = await context.Deposits + .FirstOrDefaultAsync(x => x.Clusters == cluster, token); + var documentCategory = await context.FileCategories + .FirstOrDefaultAsync(x => x.Id == part.IdFileCategory, token); context.RelationDrillingProgramPartUsers.Add(newRelation); if (idUserRole == idUserRoleApprover) await RemoveDrillingProgramAsync(part.IdWell, token); + // проверяем роль пользователя - если публикатор формируем для него сообщение на отправку + if (idUserRole == idUserRolePublisher) + { + //создаем тело письма + var bodyHtml = new MailUserPublisher() + { + idWell=idWell, + userName = $"{user.Name} {user.Surname}", + wellName = well.Caption, + clusterName = cluster.Caption, + fieldName = deposit.Caption, + documentCategory = documentCategory.Name + }; + + emailService.EnqueueSend(new List {user.Email}, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + //emailService.EnqueueSend(new List { "79827873134@yandex.ru" }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + } return await context.SaveChangesAsync(token); } @@ -263,6 +322,9 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram return await context.SaveChangesAsync(token); } + // 4 trigger?? + + public async Task AddOrReplaceFileMarkAsync(FileMarkDto fileMarkDto, int idUser, CancellationToken token) { if (fileMarkDto.IdMarkType != idMarkTypeApprove && @@ -297,7 +359,74 @@ namespace AsbCloudInfrastructure.Services.DrillingProgram .ConfigureAwait(false); if (fileMarkDto.IdMarkType == idMarkTypeReject) + { await RemoveDrillingProgramAsync(fileInfo.IdWell, token); + + var user = await context.Users + .FirstOrDefaultAsync(x => x.Id == idUser, token); + var well = await context.Wells + .FirstOrDefaultAsync(x => x.Id == part.IdWell, token); + var cluster = await context.Clusters + .FirstOrDefaultAsync(x => x.Wells == well, token); + + var deposit = await context.Deposits + .FirstOrDefaultAsync(x => x.Clusters == cluster, token); + var document = await context.Files + .FirstOrDefaultAsync(x => x.Id == fileMarkDto.IdFile, token); + var author = await context.Users + .FirstOrDefaultAsync(x => x.Id == document.IdAuthor, token); + + var bodyHtml = new PublisherRejected + { + idWell=well.Id, + idDocument=document.Id, + coordinatingName = $"{user.Name} {user.Surname}", + coordinatingComment = fileMarkDto.Comment, + wellName = well.Caption, + clusterName = cluster.Caption, + fieldName = deposit.Caption, + documentName = document.Name, + userName = $"{author.Name} {author.Surname}" + }; + + emailService.EnqueueSend(new List { author.Email }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + //emailService.EnqueueSend(new List { "79827873134@yandex.ru" }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + } + + // если все части согласованы уведомляем публикатора + if (part.RelatedUsers + .Where(u => u.IdUserRole == idUserRoleApprover) + .All(user => fileInfo.FileMarks + .Any(mark => mark.IdMarkType == idMarkTypeApprove && mark.User.Id == user.IdUser))) + { + var well = await context.Wells + .FirstOrDefaultAsync(x => x.Id == part.IdWell, token); + var cluster = await context.Clusters + .FirstOrDefaultAsync(x => x.Wells == well, token); + + var deposit = await context.Deposits + .FirstOrDefaultAsync(x => x.Clusters == cluster, token); + var document = await context.Files + .FirstOrDefaultAsync(x => x.Id == fileMarkDto.IdFile, token); + var author = await context.Users + .FirstOrDefaultAsync(x => x.Id == document.IdAuthor, token); + var documentCategory = await context.FileCategories + .FirstOrDefaultAsync(x => x.Id == part.IdFileCategory, token); + var bodyHtml = new AllApprovals + { + idWell=well.Id, + idDocument=document.Id, + documentCategory = documentCategory.Name, + wellName = well.Caption, + clusterName = cluster.Caption, + fieldName = deposit.Caption, + documentName = document.Name, + userName = $"{author.Name} {author.Surname}" + }; + emailService.EnqueueSend(new List { author.Email }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + //emailService.EnqueueSend(new List { "79827873134@yandex.ru" }, bodyHtml.mailSubject, bodyHtml.GetBodyHTML()); + } + return result; } diff --git a/AsbCloudInfrastructure/Services/Email/CreationBody.cs b/AsbCloudInfrastructure/Services/Email/CreationBody.cs new file mode 100644 index 00000000..25405f55 --- /dev/null +++ b/AsbCloudInfrastructure/Services/Email/CreationBody.cs @@ -0,0 +1,96 @@ +using System; +using System.Collections.Generic; + +namespace AsbCloudInfrastructure +{ + + abstract class BodyCreation + { + public int idWell { get; set; } + public int idDocument { get; set; } + public string mailSubject { get; set; } = "#fieldName, #clusterName, #wellName. Программа бурения. #action."; + public string wellName { get; set; } + public string clusterName { get; set; } + public string fieldName { get; set; } + public string documentCategory { get; set; } + public string documentName { get; set; } + public string coordinatingName { get; set; } + public string coordinatingComment { get; set; } + public string userName { get; set; } + public static string logo { get; } = ""; + public static string bodySign { get; } = $"



" + + $"{companyName}
Это письмо сформировано автоматически.
Для получения помощи по работе на портале " + + $"{webPlatform} обращайтесь по адресу {supportMail}
"; + public static string webPlatform { get; } = ""; + public static string companyName { get; } = "ООО \"НафтаГаз\""; + public static string supportMail { get; } = "support@autodrilling.ru"; + public abstract string GetBodyHTML(); + + } + + class MailUserPublisher : BodyCreation + { + public override string GetBodyHTML() + { + mailSubject = mailSubject.Replace("#fieldName", fieldName); + mailSubject = mailSubject.Replace("#clusterName", clusterName); + mailSubject = mailSubject.Replace("#wellName", fieldName); + mailSubject = mailSubject.Replace("#action", "Добавление публикатора"); + return $"
Здравствуйте, {userName}.

< font size = \"5\">На портале " + + $"{webPlatform} началось создание программы бурения скважины {wellName}, " + + $"куст {clusterName}, месторождение {fieldName}." + + $"

От вас ожидается загрузка на портал документа «{documentCategory}» в формате excel (*.xlsx)." + bodySign; + } + + + } + + class MailCoordinating : BodyCreation + { + public override string GetBodyHTML() + { + mailSubject = mailSubject.Replace("#fieldName", fieldName); + mailSubject = mailSubject.Replace("#clusterName", clusterName); + mailSubject = mailSubject.Replace("#wellName", fieldName); + mailSubject = mailSubject.Replace("#action", "Документ на согласование"); + return $"
Здравствуйте, {userName}.

На портал " + + $"{webPlatform} загружен документ " + + $" для согласования при создании программы бурения скважины , куст ({clusterName})" + + $", месторождение ({fieldName})." + bodySign; + } + } + + class PublisherRejected : BodyCreation + { + public override string GetBodyHTML() + { + mailSubject = mailSubject.Replace("#fieldName", fieldName); + mailSubject = mailSubject.Replace("#clusterName", clusterName); + mailSubject = mailSubject.Replace("#wellName", fieldName); + mailSubject = mailSubject.Replace("#action", "Документ отклонен"); + return $"
Здравствуйте, {userName}.

На портале " + + $"{webPlatform} отклонен загруженный вами документ
" + + $" по программе бурения скважины , куст " + + $"({clusterName}), месторождение ({fieldName}). Комментарий согласующего " + + $"({coordinatingName}):

{coordinatingComment}" + bodySign; + } + } + + class AllApprovals : BodyCreation + { + public override string GetBodyHTML() + { + mailSubject = mailSubject.Replace("#fieldName", fieldName); + mailSubject = mailSubject.Replace("#clusterName", clusterName); + mailSubject = mailSubject.Replace("#wellName", fieldName); + mailSubject = mailSubject.Replace("#action", "Все документы согласованы"); + return $"
Здравствуйте, {userName}.

На портале " + + $"{webPlatform} полностью согласован документ " + + $"
{documentName} " + + $"по программе бурения скважины , куст ({clusterName}), " + + $"месторождение ({fieldName}).
От вас ожидается загрузка на портал документа " + + $"«{documentCategory}» в формате excel (*.xlsx)." + bodySign; + } + } +} + diff --git a/AsbCloudInfrastructure/Services/EmailService.cs b/AsbCloudInfrastructure/Services/Email/EmailService.cs similarity index 96% rename from AsbCloudInfrastructure/Services/EmailService.cs rename to AsbCloudInfrastructure/Services/Email/EmailService.cs index dac8e438..d71543bd 100644 --- a/AsbCloudInfrastructure/Services/EmailService.cs +++ b/AsbCloudInfrastructure/Services/Email/EmailService.cs @@ -76,8 +76,9 @@ namespace AsbCloudInfrastructure.Services foreach (var mailAddress in mailAddresses) message.To.Add(mailAddress); - message.Subject = subject; + message.Body = htmlBody; + message.Subject = subject; message.IsBodyHtml = true; var client = new SmtpClient(smtpServer); @@ -86,9 +87,9 @@ namespace AsbCloudInfrastructure.Services client.Credentials = new System.Net.NetworkCredential(sender, smtpPassword); // TODO: uncomment next when tested - //await client.SendMailAsync(message, token); + await client.SendMailAsync(message, token); await Task.Delay(0); - Trace.TraceInformation($"Send email to {string.Join(',', addresses)} subj:{subject}"); + Trace.TraceInformation($"Send email to {string.Join(',', addresses)} subj:{subject} html body count {htmlBody.Count()}"); }; return func; } diff --git a/AsbCloudInfrastructure/Services/Email/logo_32.png b/AsbCloudInfrastructure/Services/Email/logo_32.png new file mode 100644 index 00000000..827f705d Binary files /dev/null and b/AsbCloudInfrastructure/Services/Email/logo_32.png differ diff --git a/AsbCloudWebApi.Tests/ServicesTests/DrillingProgramServiceTest.cs b/AsbCloudWebApi.Tests/ServicesTests/DrillingProgramServiceTest.cs index ffd39dd2..29eec8c9 100644 --- a/AsbCloudWebApi.Tests/ServicesTests/DrillingProgramServiceTest.cs +++ b/AsbCloudWebApi.Tests/ServicesTests/DrillingProgramServiceTest.cs @@ -83,6 +83,7 @@ namespace AsbCloudWebApi.Tests.ServicesTests private readonly Mock wellServiceMock; private readonly Mock configurationMock; private readonly Mock backgroundWorkerMock; + private readonly Mock emailService; public DrillingProgramServiceTest() { @@ -112,7 +113,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var users = await service.GetAvailableUsers(idWell, CancellationToken.None); @@ -128,7 +130,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var result = await service.AddPartsAsync(idWell, new int[] { 1001, 1002 }, CancellationToken.None); @@ -146,7 +149,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var result = await service.RemovePartsAsync(idWell, new int[] { 1005 }, CancellationToken.None); @@ -168,7 +172,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var result = await service.AddUserAsync(idWell, 1001, publisher1.Id, 1, CancellationToken.None); @@ -201,7 +206,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var result = await service.RemoveUserAsync(idWell, idFileCategory, publisher1.Id, idUserRole, CancellationToken.None); @@ -226,7 +232,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var fileMark = new FileMarkDto { @@ -256,7 +263,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var fileMark = new FileMarkDto { IdFile = file1001.Id, @@ -293,7 +301,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var fileMark = new FileMarkDto { @@ -319,7 +328,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var state = await service.GetStateAsync(idWell, publisher1.Id, CancellationToken.None); @@ -345,7 +355,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var state = await service.GetStateAsync(idWell, publisher1.Id, CancellationToken.None); @@ -374,7 +385,8 @@ namespace AsbCloudWebApi.Tests.ServicesTests userServiceMock.Object, wellServiceMock.Object, configurationMock.Object, - backgroundWorkerMock.Object); + backgroundWorkerMock.Object, + emailService.Object); var state = await service.GetStateAsync(idWell, publisher1.Id, CancellationToken.None);