2 Commits 
		
	
	
		
			604b117358
			...
			93d7201666
		
	
    
				 22 changed files with 2077 additions and 219 deletions
			
			
		- 
					2CDP-LTS-80.sln
- 
					0CDP-LTS-80/Audit/AuditDB.cs
- 
					0CDP-LTS-80/Audit/AuditDocument.cs
- 
					263CDP-LTS-80/Audit/AuditFunctions.cs
- 
					6CDP-LTS-80/CDP-LTS-80.csproj
- 
					0CDP-LTS-80/CDPCore/CDPBlobStorage.cs
- 
					0CDP-LTS-80/CDPCore/CDPDB.cs
- 
					344CDP-LTS-80/CDPCore/CDPLite.cs
- 
					84CDP-LTS-80/CDPCore/FileRevision.cs
- 
					0CDP-LTS-80/CDPCore/MailProcessor.cs
- 
					111CDP-LTS-80/CDPCore/SignalRFunctions.cs
- 
					207CDP-LTS-80/CDPLite.cs
- 
					81CDP-LTS-80/Contacts/ContactFunctions.cs
- 
					327CDP-LTS-80/Contacts/ContactsDB.cs
- 
					0CDP-LTS-80/Contacts/ContactsDocument.cs
- 
					5CDP-LTS-80/EventBased/EventProcessor.cs
- 
					311CDP-LTS-80/EventBased/MetaProcessor.cs
- 
					388CDP-LTS-80/Groups/GroupsDB.cs
- 
					0CDP-LTS-80/Groups/GroupsDocument.cs
- 
					107CDP-LTS-80/Groups/GroupsFunctions.cs
- 
					39CDP-LTS-80/Helpers.cs
- 
					21CDP-LTS-80/host.json
| @ -0,0 +1,263 @@ | |||||
|  | using Microsoft.AspNetCore.Http; | ||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using Newtonsoft.Json; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class AuditFunctions | ||||
|  |     { | ||||
|  |         private readonly ILogger<AuditFunctions> _logger; | ||||
|  |         public static string FileAuditContainer = "FileAudits"; | ||||
|  |         public static string UserAuditContainer = "UserAudits"; | ||||
|  |         public static string GroupAuditContainer = "GroupAudits"; | ||||
|  |         public static string TenantAuditContainer = "TenantAudits"; | ||||
|  |         public AuditFunctions(ILogger<AuditFunctions> logger) | ||||
|  |         { | ||||
|  |             _logger = logger; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetAuditLogForFile")] | ||||
|  |         public async Task<IActionResult> GetAuditLogForFile([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetAuditLogForFile invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             if (string.IsNullOrEmpty(requestBody)) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "The body is empty" }); | ||||
|  | 
 | ||||
|  |             _logger.LogInformation(requestBody); | ||||
|  | 
 | ||||
|  |             GetAuditLogForFileDto dto = JsonConvert.DeserializeObject<GetAuditLogForFileDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             List<AuditRecord> ad = await AuditDB.GetAuditRecordsBetweenDates(dto.FileId, DateTime.MinValue, DateTime.MaxValue, CDPLite.FileAuditContainer); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(ad); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetAuditLogForUser")] | ||||
|  |         public async Task<IActionResult> GetAuditLogForUser([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetAuditLogForUser invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             if (string.IsNullOrEmpty(requestBody)) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "The body is empty" }); | ||||
|  | 
 | ||||
|  |             _logger.LogInformation(requestBody); | ||||
|  | 
 | ||||
|  |             GetAuditLogForUserDto dto = JsonConvert.DeserializeObject<GetAuditLogForUserDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  |             List<AuditRecord> ad = await AuditDB.GetAuditRecordsBetweenDates(userId, DateTime.MinValue, DateTime.MaxValue, UserAuditContainer); | ||||
|  |             return new OkObjectResult(ad); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetAuditLogForGroup")] | ||||
|  |         public async Task<IActionResult> GetAuditLogForGroup([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetAuditLogForGroup invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             if (string.IsNullOrEmpty(requestBody)) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "The body is empty" }); | ||||
|  | 
 | ||||
|  |             _logger.LogInformation(requestBody); | ||||
|  | 
 | ||||
|  |             GetAuditLogForGroupDto dto = JsonConvert.DeserializeObject<GetAuditLogForGroupDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             List<AuditRecord> ad = await AuditDB.GetAuditRecordsBetweenDates(dto.GroupId, DateTime.MinValue, DateTime.MaxValue, GroupAuditContainer); | ||||
|  |             return new OkObjectResult(ad); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetAuditLogForTenant")] | ||||
|  |         public async Task<IActionResult> GetAuditLogForTenant([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetAuditLogForGroup invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             if (string.IsNullOrEmpty(requestBody)) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "The body is empty" }); | ||||
|  | 
 | ||||
|  |             _logger.LogInformation(requestBody); | ||||
|  | 
 | ||||
|  |             GetAuditLogForTenantDto dto = JsonConvert.DeserializeObject<GetAuditLogForTenantDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             List<AuditRecord> ad = await AuditDB.GetAuditRecordsBetweenDates(dto.AppKey, DateTime.MinValue, DateTime.MaxValue, TenantAuditContainer); | ||||
|  |             return new OkObjectResult(ad); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddAccessViolation")] | ||||
|  |         public async Task<Boolean> AddAccessViolation([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("AddAccessViolation invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             if (string.IsNullOrEmpty(requestBody)) | ||||
|  |                 return false; | ||||
|  | 
 | ||||
|  |             _logger.LogInformation(requestBody); | ||||
|  | 
 | ||||
|  |             AddAccessViolationDto dto = JsonConvert.DeserializeObject<AddAccessViolationDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return false; | ||||
|  | 
 | ||||
|  |             string message = string.Format($"Access violation recorded for file {dto.FileName}"); | ||||
|  |             string action = "Access Violation"; | ||||
|  |             await AddAudits(dto.AppKey, dto.FileId, dto.FileName, "", "", action, message); | ||||
|  |             return true; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<string> AddAuditsEvent(string appKey, string fileId, string fileName, string userId, string groupid, string action, string message) | ||||
|  |         { | ||||
|  |             using (var mt = new MethodTimer("AddAuditsEventMessage")) | ||||
|  |             { | ||||
|  |                 if (string.IsNullOrEmpty(appKey) || string.IsNullOrEmpty(fileId) || string.IsNullOrEmpty(action) || string.IsNullOrEmpty(message)) | ||||
|  |                     return ""; | ||||
|  |                 AuditEventMetadata auditEvent = new AuditEventMetadata | ||||
|  |                 { | ||||
|  |                     FileId = fileId, | ||||
|  |                     FileName = fileName, | ||||
|  |                     UserId = userId, | ||||
|  |                     GroupId = groupid, | ||||
|  |                     Action = action, | ||||
|  |                     Message = message | ||||
|  |                 }; | ||||
|  |                 string jobId = Guid.NewGuid().ToString(); | ||||
|  |                 string jobMeta = JsonConvert.SerializeObject(auditEvent); | ||||
|  |                 Job job = new Job { AppKey = appKey, EventType = JobType.AddAudits, Id = jobId, JobMetadata = jobMeta }; | ||||
|  |                 await MetaProcessor.PublishJob(job); | ||||
|  |                 return jobId; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  | 
 | ||||
|  |         public static async Task AddAudits(string appKey, string fileId, string fileName, string userId, string groupid, string action, string message) | ||||
|  |         { | ||||
|  |             if (string.IsNullOrEmpty(appKey) || string.IsNullOrEmpty(fileId) || string.IsNullOrEmpty(action) || string.IsNullOrEmpty(message)) | ||||
|  |             { | ||||
|  |                 Console.WriteLine(string.Format("something weird? appKey, fileId, action, message: {0} {1} {2} {3}", appKey, fileId, action, message)); | ||||
|  |                 return; | ||||
|  |             } | ||||
|  | 
 | ||||
|  | 
 | ||||
|  |             AuditRecord faRec = new FileAuditRecord() | ||||
|  |             { | ||||
|  |                 AppKey = appKey, | ||||
|  |                 FileId = fileId, | ||||
|  |                 FileName = fileName, | ||||
|  |                 UserId = userId, | ||||
|  |                 GroupId = groupid, | ||||
|  |                 Action = action, | ||||
|  |                 Message = message, | ||||
|  |                 EventTime = DateTime.UtcNow, | ||||
|  |             }; | ||||
|  |             Console.WriteLine("Adding File Audit Record"); | ||||
|  |             await AuditDB.AppendRecord(faRec.id, faRec, FileAuditContainer); | ||||
|  | 
 | ||||
|  |             AuditRecord faRecTenant = new TenantAuditRecord() | ||||
|  |             { | ||||
|  |                 AppKey = appKey, | ||||
|  |                 FileId = fileId, | ||||
|  |                 FileName = fileName, | ||||
|  |                 UserId = userId, | ||||
|  |                 GroupId = groupid, | ||||
|  |                 Action = action, | ||||
|  |                 Message = message, | ||||
|  |                 EventTime = DateTime.UtcNow, | ||||
|  |             }; | ||||
|  | 
 | ||||
|  |             await AuditDB.AppendRecord(faRecTenant.id, faRecTenant, TenantAuditContainer); | ||||
|  | 
 | ||||
|  |             if (!string.IsNullOrEmpty(groupid)) | ||||
|  |             { | ||||
|  |                 AuditRecord faRecGroup = new GroupAuditRecord() | ||||
|  |                 { | ||||
|  |                     AppKey = appKey, | ||||
|  |                     FileId = fileId, | ||||
|  |                     FileName = fileName, | ||||
|  |                     UserId = userId, | ||||
|  |                     GroupId = groupid, | ||||
|  |                     Action = action, | ||||
|  |                     Message = message, | ||||
|  |                     EventTime = DateTime.UtcNow, | ||||
|  |                 }; | ||||
|  | 
 | ||||
|  |                 await AuditDB.AppendRecord(faRecGroup.id, faRecGroup, GroupAuditContainer); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             AuditRecord faRecUser = new UserAuditRecord() | ||||
|  |             { | ||||
|  |                 AppKey = appKey, | ||||
|  |                 FileId = fileId, | ||||
|  |                 FileName = fileName, | ||||
|  |                 UserId = userId, | ||||
|  |                 GroupId = groupid, | ||||
|  |                 Action = action, | ||||
|  |                 Message = message, | ||||
|  |                 EventTime = DateTime.UtcNow, | ||||
|  |             }; | ||||
|  | 
 | ||||
|  |             await AuditDB.AppendRecord(faRecUser.id, faRecUser, UserAuditContainer); | ||||
|  | 
 | ||||
|  |         } | ||||
|  |         /// <summary>
 | ||||
|  |         ///  Adds the audit record on a background thread.
 | ||||
|  |         /// </summary>
 | ||||
|  |         private static async Task AddFileAudit(AuditRecord far) | ||||
|  |         { | ||||
|  |             await AuditDB.AppendRecord(far.id, far, FileAuditContainer); | ||||
|  | 
 | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         private static async Task AddUserAudit(AuditRecord far) | ||||
|  |         { | ||||
|  |             await AuditDB.AppendRecord(far.id, far, UserAuditContainer); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         private static async Task AddTenantAudit(AuditRecord far) | ||||
|  |         { | ||||
|  |             await Task.Run(async () => | ||||
|  |             { | ||||
|  |                 try | ||||
|  |                 { | ||||
|  |                     await AuditDB.AppendRecord(far.id, far, TenantAuditContainer); | ||||
|  |                 } | ||||
|  |                 catch (Exception e) | ||||
|  |                 { | ||||
|  |                 } | ||||
|  |             }); | ||||
|  | 
 | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         private static async Task AddGroupAudit(AuditRecord far) | ||||
|  |         { | ||||
|  |             await Task.Run(async () => | ||||
|  |             { | ||||
|  |                 try | ||||
|  |                 { | ||||
|  |                     await AuditDB.AppendRecord(far.id, far, GroupAuditContainer); | ||||
|  |                 } | ||||
|  |                 catch (Exception e) | ||||
|  |                 { | ||||
|  |                 } | ||||
|  |             }); | ||||
|  | 
 | ||||
|  |         } | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,344 @@ | |||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Cosmos; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using Newtonsoft.Json; | ||||
|  | using System; | ||||
|  | using System.Collections.Generic; | ||||
|  | using System.Linq; | ||||
|  | using System.Text; | ||||
|  | using System.Threading.Tasks; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class CDPLite | ||||
|  |     { | ||||
|  |         private readonly ILogger<CDPLite> _logger; | ||||
|  |         public static string FileAuditContainer = "FileAudits"; | ||||
|  |         public static string UserAuditContainer = "UserAudits"; | ||||
|  |         public static string GroupAuditContainer = "GroupAudits"; | ||||
|  |         public static string TenantAuditContainer = "TenantAudits"; | ||||
|  | 
 | ||||
|  |         public CDPLite(ILogger<CDPLite> log) | ||||
|  |         { | ||||
|  |             _logger = log; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         /*internal async Task<IActionResult> AddFilesBatchedInternal(AddFileBatchedDto dto) | ||||
|  |         { | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  |             string jobId = Guid.NewGuid().ToString(); | ||||
|  |             List<KeyVaultEvent> vaultEvents = new List<KeyVaultEvent>(); | ||||
|  |             //List<AuditEventMetadata> auditEvents = new List<AuditEventMetadata>();
 | ||||
|  |             List<FileRecord> fileRecords = new List<FileRecord>(); | ||||
|  |             for (int i = 0; i < dto.Count; i++) | ||||
|  |             { | ||||
|  |                 string fileId = Guid.NewGuid().ToString(); | ||||
|  |                 string fileName = dto.FileNames[i]; | ||||
|  |                 string aesKey = Helpers.GenerateAES256KeyToBase64(); | ||||
|  |                 //string message = string.Format($"{dto.Email} protected {fileName} file having {fileId} id.");
 | ||||
|  |                 KeyVaultEvent vaultEvent = new KeyVaultEvent { AESKey = aesKey, FileId = fileId }; | ||||
|  |                 //AuditEventMetadata auditEvent = new AuditEventMetadata { Action = "Addded", FileId = fileId, FileName = dto.FileNames[i], Message = message, UserId = userId };
 | ||||
|  |                 //auditEvents.Add(auditEvent);
 | ||||
|  |                 vaultEvents.Add(vaultEvent); | ||||
|  | 
 | ||||
|  |                 AccessPolicy ac = new AccessPolicy() | ||||
|  |                 { | ||||
|  |                     Access = "Owner", | ||||
|  |                     Key = aesKey, | ||||
|  |                     Email = dto.Email | ||||
|  |                 }; | ||||
|  |                 FileRecord fr = await CDPDB.UpsertFile(dto.AppKey, fileId, fileName, userId, "", ac); | ||||
|  |                 fileRecords.Add(fr); | ||||
|  |             } | ||||
|  |             List<Job> jobs = new List<Job>(); | ||||
|  |             Job vaultJob = await AddKeyVaultBatchedEvent(vaultEvents, dto.AppKey); | ||||
|  |             //Job auditJob = await AddAuditsBatchedEvent(auditEvents, dto.AppKey);
 | ||||
|  |             //jobs.Add(auditJob);
 | ||||
|  |             jobs.Add(vaultJob); | ||||
|  |             await MetaProcessor.PublishBatchJobs(jobs); | ||||
|  |             return new OkObjectResult(fileRecords); | ||||
|  | 
 | ||||
|  |         }*/ | ||||
|  | 
 | ||||
|  | 
 | ||||
|  |         internal async Task<IActionResult> AddFileInternal(AddFileDto dto, bool useKeyVaultEvent = false) | ||||
|  |         { | ||||
|  |             string fileId = Guid.NewGuid().ToString(); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  |             string fileName = dto.FileName; | ||||
|  |             string aesKey = Helpers.GenerateAES256KeyToBase64(); | ||||
|  | 
 | ||||
|  |             // the KeyVault is slow for some reason and while it's dangerous to return a key
 | ||||
|  |             // before we're sure it got added to the database...I'm going to do it anyway.
 | ||||
|  |             if (useKeyVaultEvent) | ||||
|  |             { | ||||
|  |                 await AddKeyVaultEvent(fileId, aesKey, dto.AppKey); | ||||
|  |             } | ||||
|  |             else | ||||
|  |             { | ||||
|  |                 await Task.Run(async () => | ||||
|  |                 { | ||||
|  |                     try | ||||
|  |                     { | ||||
|  |                         KeyVaultService kvs = new KeyVaultService(Constants.KeyVaultURI); | ||||
|  |                         await kvs.SetSecretAsync(fileId, aesKey); | ||||
|  |                     } | ||||
|  |                     catch (Exception e) | ||||
|  |                     { | ||||
|  |                         Console.WriteLine(e); | ||||
|  |                     } | ||||
|  |                 }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             // when you add a file, you have rights to manage access to it.
 | ||||
|  |             AccessPolicy ac = new AccessPolicy() | ||||
|  |             { | ||||
|  |                 Access = "Owner", | ||||
|  |                 Key = aesKey, | ||||
|  |                 Email = dto.Email | ||||
|  |             }; | ||||
|  | 
 | ||||
|  |             // since we're generating a new file id, a new entry will always be created.
 | ||||
|  |             FileRecord fr = await CDPDB.UpsertFile(dto.AppKey, fileId, fileName, userId, "", ac); | ||||
|  | 
 | ||||
|  |             string message = string.Format($"{dto.Email} protected {fileName} file having {fileId} id."); | ||||
|  |             string action = "Added"; | ||||
|  | 
 | ||||
|  |             await AuditFunctions.AddAuditsEvent(dto.AppKey, fileId, fileName, userId, dto.GroupId, action, message); //commenting for speed test
 | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         internal static async Task<IActionResult> AddFileUserInternal(AddFileUserDto dto) | ||||
|  |         { | ||||
|  |             // check to see if the email has the power to add a user
 | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  | 
 | ||||
|  |             FileRecord fr = await CDPDB.GetFile(dto.AppKey, dto.FileId, userId); | ||||
|  |             if (fr == null) | ||||
|  |             { | ||||
|  |                 string message = string.Format($"{dto.Email} attempted to add/change access policy for {dto.EmailToAdd} on {dto.FileName} file having {dto.FileId} id, but didn't have ANY access"); | ||||
|  |                 Console.WriteLine(message); | ||||
|  |                 string action = "Policy change failed"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, dto.FileId, dto.FileName, userId, "", action, message); | ||||
|  | 
 | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "File not found for user " + dto.Email }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if ((!fr.Policy.CheckAccess("Manage")) && (!fr.Policy.CheckAccess("Owner"))) | ||||
|  |             { | ||||
|  |                 string message = string.Format($"{dto.Email} attempted to add/change access policy for {dto.EmailToAdd} on {dto.FileName} file having {dto.FileId} id, but didn't have manage access"); | ||||
|  |                 Console.WriteLine(message); | ||||
|  |                 string action = "Policy change failed"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, dto.FileId, dto.FileName, userId, "", action, message); | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = $"{dto.Email} doesn't have the rights to add a user." }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             string fileId = dto.FileId; | ||||
|  |             string fileName = dto.FileName; | ||||
|  |             string userIdToAdd = ""; | ||||
|  | 
 | ||||
|  |             if (dto.EmailToAdd != "") | ||||
|  |             { | ||||
|  |                 userIdToAdd = Helpers.HashAndShortenText(dto.EmailToAdd.ToLower()); | ||||
|  |             } | ||||
|  |             else if (dto.Group != null) | ||||
|  |             { | ||||
|  |                 userIdToAdd = dto.GroupId; | ||||
|  |             } | ||||
|  |             else if (dto.Group != null) | ||||
|  |             { | ||||
|  |                 userIdToAdd = dto.GroupId; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             AccessPolicy ac = new AccessPolicy() | ||||
|  |             { | ||||
|  |                 Access = dto.Policy, | ||||
|  |                 Email = dto.EmailToAdd.ToLower(), | ||||
|  |                 Group = dto.Group, | ||||
|  |                 GroupId = dto.GroupId, | ||||
|  |                 Key = "" | ||||
|  |             }; | ||||
|  | 
 | ||||
|  |             fr = await CDPDB.UpsertFile(dto.AppKey, fileId, fileName, userIdToAdd, "", ac); | ||||
|  | 
 | ||||
|  |             if (dto.EmailToAdd != "") | ||||
|  |             { | ||||
|  |                 string message = string.Format($"{dto.Email} added/changed the access policy for User : {dto.EmailToAdd} to {dto.Policy} on {fileName} file having {fileId} id"); | ||||
|  |                 string action = "Policy change"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, fileId, fileName, userId, "", action, message); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (dto.Group != null) | ||||
|  |             { | ||||
|  |                 string message = string.Format($"{dto.Email} added/changed the access policy for Group : {dto.Group} to {dto.Policy} on {fileName} file having {fileId} id"); | ||||
|  |                 string action = "Policy change"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, fileId, fileName, "", dto.Group.id, action, message); | ||||
|  |             } | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         #region CDP File Functions
 | ||||
|  | 
 | ||||
|  |         [Function("AddFile")] | ||||
|  |         public async Task<IActionResult> AddFile([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("AddFile invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             AddFileDto dto = JsonConvert.DeserializeObject<AddFileDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             return await AddFileInternal(dto, true); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddFileUser")] | ||||
|  |         public async Task<IActionResult> AddFileUser([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("AddFileUser invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             AddFileUserDto dto = JsonConvert.DeserializeObject<AddFileUserDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             return await AddFileUserInternal(dto); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetFileForUser")] | ||||
|  |         public async Task<IActionResult> GetFileForUser([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetFile invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             AddFileUserDto dto = JsonConvert.DeserializeObject<AddFileUserDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             // check to see if the email has the power to add a user
 | ||||
|  |             // string a = Helpers.HashAndShortenText(dto.Email.ToLower());
 | ||||
|  |             // string b = Helpers.HashToHex(dto.Email.ToLower());
 | ||||
|  |             // string c = Helpers.ConvertShortHashToHex(a);
 | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  |             // userId = "user-" + Helpers.HashToHex(dto.Email.ToLower());
 | ||||
|  | 
 | ||||
|  |             FileRecord fr = await CDPDB.GetFile(dto.AppKey, dto.FileId, userId); | ||||
|  |             if (fr == null) | ||||
|  |             { | ||||
|  |                 string AuditMessage = string.Format($"File not found for user {dto.Email} having fileId {dto.FileId}"); | ||||
|  |                 string AuditAction = "Decrypt failed"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, dto.FileId, dto.FileName, userId, "", AuditAction, AuditMessage); | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "File not found for user " + dto.Email }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (fr.Policy.CheckAccess("None")) | ||||
|  |             { | ||||
|  |                 string AuditMessage = string.Format($"{dto.Email} don't have the rights to decrypt {fr.FileName} file having {dto.FileId} id"); | ||||
|  |                 string AuditAction = "Decrypt failed"; | ||||
|  |                 await AuditFunctions.AddAuditsEvent(dto.AppKey, dto.FileId, fr.FileName, userId, "", AuditAction, AuditMessage); | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Access is denied for user " + dto.Email }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             KeyVaultService kvs = new KeyVaultService(Constants.KeyVaultURI); | ||||
|  |             fr.Policy.Key = await kvs.GetSecretAsync(fr.FileId); | ||||
|  | 
 | ||||
|  |             string message = string.Format($"{dto.Email} decrypted {fr.FileName} file having {dto.FileId} id"); | ||||
|  |             string action = "Decrypted"; | ||||
|  |             await AuditFunctions.AddAuditsEvent(dto.AppKey, dto.FileId, dto.FileName, userId, "", action, message); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetPoliciesForFile")] | ||||
|  |         public async Task<IActionResult> GetPoliciesForFile([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("GetFile invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             GetPoliciesForFileDto dto = JsonConvert.DeserializeObject<GetPoliciesForFileDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             List<FileRecord> fr = await CDPDB.GetPoliciesForFile(dto.AppKey, dto.FileId); | ||||
|  |             if (fr == null) | ||||
|  |             { | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "File not found " + dto.FileId }); | ||||
|  |             } | ||||
|  |             KeyVaultService kvs = new KeyVaultService(Constants.KeyVaultURI); | ||||
|  |             string aesKey = await kvs.GetSecretAsync(dto.FileId); | ||||
|  |             foreach (var f in fr) | ||||
|  |             { | ||||
|  |                 f.Policy.Key = aesKey; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("DeleteRegisteredUserPolicies")] | ||||
|  |         public async Task<IActionResult> DeleteRegisteredUserPolicies([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Deleting Registered User invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             DeleteRegisteredUserDto deleteContactdto = JsonConvert.DeserializeObject<DeleteRegisteredUserDto>(requestBody); | ||||
|  |             if (deleteContactdto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(deleteContactdto.UserEmail.ToLower()); | ||||
|  |             await CDPDB.revokeRegisteredUserPolicies(deleteContactdto.AppKey, deleteContactdto.UserEmail.ToLower(), deleteContactdto.ContactEmail.ToLower(), deleteContactdto.AdminEmail.ToLower()); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(true); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         #endregion
 | ||||
|  | 
 | ||||
|  |         #region Background Processing Support
 | ||||
|  | 
 | ||||
|  |         public static async Task<Job> AddAuditsBatchedEvent(List<AuditEventMetadata> auditEvents, string appKey) | ||||
|  |         { | ||||
|  |             using (var mt = new MethodTimer("AddAuditsBatchedEvent")) | ||||
|  |             { | ||||
|  |                 string jobMeta = JsonConvert.SerializeObject(auditEvents); | ||||
|  |                 string jobId = Guid.NewGuid().ToString(); | ||||
|  |                 Job job = new Job { AppKey = appKey, EventType = JobType.AddAuditsBatch, Id = jobId, JobMetadata = jobMeta }; | ||||
|  |                 return job; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<Job> AddKeyVaultBatchedEvent(List<KeyVaultEvent> vaultEvents, string appKey) | ||||
|  |         { | ||||
|  |             using (var mt = new MethodTimer("AddKeyVaultBatchedEvent")) | ||||
|  |             { | ||||
|  |                 string jobMeta = JsonConvert.SerializeObject(vaultEvents); | ||||
|  |                 string jobId = Guid.NewGuid().ToString(); | ||||
|  |                 Job job = new Job { AppKey = appKey, EventType = JobType.KeyVaultInsertionBatch, Id = jobId, JobMetadata = jobMeta }; | ||||
|  |                 //await MetaProcessor.PublishJob(job);
 | ||||
|  |                 return job; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<string> AddKeyVaultEvent(string fileId, string aesKey, string appKey) | ||||
|  |         { | ||||
|  |             using (var mt = new MethodTimer("AddKeyVaultEvent")) | ||||
|  |             { | ||||
|  |                 KeyVaultEvent vaultEvent = new KeyVaultEvent { AESKey = aesKey, FileId = fileId }; | ||||
|  |                 string jobMeta = JsonConvert.SerializeObject(vaultEvent); | ||||
|  |                 string jobId = Guid.NewGuid().ToString(); | ||||
|  |                 Job keyVaultJob = new Job { AppKey = appKey, EventType = JobType.KeyVaultInsertion, Id = jobId, JobMetadata = jobMeta }; | ||||
|  |                 await MetaProcessor.PublishJob(keyVaultJob); | ||||
|  |                 return jobId; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         #endregion
 | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,111 @@ | |||||
|  | using Microsoft.AspNetCore.Http; | ||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using Newtonsoft.Json; | ||||
|  | using System.Net; | ||||
|  | 
 | ||||
|  | namespace CDP.CDPCore | ||||
|  | { | ||||
|  |     /*public class SignalRFunctions | ||||
|  |     { | ||||
|  |         private readonly ILogger<SignalRFunctions> _logger; | ||||
|  | 
 | ||||
|  |         private static readonly HttpClient HttpClient = new(); | ||||
|  |         private static string Etag = string.Empty; | ||||
|  |         private static int StarCount = 0; | ||||
|  | 
 | ||||
|  |         public SignalRFunctions(ILogger<SignalRFunctions> log) | ||||
|  |         { | ||||
|  |             _logger = log; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("index")] | ||||
|  |         public static HttpResponseData GetHomePage([HttpTrigger(AuthorizationLevel.Anonymous)] HttpRequestData req) | ||||
|  |         { | ||||
|  |             var response = req.CreateResponse(HttpStatusCode.OK); | ||||
|  |             response.WriteString(File.ReadAllText("content/index.html")); | ||||
|  |             response.Headers.Add("Content-Type", "text/html"); | ||||
|  |             return response; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("negotiate")] | ||||
|  |         public static HttpResponseData Negotiate([HttpTrigger(AuthorizationLevel.Anonymous)] HttpRequestData req, | ||||
|  |             [SignalRConnectionInfoInput(HubName = "cdp")] string connectionInfo) | ||||
|  |         { | ||||
|  |             var response = req.CreateResponse(HttpStatusCode.OK); | ||||
|  |             response.Headers.Add("Content-Type", "application/json"); | ||||
|  |             response.WriteString(connectionInfo); | ||||
|  |             return response; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function(nameof(SendMessage))] | ||||
|  |         [SignalROutput(HubName = "cdp")] | ||||
|  |         public static async Task<SignalRMessageAction> SendMessage([HttpTrigger(AuthorizationLevel.Anonymous, "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             string requestBody = new StreamReader(req.Body).ReadToEnd(); | ||||
|  |             SendMessagePayload pl = JsonConvert.DeserializeObject<SendMessagePayload>(requestBody); | ||||
|  |             if (pl == null) | ||||
|  |                 return null; | ||||
|  | 
 | ||||
|  |             return await SendMessageInternal(pl); | ||||
|  | 
 | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<SignalRMessageAction> SendMessageInternal(SendMessagePayload pl) | ||||
|  |         { | ||||
|  |             if (pl == null) | ||||
|  |                 return null; | ||||
|  | 
 | ||||
|  |             if (string.IsNullOrEmpty(pl.ConnectionId) && string.IsNullOrEmpty(pl.UserId)) | ||||
|  |                 return null; | ||||
|  | 
 | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 Console.WriteLine($"Sending message to {pl.ConnectionId}"); | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 Console.WriteLine(ex.Message); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return new SignalRMessageAction(pl.Target) | ||||
|  |             { | ||||
|  |                 Arguments = new[] { pl.Message }, | ||||
|  |                 ConnectionId = pl.ConnectionId.Trim() | ||||
|  |             }; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         //[Function("broadcast")]
 | ||||
|  |         //[SignalROutput(HubName = "cdp")]
 | ||||
|  |         //public static async Task<SignalRMessageAction> Broadcast([TimerTrigger("5 * * * * *] TimerInfo timerInfo)
 | ||||
|  |         //{
 | ||||
|  |         //    var request = new HttpRequestMessage(HttpMethod.Get, "https://api.github.com/repos/azure/azure-signalr");
 | ||||
|  |         //    request.Headers.UserAgent.ParseAdd("Serverless");
 | ||||
|  |         //    request.Headers.Add("If-None-Match", Etag);
 | ||||
|  |         //    var response = await HttpClient.SendAsync(request);
 | ||||
|  |         //    if (response.Headers.Contains("Etag"))
 | ||||
|  |         //    {
 | ||||
|  |         //        Etag = response.Headers.GetValues("Etag").First();
 | ||||
|  |         //    }
 | ||||
|  |         //    if (response.StatusCode == HttpStatusCode.OK)
 | ||||
|  |         //    {
 | ||||
|  |         //        var result = await response.Content.ReadFromJsonAsync<GitResult>();
 | ||||
|  |         //        if (result != null)
 | ||||
|  |         //        {
 | ||||
|  |         //            StarCount = result.StarCount;
 | ||||
|  |         //        }
 | ||||
|  |         //    }
 | ||||
|  |         //    return new SignalRMessageAction("newMessage", new object[] { $"Current star count of https://github.com/Azure/azure-signalr is: {StarCount}" });
 | ||||
|  |         //}
 | ||||
|  |     }*/ | ||||
|  | 
 | ||||
|  |     public class SendMessagePayload | ||||
|  |     { | ||||
|  |         public string ConnectionId { get; set; } | ||||
|  |         public string UserId { get; set; } | ||||
|  |         public string Message { get; set; } | ||||
|  |         public string Target { get; set; }  // this will default to 'newMessage'
 | ||||
|  |     } | ||||
|  | } | ||||
| @ -1,207 +0,0 @@ | |||||
| using Microsoft.AspNetCore.Mvc; |  | ||||
| using Microsoft.Azure.Cosmos; |  | ||||
| using Microsoft.Extensions.Logging; |  | ||||
| using System; |  | ||||
| using System.Collections.Generic; |  | ||||
| using System.Linq; |  | ||||
| using System.Text; |  | ||||
| using System.Threading.Tasks; |  | ||||
| 
 |  | ||||
| namespace CDP |  | ||||
| { |  | ||||
|     public class CDPLite |  | ||||
|     { |  | ||||
|         private readonly ILogger<CDPLite> _logger; |  | ||||
|         private static string FileAuditContainer = "FileAudits"; |  | ||||
|         private static string UserAuditContainer = "UserAudits"; |  | ||||
|         private static string GroupAuditContainer = "GroupAudits"; |  | ||||
|         private static string TenantAuditContainer = "TenantAudits"; |  | ||||
| 
 |  | ||||
|         public CDPLite(ILogger<CDPLite> log) |  | ||||
|         { |  | ||||
|             _logger = log; |  | ||||
|         } |  | ||||
| 
 |  | ||||
|         internal static async Task<IActionResult> AddFileUserInternal(AddFileUserDto dto) |  | ||||
|         { |  | ||||
|             // check to see if the email has the power to add a user
 |  | ||||
|             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); |  | ||||
| 
 |  | ||||
|             FileRecord fr = await CDPDB.GetFile(dto.AppKey, dto.FileId, userId); |  | ||||
|             if (fr == null) |  | ||||
|             { |  | ||||
|                 string message = string.Format($"{dto.Email} attempted to add/change access policy for {dto.EmailToAdd} on {dto.FileName} file having {dto.FileId} id, but didn't have ANY access"); |  | ||||
|                 Console.WriteLine(message); |  | ||||
|                 string action = "Policy change failed"; |  | ||||
|                 await AddAudits(dto.AppKey, dto.FileId, dto.FileName, userId, "", action, message); |  | ||||
| 
 |  | ||||
|                 return new BadRequestObjectResult(new { error = true, message = "File not found for user " + dto.Email }); |  | ||||
|             } |  | ||||
| 
 |  | ||||
|             if ((!fr.Policy.CheckAccess("Manage")) && (!fr.Policy.CheckAccess("Owner"))) |  | ||||
|             { |  | ||||
|                 string message = string.Format($"{dto.Email} attempted to add/change access policy for {dto.EmailToAdd} on {dto.FileName} file having {dto.FileId} id, but didn't have manage access"); |  | ||||
|                 Console.WriteLine(message); |  | ||||
|                 string action = "Policy change failed"; |  | ||||
|                 await AddAudits(dto.AppKey, dto.FileId, dto.FileName, userId, "", action, message); |  | ||||
|                 return new BadRequestObjectResult(new { error = true, message = $"{dto.Email} doesn't have the rights to add a user." }); |  | ||||
|             } |  | ||||
| 
 |  | ||||
|             string fileId = dto.FileId; |  | ||||
|             string fileName = dto.FileName; |  | ||||
|             string userIdToAdd = ""; |  | ||||
| 
 |  | ||||
|             if (dto.EmailToAdd != "") |  | ||||
|             { |  | ||||
|                 userIdToAdd = Helpers.HashAndShortenText(dto.EmailToAdd.ToLower()); |  | ||||
|             } |  | ||||
|             else if (dto.Group != null) |  | ||||
|             { |  | ||||
|                 userIdToAdd = dto.GroupId; |  | ||||
|             } |  | ||||
|             else if (dto.Group != null) |  | ||||
|             { |  | ||||
|                 userIdToAdd = dto.GroupId; |  | ||||
|             } |  | ||||
| 
 |  | ||||
|             AccessPolicy ac = new AccessPolicy() |  | ||||
|             { |  | ||||
|                 Access = dto.Policy, |  | ||||
|                 Email = dto.EmailToAdd.ToLower(), |  | ||||
|                 Group = dto.Group, |  | ||||
|                 GroupId = dto.GroupId, |  | ||||
|                 Key = "" |  | ||||
|             }; |  | ||||
| 
 |  | ||||
|             fr = await CDPDB.UpsertFile(dto.AppKey, fileId, fileName, userIdToAdd, "", ac); |  | ||||
| 
 |  | ||||
|             if (dto.EmailToAdd != "") |  | ||||
|             { |  | ||||
|                 string message = string.Format($"{dto.Email} added/changed the access policy for User : {dto.EmailToAdd} to {dto.Policy} on {fileName} file having {fileId} id"); |  | ||||
|                 string action = "Policy change"; |  | ||||
|                 await AddAudits(dto.AppKey, fileId, fileName, userId, "", action, message); |  | ||||
|             } |  | ||||
| 
 |  | ||||
|             if (dto.Group != null) |  | ||||
|             { |  | ||||
|                 string message = string.Format($"{dto.Email} added/changed the access policy for Group : {dto.Group} to {dto.Policy} on {fileName} file having {fileId} id"); |  | ||||
|                 string action = "Policy change"; |  | ||||
|                 await AddAudits(dto.AppKey, fileId, fileName, "", dto.Group.id, action, message); |  | ||||
|             } |  | ||||
|             return new OkObjectResult(fr); |  | ||||
|         } |  | ||||
| 
 |  | ||||
|         public static async Task AddAudits(string appKey, string fileId, string fileName, string userId, string groupid, string action, string message) |  | ||||
|         { |  | ||||
|             if (string.IsNullOrEmpty(appKey) || string.IsNullOrEmpty(fileId) || string.IsNullOrEmpty(action) || string.IsNullOrEmpty(message)) |  | ||||
|             { |  | ||||
|                 Console.WriteLine(string.Format("something weird? appKey, fileId, action, message: {0} {1} {2} {3}", appKey, fileId, action, message)); |  | ||||
|                 return; |  | ||||
|             } |  | ||||
|                  |  | ||||
| 
 |  | ||||
|             AuditRecord faRec = new FileAuditRecord() |  | ||||
|             { |  | ||||
|                 AppKey = appKey, |  | ||||
|                 FileId = fileId, |  | ||||
|                 FileName = fileName, |  | ||||
|                 UserId = userId, |  | ||||
|                 GroupId = groupid, |  | ||||
|                 Action = action, |  | ||||
|                 Message = message, |  | ||||
|                 EventTime = DateTime.UtcNow, |  | ||||
|             }; |  | ||||
|             Console.WriteLine("Adding File Audit Record"); |  | ||||
|             await AuditDB.AppendRecord(faRec.id, faRec, FileAuditContainer); |  | ||||
| 
 |  | ||||
|             AuditRecord faRecTenant = new TenantAuditRecord() |  | ||||
|             { |  | ||||
|                 AppKey = appKey, |  | ||||
|                 FileId = fileId, |  | ||||
|                 FileName = fileName, |  | ||||
|                 UserId = userId, |  | ||||
|                 GroupId = groupid, |  | ||||
|                 Action = action, |  | ||||
|                 Message = message, |  | ||||
|                 EventTime = DateTime.UtcNow, |  | ||||
|             }; |  | ||||
| 
 |  | ||||
|             await AuditDB.AppendRecord(faRecTenant.id, faRecTenant, TenantAuditContainer); |  | ||||
| 
 |  | ||||
|             if (!string.IsNullOrEmpty(groupid)) |  | ||||
|             { |  | ||||
|                 AuditRecord faRecGroup = new GroupAuditRecord() |  | ||||
|                 { |  | ||||
|                     AppKey = appKey, |  | ||||
|                     FileId = fileId, |  | ||||
|                     FileName = fileName, |  | ||||
|                     UserId = userId, |  | ||||
|                     GroupId = groupid, |  | ||||
|                     Action = action, |  | ||||
|                     Message = message, |  | ||||
|                     EventTime = DateTime.UtcNow, |  | ||||
|                 }; |  | ||||
| 
 |  | ||||
|                 await AuditDB.AppendRecord(faRecGroup.id, faRecGroup, GroupAuditContainer); |  | ||||
|             } |  | ||||
| 
 |  | ||||
|             AuditRecord faRecUser = new UserAuditRecord() |  | ||||
|             { |  | ||||
|                 AppKey = appKey, |  | ||||
|                 FileId = fileId, |  | ||||
|                 FileName = fileName, |  | ||||
|                 UserId = userId, |  | ||||
|                 GroupId = groupid, |  | ||||
|                 Action = action, |  | ||||
|                 Message = message, |  | ||||
|                 EventTime = DateTime.UtcNow, |  | ||||
|             }; |  | ||||
| 
 |  | ||||
|             await AuditDB.AppendRecord(faRecUser.id, faRecUser, UserAuditContainer); |  | ||||
| 
 |  | ||||
|         } |  | ||||
|         /// <summary>
 |  | ||||
|         ///  Adds the audit record on a background thread.
 |  | ||||
|         /// </summary>
 |  | ||||
|         private static async Task AddFileAudit(AuditRecord far) |  | ||||
|         { |  | ||||
|             await AuditDB.AppendRecord(far.id, far, FileAuditContainer); |  | ||||
| 
 |  | ||||
|         } |  | ||||
| 
 |  | ||||
|         private static async Task AddUserAudit(AuditRecord far) |  | ||||
|         { |  | ||||
|             await AuditDB.AppendRecord(far.id, far, UserAuditContainer); |  | ||||
|         } |  | ||||
| 
 |  | ||||
|         private static async Task AddTenantAudit(AuditRecord far) |  | ||||
|         { |  | ||||
|             await Task.Run(async () => |  | ||||
|             { |  | ||||
|                 try |  | ||||
|                 { |  | ||||
|                     await AuditDB.AppendRecord(far.id, far, TenantAuditContainer); |  | ||||
|                 } |  | ||||
|                 catch (Exception e) |  | ||||
|                 { |  | ||||
|                 } |  | ||||
|             }); |  | ||||
| 
 |  | ||||
|         } |  | ||||
| 
 |  | ||||
|         private static async Task AddGroupAudit(AuditRecord far) |  | ||||
|         { |  | ||||
|             await Task.Run(async () => |  | ||||
|             { |  | ||||
|                 try |  | ||||
|                 { |  | ||||
|                     await AuditDB.AppendRecord(far.id, far, GroupAuditContainer); |  | ||||
|                 } |  | ||||
|                 catch (Exception e) |  | ||||
|                 { |  | ||||
|                 } |  | ||||
|             }); |  | ||||
| 
 |  | ||||
|         } |  | ||||
|     } |  | ||||
| } |  | ||||
| @ -0,0 +1,81 @@ | |||||
|  | using System; | ||||
|  | using System.Threading.Tasks; | ||||
|  | using Azure.Messaging.ServiceBus; | ||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using Newtonsoft.Json; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class ContactFunctions | ||||
|  |     { | ||||
|  |         private readonly ILogger<ContactFunctions> _logger; | ||||
|  | 
 | ||||
|  |         public ContactFunctions(ILogger<ContactFunctions> logger) | ||||
|  |         { | ||||
|  |             _logger = logger; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetContacts")] | ||||
|  |         public async Task<IActionResult> GetContacts([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Get Contacts invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             GetContactsDto dto = JsonConvert.DeserializeObject<GetContactsDto>(requestBody); | ||||
|  |             if (dto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(dto.Email.ToLower()); | ||||
|  |             // userId = "user-" + Helpers.HashToHex(dto.Email.ToLower());
 | ||||
|  | 
 | ||||
|  |             List<ContactRecord> fr = await ContactsDB.GetUserContacts(dto.AppKey, userId); | ||||
|  |             if (fr == null) | ||||
|  |             { | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "File not found " + dto.Email }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddContact")] | ||||
|  |         public async Task<IActionResult> AddContacts([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Add Contacts invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             AddUserContactsDto addContactdto = JsonConvert.DeserializeObject<AddUserContactsDto>(requestBody); | ||||
|  |             if (addContactdto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(addContactdto.EmailId.ToLower()); | ||||
|  |             ContactRecord cr = new ContactRecord(addContactdto.ContactName, addContactdto.ContactEmail, addContactdto.ContactPhone, addContactdto.ContactAddress, addContactdto.ContactCompany); | ||||
|  | 
 | ||||
|  |             Boolean resp = await ContactsDB.AppendRecord(addContactdto.AppKey, userId, cr); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(resp); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("DeleteContact")] | ||||
|  |         public async Task<IActionResult> DeleteContact([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Deleting Contact invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             DeleteUserContactsDto deleteContactdto = JsonConvert.DeserializeObject<DeleteUserContactsDto>(requestBody); | ||||
|  |             if (deleteContactdto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(deleteContactdto.UserEmail.ToLower()); | ||||
|  |             Boolean resp = await ContactsDB.RemoveRecord(deleteContactdto.AppKey, userId, deleteContactdto.ContactEmail.ToLower()); | ||||
|  |             await CDPDB.revokePolicies(deleteContactdto.AppKey, deleteContactdto.UserEmail.ToLower(), deleteContactdto.ContactEmail.ToLower()); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(resp); | ||||
|  |         } | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,327 @@ | |||||
|  | using Microsoft.Azure.Cosmos; | ||||
|  | using System; | ||||
|  | using System.Collections.Generic; | ||||
|  | using System.Linq; | ||||
|  | using System.Text; | ||||
|  | using System.Threading.Tasks; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class ContactsDB | ||||
|  |     { | ||||
|  |         private static Lazy<CosmosClient> lazyClient = new Lazy<CosmosClient>(InitializeCosmosClient); | ||||
|  |         private static CosmosClient cosmosClient => lazyClient.Value; | ||||
|  |         private static string DatabaseName = "CDP"; | ||||
|  |         private static string ContainerName = "Contacts"; | ||||
|  | 
 | ||||
|  |         private static CosmosClient InitializeCosmosClient() | ||||
|  |         { | ||||
|  |             // Perform any initialization here
 | ||||
|  |             var uri = "https://cdplite.documents.azure.com:443/"; | ||||
|  |             var authKey = "VPbg8RpzyI3XwhC2o0dIUtYFs33ghxORCqZeNAyg8vg4HWUBjM41BUxP0qLFXEvFh6ewQY1uKv52ACDbsEN1AQ=="; | ||||
|  | 
 | ||||
|  |             return new CosmosClient(uri, authKey); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<List<ContactRecord>> GetUserContacts(string AppKey, string UserId) | ||||
|  |         { | ||||
|  |             var results = new List<ContactRecord>(); | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             // Fetch the metadata document for the customer ID
 | ||||
|  |             var metadataDocumentId = GetMetaDocumentKey(AppKey, UserId); | ||||
|  |             MetadataDocumentContact metadataDocument = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocumentResponse = | ||||
|  |                 await container.ReadItemAsync<MetadataDocumentContact>(metadataDocumentId, new PartitionKey(metadataDocumentId)); | ||||
|  |                 metadataDocument = metadataDocumentResponse.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (metadataDocument == null) | ||||
|  |                 return results; | ||||
|  | 
 | ||||
|  |             // Determine the partition keys within the date range
 | ||||
|  |             var partitionKeysInDocument = metadataDocument.PartitionKeys; | ||||
|  |             // Fetch the audit records for each partition key within the date range
 | ||||
|  |             foreach (var partitionKey in partitionKeysInDocument) | ||||
|  |             { | ||||
|  | 
 | ||||
|  |                 ItemResponse<ContactsDocument> response = await container.ReadItemAsync<ContactsDocument>(partitionKey, new PartitionKey(partitionKey)); | ||||
|  |                 if (response == null) | ||||
|  |                     continue; | ||||
|  | 
 | ||||
|  |                 ContactsDocument t = response.Resource; | ||||
|  |                 results.AddRange(t.Records); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return results; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<bool> AppendRecord(string appKey, string userId, ContactRecord rec) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocument = await GetMetadataDocument(appKey, userId); | ||||
|  |                 if (metadataDocument == null) | ||||
|  |                     return false; | ||||
|  | 
 | ||||
|  |                 string dayKey = metadataDocument.GetLatestKey(appKey, userId); | ||||
|  |                 ContactsDocument al = await GetContactDocument(dayKey); | ||||
|  |                 if (al == null) | ||||
|  |                 { | ||||
|  |                     al = new ContactsDocument(); | ||||
|  |                     al.AppKey = appKey; | ||||
|  |                     al.UserId = userId; | ||||
|  |                 } | ||||
|  |                 al.Records.Add(rec); | ||||
|  |                 await UpdateContactsDocument(al); | ||||
|  |                 return true; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 return false; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<bool> RemoveRecord(string appKey, string userId, string EmailId) | ||||
|  |         { | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             // Fetch the metadata document for the customer ID
 | ||||
|  |             var metadataDocumentId = GetMetaDocumentKey(appKey, userId); | ||||
|  |             MetadataDocumentContact metadataDocument = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocumentResponse = await container.ReadItemAsync<MetadataDocumentContact>(metadataDocumentId, new PartitionKey(metadataDocumentId)); | ||||
|  |                 metadataDocument = metadataDocumentResponse.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (metadataDocument == null) | ||||
|  |                 return false; | ||||
|  | 
 | ||||
|  |             // Determine the partition keys within the date range
 | ||||
|  |             var partitionKeysInDocument = metadataDocument.PartitionKeys; | ||||
|  |             // Fetch the audit records for each partition key within the date range
 | ||||
|  |             foreach (var partitionKey in partitionKeysInDocument) | ||||
|  |             { | ||||
|  | 
 | ||||
|  |                 ItemResponse<ContactsDocument> response = await container.ReadItemAsync<ContactsDocument>(partitionKey, new PartitionKey(partitionKey)); | ||||
|  |                 if (response == null) | ||||
|  |                     continue; | ||||
|  | 
 | ||||
|  |                 ContactsDocument t = response.Resource; | ||||
|  |                 var originalCount = t.Records.Count; | ||||
|  |                 t.Records = t.Records.Where(item => item.Email != EmailId).ToList(); | ||||
|  |                 var afterFilterCount = t.Records.Count; | ||||
|  | 
 | ||||
|  |                 if (originalCount != afterFilterCount) | ||||
|  |                 { | ||||
|  |                     ItemResponse<ContactsDocument> updateResponse = await container.ReplaceItemAsync( | ||||
|  |                         item: t, | ||||
|  |                         id: t.id, | ||||
|  |                         partitionKey: new PartitionKey(partitionKey)); | ||||
|  |                     return true; | ||||
|  |                 } | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return false; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task UpdateContactsDocument(ContactsDocument al) | ||||
|  |         { | ||||
|  |             if (al.Records.Count == 0) | ||||
|  |                 return; | ||||
|  | 
 | ||||
|  |             List<ContactsDocument> lal = await SplitAuditlog(al); | ||||
|  | 
 | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             foreach (ContactsDocument ial in lal) | ||||
|  |             { | ||||
|  |                 ItemResponse<ContactsDocument> r = await container.UpsertItemAsync(ial, new PartitionKey(ial.id)); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             await UpdateMetadata(container, lal); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task UpdateMetadata(Container container, List<ContactsDocument> lal) | ||||
|  |         { | ||||
|  |             bool update = false; | ||||
|  |             string pKey = GetMetaDocumentKey(lal[0].AppKey, lal[0].UserId); | ||||
|  |             MetadataDocument md = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 ItemResponse<MetadataDocument> response = | ||||
|  |                     await container.ReadItemAsync<MetadataDocument>(pKey, new PartitionKey(pKey)); | ||||
|  |                 md = response.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |                 md = new MetadataDocument() | ||||
|  |                 { | ||||
|  |                     id = GetMetaDocumentKey(lal[0].AppKey, lal[0].UserId), | ||||
|  |                     PartitionKeys = new List<string>() | ||||
|  |                 }; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |                 return; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (md == null) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt("Something ugly happened!");
 | ||||
|  |                 return; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             foreach (ContactsDocument log in lal) | ||||
|  |             { | ||||
|  |                 if (md.PartitionKeys.Contains(log.id)) | ||||
|  |                     continue; | ||||
|  |                 md.PartitionKeys.Add(log.id); | ||||
|  |                 update = true; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (update) | ||||
|  |             { | ||||
|  |                 try | ||||
|  |                 { | ||||
|  |                     ItemResponse<MetadataDocument> r = await container.UpsertItemAsync(md, new PartitionKey(pKey)); | ||||
|  |                 } | ||||
|  |                 catch (Exception e) | ||||
|  |                 { | ||||
|  |                     //                    Helpers.LogIt(e.Message);   
 | ||||
|  |                     return; | ||||
|  |                 } | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task<List<ContactsDocument>> SplitAuditlog(ContactsDocument al) | ||||
|  |         { | ||||
|  |             List<ContactsDocument> lal = new List<ContactsDocument>(); | ||||
|  |             var sortedRecords = al.Records.OrderBy(record => record.EventTime).ToList(); | ||||
|  | 
 | ||||
|  |             var currentGroup = new List<ContactRecord>(); | ||||
|  |             var currentGroupSize = 0; | ||||
|  | 
 | ||||
|  |             int MaxDocumentSizeInBytes = 2 * 1024 * 1024; // 2MB
 | ||||
|  | 
 | ||||
|  |             int index = al.Index; // start for index passed in
 | ||||
|  |             foreach (var record in sortedRecords) | ||||
|  |             { | ||||
|  |                 var recordSize = record.CalculateRecordSize(); | ||||
|  | 
 | ||||
|  |                 if (currentGroupSize + recordSize > MaxDocumentSizeInBytes) | ||||
|  |                 { | ||||
|  |                     ContactsDocument i = new ContactsDocument(); | ||||
|  |                     i.Index = index++; | ||||
|  |                     i.Records = currentGroup; | ||||
|  |                     i.AppKey = al.AppKey; | ||||
|  |                     i.UserId = al.UserId; | ||||
|  |                     lal.Add(i); | ||||
|  | 
 | ||||
|  |                     currentGroup = new List<ContactRecord>(); | ||||
|  |                     currentGroupSize = 0; | ||||
|  |                 } | ||||
|  | 
 | ||||
|  |                 currentGroup.Add(record); | ||||
|  |                 currentGroupSize += recordSize; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (currentGroup.Any()) | ||||
|  |             { | ||||
|  |                 ContactsDocument i = new ContactsDocument(); | ||||
|  |                 i.Index = index++; | ||||
|  |                 i.Records = currentGroup; | ||||
|  |                 i.AppKey = al.AppKey; | ||||
|  |                 i.UserId = al.UserId; | ||||
|  |                 lal.Add(i); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return lal; | ||||
|  |         } | ||||
|  | 
 | ||||
|  | 
 | ||||
|  |         public static async Task<ContactsDocument> GetContactDocument(string key) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  |                 ItemResponse<ContactsDocument> response = await container.ReadItemAsync<ContactsDocument>(key, new PartitionKey(key)); | ||||
|  |                 if (response == null) | ||||
|  |                     return null; | ||||
|  | 
 | ||||
|  |                 ContactsDocument t = response.Resource; | ||||
|  |                 return t; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task<MetadataDocumentContact> GetMetadataDocument(string appKey, string userId) | ||||
|  |         { | ||||
|  |             MetadataDocumentContact md = null; | ||||
|  |             string pKey = GetMetaDocumentKey(appKey, userId); | ||||
|  |             string id = GetDocumentId(appKey, userId); | ||||
|  | 
 | ||||
|  |             PartitionKey partitionKey = new PartitionKeyBuilder() | ||||
|  |                     .Add(pKey) | ||||
|  |                     .Build(); | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 ItemResponse<MetadataDocumentContact> response = | ||||
|  |                     await container.ReadItemAsync<MetadataDocumentContact>(pKey, partitionKey); | ||||
|  |                 md = response.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |                 md = new MetadataDocumentContact() | ||||
|  |                 { | ||||
|  |                     id = id, | ||||
|  |                     PartitionKeys = new List<string>() | ||||
|  |                 }; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (md == null) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt("Something ugly happened!");
 | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  |             return md; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static string GetMetaDocumentKey(string appKey, string userId) | ||||
|  |         { | ||||
|  |             return $"{appKey}-{userId}-meta"; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static string GetDocumentId(string appKey, string userId) | ||||
|  |         { | ||||
|  |             return $"{appKey}-{userId}"; | ||||
|  |         } | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,311 @@ | |||||
|  | using Azure.Messaging.ServiceBus; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using System; | ||||
|  | using System.Text.Json; | ||||
|  | using System.Collections.Generic; | ||||
|  | using System.Linq; | ||||
|  | using System.Text; | ||||
|  | using System.Threading.Tasks; | ||||
|  | using Azure.Storage.Blobs; | ||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using System.Net; | ||||
|  | using Microsoft.DurableTask; | ||||
|  | using Microsoft.DurableTask.Client; | ||||
|  | using FromBodyAttribute = Microsoft.Azure.Functions.Worker.Http.FromBodyAttribute; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class MetaProcessor | ||||
|  |     { | ||||
|  |         private readonly ILogger<MetaProcessor> _logger; | ||||
|  |         private static Lazy<ServiceBusClient> lazyBusClient = new Lazy<ServiceBusClient>(InitializeServiceBusClient); | ||||
|  |         private static ServiceBusClient _serviceBusClient = lazyBusClient.Value; | ||||
|  | 
 | ||||
|  |         public MetaProcessor(ILogger<MetaProcessor> logger) | ||||
|  |         { | ||||
|  |             _logger = logger; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         private static ServiceBusClient InitializeServiceBusClient() | ||||
|  |         { | ||||
|  |             return new ServiceBusClient(Constants.SvcBusConnectionString); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task PublishBatchJobs(List<Job> jobs) | ||||
|  |         { | ||||
|  |             //Job[] jobsCopy = jobs.ToArray();
 | ||||
|  |             var sender = _serviceBusClient.CreateSender("mail-events-queue"); | ||||
|  |             ServiceBusMessageBatch batch = await sender.CreateMessageBatchAsync(); | ||||
|  |             jobs.ForEach(job => | ||||
|  |             { | ||||
|  |                 var msg = new ServiceBusMessage(JsonSerializer.Serialize(job)); | ||||
|  |                 bool isAdded = batch.TryAddMessage(msg); | ||||
|  |             }); | ||||
|  |             await sender.SendMessagesAsync(batch); | ||||
|  | 
 | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task PublishJob(Job job) | ||||
|  |         { | ||||
|  |             var sender = _serviceBusClient.CreateSender("mail-events-queue"); | ||||
|  |             ServiceBusMessage msg = new ServiceBusMessage(JsonSerializer.Serialize(job)); | ||||
|  |             await sender.SendMessageAsync(msg); | ||||
|  |         } | ||||
|  |         [Function("MailMetaProcessor")] | ||||
|  |         public async Task<Job> MailMetaProcessor([ActivityTrigger] MailRecord mailRecord, FunctionContext functionContext) | ||||
|  |         { | ||||
|  |             Job job = await TriggerMailProcessor(mailRecord); | ||||
|  |             return job; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("MailMetaOrchestrator")] | ||||
|  |         public async Task<Job> MailMetaOrchestrator([OrchestrationTrigger] TaskOrchestrationContext ctx) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 MailRecord record = ctx.GetInput<MailRecord>(); | ||||
|  |                 string jobId = await ctx.CallActivityAsync<string>(nameof(AddProtectionAudits), record); | ||||
|  |                 Job job = await ctx.CallActivityAsync<Job>(nameof(MailMetaProcessor), record); | ||||
|  |                 string revId = await ctx.CallActivityAsync<string>(nameof(AddMailCompundRevision), record); | ||||
|  |                 return job; | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogError(ex.Message); | ||||
|  |                 return new Job() { AppKey = "", EventType = JobType.MailMetaProcessing, Id = "failed" }; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddMailMetaDurable")] | ||||
|  |         public async Task<HttpResponseData> AddMailMetaDurable([HttpTrigger(AuthorizationLevel.Anonymous, "get", "post")] HttpRequestData req, | ||||
|  |             [FromBody] MailRecord recordMail, | ||||
|  |             [DurableClient] DurableTaskClient client) | ||||
|  |         { | ||||
|  |             bool isValid = true; | ||||
|  |             //bool isValid = await CDPLite.CheckJwtRequest(req);
 | ||||
|  |              | ||||
|  |             if (!isValid) | ||||
|  |             { | ||||
|  |                 HttpResponseData res = req.CreateResponse(HttpStatusCode.Unauthorized); | ||||
|  |                 return res; | ||||
|  |             } | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 string instanceId = await client.ScheduleNewOrchestrationInstanceAsync(nameof(MailMetaOrchestrator), recordMail); | ||||
|  |                 _logger.LogInformation("created instance: {0}", instanceId); | ||||
|  |                 return client.CreateCheckStatusResponse(req, instanceId); | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogError(ex.ToString()); | ||||
|  |                 HttpResponseData response = req.CreateResponse(HttpStatusCode.InternalServerError); | ||||
|  |                 return response; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddMailMeta")] | ||||
|  |         public async Task<ActionResult> AddMailMeta([HttpTrigger(AuthorizationLevel.Anonymous, "get", "post")] HttpRequestData req, [FromBody] MailRecord recordMail) | ||||
|  |         { | ||||
|  |             bool isValid = true; | ||||
|  |             //bool isValid = await CDPLite.CheckJwtRequest(req);
 | ||||
|  |             if (!isValid) | ||||
|  |             { | ||||
|  |                 return new UnauthorizedObjectResult("Jwt has expired please validate"); | ||||
|  |             } | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 MailRecord mailRecord = recordMail; | ||||
|  |                 //_logger.LogInformation(JsonSerializer.Serialize(recordMail));
 | ||||
|  |                 string res = await TriggerProtectionAuditsJob(mailRecord); | ||||
|  |                 res = await AddMailCompoundRevisionInternal(mailRecord); | ||||
|  |                 _logger.LogInformation($"protection audits? {res}"); | ||||
|  |                 Job job = await TriggerMailProcessor(mailRecord); | ||||
|  |                 return new OkObjectResult(job); | ||||
|  | 
 | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogError(ex.ToString()); | ||||
|  |                 return new StatusCodeResult(500); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddMailForwardingRequest")] | ||||
|  |         public async Task<IActionResult> AddMailForwardingRequest([HttpTrigger(AuthorizationLevel.Anonymous, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             ForwardMailDto forwardMail; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |                 _logger.LogInformation("requestbody: {0}", requestBody); | ||||
|  |                 forwardMail = JsonSerializer.Deserialize<ForwardMailDto>(requestBody); | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogInformation(ex.ToString()); | ||||
|  |                 return new StatusCodeResult(500); | ||||
|  |             } | ||||
|  |             string appKey = forwardMail.AppKey; | ||||
|  |             var userId = Helpers.HashAndShortenText(forwardMail.SenderEmail.ToLower()); | ||||
|  |             var fr = await CDPDB.GetFile(appKey, forwardMail.MailId, userId); | ||||
|  |             if ((!fr.Policy.CheckAccess("Manage")) && (!fr.Policy.CheckAccess("Owner"))) | ||||
|  |             { | ||||
|  |                 return new UnauthorizedObjectResult("User doesn't have rights to forward this email"); | ||||
|  |             } | ||||
|  |             RevisionEntry entry = await FileRevision.GetFileRevisionInternal(forwardMail.AppKey, forwardMail.MailId); | ||||
|  |             if (entry.CompoundDocument == null) | ||||
|  |             { | ||||
|  |                 return new BadRequestObjectResult("IncorrectMailId"); | ||||
|  |             } | ||||
|  |             MailMetaRecord mailRecord = JsonSerializer.Deserialize<MailMetaRecord>(entry.CompoundDocument.Contents); | ||||
|  | 
 | ||||
|  |             mailRecord.AttachmentDetails.ForEach(async (att) => | ||||
|  |             { | ||||
|  |                 forwardMail.ReceiverEmails.ForEach(async (receiver) => | ||||
|  |                 { | ||||
|  |                     await AddMailForwardPolicy(appKey, att.FileId, att.FileName, forwardMail.SenderEmail, receiver, "Read"); | ||||
|  |                 }); | ||||
|  |             }); | ||||
|  | 
 | ||||
|  |             forwardMail.ReceiverEmails.ForEach(async (receiver) => | ||||
|  |             { | ||||
|  |                 await AddMailForwardPolicy(appKey, mailRecord.BodyId, string.Concat(mailRecord.BodyId, "-", "MailBody"), forwardMail.SenderEmail, receiver, "Read"); | ||||
|  |             }); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult("Done"); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         internal async Task AddMailForwardPolicy(string appKey, string fileId, string fileName, string sender, string receiver, string policy = "Read") | ||||
|  |         { | ||||
|  |             AddFileUserDto addFileUserDto = new AddFileUserDto | ||||
|  |             { | ||||
|  |                 AppKey = appKey, | ||||
|  |                 FileId = fileId, | ||||
|  |                 FileName = fileName, | ||||
|  |                 Email = sender, | ||||
|  |                 EmailToAdd = receiver, | ||||
|  |                 Policy = policy | ||||
|  |             }; | ||||
|  |             await CDPLite.AddFileUserInternal(addFileUserDto); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddMailCompundRevision")] | ||||
|  |         public async Task<string> AddMailCompundRevision([ActivityTrigger] MailRecord record, FunctionContext context) | ||||
|  |         { | ||||
|  | 
 | ||||
|  |             return await AddMailCompoundRevisionInternal(record); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         internal async Task<string> AddMailCompoundRevisionInternal(MailRecord record) | ||||
|  |         { | ||||
|  |             RevisionDocument doc = new RevisionDocument | ||||
|  |             { | ||||
|  |                 RevisionId = Guid.NewGuid().ToString(), | ||||
|  |                 RevisionDate = DateTime.UtcNow, | ||||
|  |                 EditedBy = record.SenderEmail, | ||||
|  |                 Comments = "MailBody" | ||||
|  |             }; | ||||
|  |             MailMetaRecord mailMeta = new MailMetaRecord | ||||
|  |             { | ||||
|  |                 AppKey = record.AppKey, | ||||
|  |                 SenderEmail = record.SenderEmail, | ||||
|  |                 BodyContent = record.BodyContent, | ||||
|  |                 BodyId = record.MailId, | ||||
|  |                 AttachmentDetails = record.AttachmentDetails | ||||
|  |             }; | ||||
|  |             CompoundDocument compound = new CompoundDocument | ||||
|  |             { | ||||
|  |                 Id = Guid.NewGuid().ToString(), | ||||
|  |                 AppKey = record.AppKey, | ||||
|  |                 Contents = JsonSerializer.Serialize(mailMeta), | ||||
|  |                 DocumentType = "MailRecord" | ||||
|  |             }; | ||||
|  | 
 | ||||
|  |             string revisionId = await FileRevision.AddMailBodyRevision(doc, compound, record.AppKey, record.MailId); | ||||
|  |             return revisionId; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddProtectionAudits")] | ||||
|  |         public async Task<string> AddProtectionAudits([ActivityTrigger] MailRecord record, FunctionContext functionContext) | ||||
|  |         { | ||||
|  |             var res = await TriggerProtectionAuditsJob(record); | ||||
|  |             return res; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         internal async Task<string> TriggerProtectionAuditsJob(MailRecord record) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 string userId = Helpers.HashAndShortenText(record.SenderEmail); | ||||
|  |                 string message = string.Format($"{record.SenderEmail} protected an email body with id: {record.MailId}."); | ||||
|  |                 List<AuditEventMetadata> auditEvents = new List<AuditEventMetadata>(); | ||||
|  |                 AuditEventMetadata auditEvent = new AuditEventMetadata { Action = "Addded", FileId = record.MailId, FileName = "Outlook-Mail-Body", Message = message, UserId = userId }; | ||||
|  |                 auditEvents.Add(auditEvent); | ||||
|  |                 record.AttachmentDetails.ForEach((att) => | ||||
|  |                 { | ||||
|  |                     message = string.Format($"{record.SenderEmail} protected the email attachment {att.FileName} with id: {att.FileId}."); | ||||
|  |                     AuditEventMetadata attachmentEvent = new AuditEventMetadata { Action = "Added", FileId = att.FileId, FileName = att.FileName, Message = message, UserId = userId }; | ||||
|  |                     auditEvents.Add(attachmentEvent); | ||||
|  |                 }); | ||||
|  |                 Job job = await CDPLite.AddAuditsBatchedEvent(auditEvents, record.AppKey); | ||||
|  |                 await PublishJob(job); | ||||
|  |                 _logger.LogInformation(JsonSerializer.Serialize(job)); | ||||
|  |                 return "Done"; | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogError(ex.ToString()); | ||||
|  |                 throw ex; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         internal async Task<Job> TriggerMailProcessor(MailRecord mailRecord) | ||||
|  |         { | ||||
|  |             string Connection = Constants.StorageConnString; | ||||
|  |             var blobSvc = new BlobServiceClient(Connection); | ||||
|  |             var blobClient = blobSvc.GetBlobContainerClient(mailRecord.AppKey.ToLower()); | ||||
|  | 
 | ||||
|  |             if (!blobClient.Exists()) | ||||
|  |             { | ||||
|  |                 await blobClient.CreateAsync(); | ||||
|  |             } | ||||
|  |             byte[] fileBytes = System.Text.Encoding.UTF8.GetBytes(JsonSerializer.Serialize(mailRecord)); | ||||
|  |             string jobId = Guid.NewGuid().ToString(); | ||||
|  |             var blob = blobClient.GetBlobClient(jobId); | ||||
|  |             Stream fs = new MemoryStream(fileBytes); | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 await blob.UploadAsync(fs); | ||||
|  |             } | ||||
|  |             catch (Exception ex) | ||||
|  |             { | ||||
|  |                 _logger.LogError(ex.ToString()); | ||||
|  |                 throw ex; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             Job job = new Job { AppKey = mailRecord.AppKey, EventType = JobType.MailMetaProcessing, Id = jobId }; | ||||
|  |             await PublishJob(job); | ||||
|  |             return job; | ||||
|  |         } | ||||
|  |     } | ||||
|  | 
 | ||||
|  |     public class MailMetaRecord | ||||
|  |     { | ||||
|  |         public required string AppKey { get; set; } | ||||
|  |         public required string SenderEmail { get; set; } | ||||
|  |         public required string BodyId { get; set; } | ||||
|  |         public required string BodyContent { get; set; } //encrypted body
 | ||||
|  |         public required List<AttachmentDetails> AttachmentDetails { get; set; } | ||||
|  |     } | ||||
|  | 
 | ||||
|  |     public class ForwardMailDto | ||||
|  |     { | ||||
|  |         public string SenderEmail { get; set; } | ||||
|  |         public List<string> ReceiverEmails { get; set; } | ||||
|  |         public string MailId { get; set; } | ||||
|  |         public string AppKey { get; set; } | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,388 @@ | |||||
|  | using Microsoft.Azure.Cosmos; | ||||
|  | using System; | ||||
|  | using System.Collections.Generic; | ||||
|  | using System.Linq; | ||||
|  | using System.Text; | ||||
|  | using System.Threading.Tasks; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class GroupsDB | ||||
|  |     { | ||||
|  |         private static Lazy<CosmosClient> lazyClient = new Lazy<CosmosClient>(InitializeCosmosClient); | ||||
|  |         private static CosmosClient cosmosClient => lazyClient.Value; | ||||
|  |         private static string DatabaseName = "CDP"; | ||||
|  |         private static string ContainerName = "Groups"; | ||||
|  | 
 | ||||
|  |         private static CosmosClient InitializeCosmosClient() | ||||
|  |         { | ||||
|  |             // Perform any initialization here
 | ||||
|  |             var uri = "https://cdplite.documents.azure.com:443/"; | ||||
|  |             var authKey = "VPbg8RpzyI3XwhC2o0dIUtYFs33ghxORCqZeNAyg8vg4HWUBjM41BUxP0qLFXEvFh6ewQY1uKv52ACDbsEN1AQ=="; | ||||
|  | 
 | ||||
|  |             return new CosmosClient(uri, authKey); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<List<GroupsRecord>> GetUserGroups(string AppKey, string UserId) | ||||
|  |         { | ||||
|  |             var results = new List<GroupsRecord>(); | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             // Fetch the metadata document for the customer ID
 | ||||
|  |             var metadataDocumentId = GetMetaDocumentKey(AppKey, UserId); | ||||
|  |             MetadataDocumentGroups metadataDocument = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocumentResponse = | ||||
|  |                 await container.ReadItemAsync<MetadataDocumentGroups>(metadataDocumentId, new PartitionKey(metadataDocumentId)); | ||||
|  |                 metadataDocument = metadataDocumentResponse.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (metadataDocument == null) | ||||
|  |                 return results; | ||||
|  | 
 | ||||
|  |             // Determine the partition keys within the date range
 | ||||
|  |             var partitionKeysInDocument = metadataDocument.PartitionKeys; | ||||
|  |             // Fetch the audit records for each partition key within the date range
 | ||||
|  |             foreach (var partitionKey in partitionKeysInDocument) | ||||
|  |             { | ||||
|  | 
 | ||||
|  |                 ItemResponse<GroupsDocument> response = await container.ReadItemAsync<GroupsDocument>(partitionKey, new PartitionKey(partitionKey)); | ||||
|  |                 if (response == null) | ||||
|  |                     continue; | ||||
|  | 
 | ||||
|  |                 GroupsDocument t = response.Resource; | ||||
|  |                 results.AddRange(t.Records); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return results; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<bool> RemoveGroup(string appKey, string userId, string groupId) | ||||
|  |         { | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             // Fetch the metadata document for the customer ID
 | ||||
|  |             var metadataDocumentId = GetMetaDocumentKey(appKey, userId); | ||||
|  |             MetadataDocumentContact metadataDocument = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocumentResponse = await container.ReadItemAsync<MetadataDocumentContact>(metadataDocumentId, new PartitionKey(metadataDocumentId)); | ||||
|  |                 metadataDocument = metadataDocumentResponse.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (metadataDocument == null) | ||||
|  |                 return false; | ||||
|  | 
 | ||||
|  |             // Determine the partition keys within the date range
 | ||||
|  |             var partitionKeysInDocument = metadataDocument.PartitionKeys; | ||||
|  |             // Fetch the audit records for each partition key within the date range
 | ||||
|  |             foreach (var partitionKey in partitionKeysInDocument) | ||||
|  |             { | ||||
|  | 
 | ||||
|  |                 ItemResponse<GroupsDocument> response = await container.ReadItemAsync<GroupsDocument>(partitionKey, new PartitionKey(partitionKey)); | ||||
|  |                 if (response == null) | ||||
|  |                     continue; | ||||
|  | 
 | ||||
|  |                 GroupsDocument t = response.Resource; | ||||
|  |                 var originalCount = t.Records.Count; | ||||
|  |                 t.Records = t.Records.Where(item => item.id != groupId).ToList(); | ||||
|  |                 var afterFilterCount = t.Records.Count; | ||||
|  | 
 | ||||
|  |                 if (originalCount != afterFilterCount) | ||||
|  |                 { | ||||
|  |                     ItemResponse<GroupsDocument> updateResponse = await container.ReplaceItemAsync( | ||||
|  |                         item: t, | ||||
|  |                         id: t.id, | ||||
|  |                         partitionKey: new PartitionKey(partitionKey)); | ||||
|  |                     return true; | ||||
|  |                 } | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return false; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<bool> AppendGroup(string appKey, string userId, String grpName, string grpDescription, List<ContactRecord> rec) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocument = await GetMetadataDocument(appKey, userId); | ||||
|  |                 if (metadataDocument == null) | ||||
|  |                     return false; | ||||
|  | 
 | ||||
|  |                 string dayKey = metadataDocument.GetLatestKey(appKey, userId); | ||||
|  |                 GroupsDocument al = await GetGroupDocument(dayKey); | ||||
|  |                 if (al == null) | ||||
|  |                 { | ||||
|  |                     al = new GroupsDocument(); | ||||
|  |                     al.AppKey = appKey; | ||||
|  |                     al.UserId = userId; | ||||
|  |                 } | ||||
|  |                 GroupsRecord gr = new GroupsRecord(grpName, grpDescription, rec); | ||||
|  |                 al.Records.Add(gr); | ||||
|  |                 await UpdateGroupsDocument(al); | ||||
|  |                 return true; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 return false; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task<bool> UpdateGroup(string appKey, string userId, string grpId, string grpName, string grpDescription, List<ContactRecord> contacts) | ||||
|  |         { | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             // Fetch the metadata document for the customer ID
 | ||||
|  |             var metadataDocumentId = GetMetaDocumentKey(appKey, userId); | ||||
|  |             MetadataDocumentContact metadataDocument = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 var metadataDocumentResponse = await container.ReadItemAsync<MetadataDocumentContact>(metadataDocumentId, new PartitionKey(metadataDocumentId)); | ||||
|  |                 metadataDocument = metadataDocumentResponse.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (metadataDocument == null) | ||||
|  |                 return false; | ||||
|  | 
 | ||||
|  |             // Determine the partition keys within the date range
 | ||||
|  |             var partitionKeysInDocument = metadataDocument.PartitionKeys; | ||||
|  |             // Fetch the audit records for each partition key within the date range
 | ||||
|  |             foreach (var partitionKey in partitionKeysInDocument) | ||||
|  |             { | ||||
|  | 
 | ||||
|  |                 ItemResponse<GroupsDocument> response = await container.ReadItemAsync<GroupsDocument>(partitionKey, new PartitionKey(partitionKey)); | ||||
|  |                 if (response == null) | ||||
|  |                     continue; | ||||
|  | 
 | ||||
|  |                 GroupsDocument t = response.Resource; | ||||
|  |                 bool groupFound = false; | ||||
|  |                 foreach (GroupsRecord gr in t.Records) | ||||
|  |                 { | ||||
|  |                     if (gr.id == grpId) | ||||
|  |                     { | ||||
|  |                         gr.Name = grpName != "" ? grpName : gr.Name; | ||||
|  |                         gr.Description = grpDescription != "" ? grpDescription : gr.Description; | ||||
|  |                         gr.Contacts = contacts; | ||||
|  |                         groupFound = true; | ||||
|  |                         break; | ||||
|  |                     } | ||||
|  |                 } | ||||
|  | 
 | ||||
|  |                 if (groupFound) | ||||
|  |                 { | ||||
|  |                     ItemResponse<GroupsDocument> updateResponse = await container.ReplaceItemAsync( | ||||
|  |                         item: t, | ||||
|  |                         id: t.id, | ||||
|  |                         partitionKey: new PartitionKey(partitionKey)); | ||||
|  |                     return true; | ||||
|  |                 } | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return false; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         public static async Task UpdateGroupsDocument(GroupsDocument al) | ||||
|  |         { | ||||
|  |             if (al.Records.Count == 0) | ||||
|  |                 return; | ||||
|  | 
 | ||||
|  |             List<GroupsDocument> lal = await SplitUserGroups(al); | ||||
|  | 
 | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  | 
 | ||||
|  |             foreach (GroupsDocument ial in lal) | ||||
|  |             { | ||||
|  |                 ItemResponse<GroupsDocument> r = await container.UpsertItemAsync(ial, new PartitionKey(ial.id)); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             await UpdateMetadata(container, lal); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task UpdateMetadata(Container container, List<GroupsDocument> lal) | ||||
|  |         { | ||||
|  |             bool update = false; | ||||
|  |             string pKey = GetMetaDocumentKey(lal[0].AppKey, lal[0].UserId); | ||||
|  |             MetadataDocument md = null; | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 ItemResponse<MetadataDocument> response = | ||||
|  |                     await container.ReadItemAsync<MetadataDocument>(pKey, new PartitionKey(pKey)); | ||||
|  |                 md = response.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |                 md = new MetadataDocument() | ||||
|  |                 { | ||||
|  |                     id = GetMetaDocumentKey(lal[0].AppKey, lal[0].UserId), | ||||
|  |                     PartitionKeys = new List<string>() | ||||
|  |                 }; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |                 return; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (md == null) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt("Something ugly happened!");
 | ||||
|  |                 return; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             foreach (GroupsDocument log in lal) | ||||
|  |             { | ||||
|  |                 if (md.PartitionKeys.Contains(log.id)) | ||||
|  |                     continue; | ||||
|  |                 md.PartitionKeys.Add(log.id); | ||||
|  |                 update = true; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (update) | ||||
|  |             { | ||||
|  |                 try | ||||
|  |                 { | ||||
|  |                     ItemResponse<MetadataDocument> r = await container.UpsertItemAsync(md, new PartitionKey(pKey)); | ||||
|  |                 } | ||||
|  |                 catch (Exception e) | ||||
|  |                 { | ||||
|  |                     //                    Helpers.LogIt(e.Message);   
 | ||||
|  |                     return; | ||||
|  |                 } | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task<List<GroupsDocument>> SplitUserGroups(GroupsDocument al) | ||||
|  |         { | ||||
|  |             List<GroupsDocument> lal = new List<GroupsDocument>(); | ||||
|  |             var sortedRecords = al.Records.OrderBy(record => record.EventTime).ToList(); | ||||
|  | 
 | ||||
|  |             var currentGroup = new List<GroupsRecord>(); | ||||
|  |             var currentGroupSize = 0; | ||||
|  | 
 | ||||
|  |             int MaxDocumentSizeInBytes = 2 * 1024 * 1024; // 2MB
 | ||||
|  | 
 | ||||
|  |             int index = al.Index; // start for index passed in
 | ||||
|  |             foreach (var record in sortedRecords) | ||||
|  |             { | ||||
|  |                 var recordSize = record.CalculateRecordSize(); | ||||
|  | 
 | ||||
|  |                 if (currentGroupSize + recordSize > MaxDocumentSizeInBytes) | ||||
|  |                 { | ||||
|  |                     GroupsDocument i = new GroupsDocument(); | ||||
|  |                     i.Index = index++; | ||||
|  |                     i.Records = currentGroup; | ||||
|  |                     i.AppKey = al.AppKey; | ||||
|  |                     i.UserId = al.UserId; | ||||
|  |                     lal.Add(i); | ||||
|  | 
 | ||||
|  |                     currentGroup = new List<GroupsRecord>(); | ||||
|  |                     currentGroupSize = 0; | ||||
|  |                 } | ||||
|  | 
 | ||||
|  |                 currentGroup.Add(record); | ||||
|  |                 currentGroupSize += recordSize; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (currentGroup.Any()) | ||||
|  |             { | ||||
|  |                 GroupsDocument i = new GroupsDocument(); | ||||
|  |                 i.Index = index++; | ||||
|  |                 i.Records = currentGroup; | ||||
|  |                 i.AppKey = al.AppKey; | ||||
|  |                 i.UserId = al.UserId; | ||||
|  |                 lal.Add(i); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return lal; | ||||
|  |         } | ||||
|  | 
 | ||||
|  | 
 | ||||
|  |         public static async Task<GroupsDocument> GetGroupDocument(string key) | ||||
|  |         { | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  |                 ItemResponse<GroupsDocument> response = await container.ReadItemAsync<GroupsDocument>(key, new PartitionKey(key)); | ||||
|  |                 if (response == null) | ||||
|  |                     return null; | ||||
|  | 
 | ||||
|  |                 GroupsDocument t = response.Resource; | ||||
|  |                 return t; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static async Task<MetadataDocumentGroups> GetMetadataDocument(string appKey, string userId) | ||||
|  |         { | ||||
|  |             MetadataDocumentGroups md = null; | ||||
|  |             string pKey = GetMetaDocumentKey(appKey, userId); | ||||
|  |             string id = GetDocumentId(appKey, userId); | ||||
|  | 
 | ||||
|  |             PartitionKey partitionKey = new PartitionKeyBuilder() | ||||
|  |                     .Add(pKey) | ||||
|  |                     .Build(); | ||||
|  |             Container container = cosmosClient.GetContainer(DatabaseName, ContainerName); | ||||
|  |             try | ||||
|  |             { | ||||
|  |                 ItemResponse<MetadataDocumentGroups> response = | ||||
|  |                     await container.ReadItemAsync<MetadataDocumentGroups>(pKey, partitionKey); | ||||
|  |                 md = response.Resource; | ||||
|  |             } | ||||
|  |             catch (CosmosException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound) | ||||
|  |             { | ||||
|  |                 md = new MetadataDocumentGroups() | ||||
|  |                 { | ||||
|  |                     id = id, | ||||
|  |                     PartitionKeys = new List<string>() | ||||
|  |                 }; | ||||
|  |             } | ||||
|  |             catch (Exception e) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt(e.Message);
 | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             if (md == null) | ||||
|  |             { | ||||
|  |                 //                Helpers.LogIt("Something ugly happened!");
 | ||||
|  |                 return null; | ||||
|  |             } | ||||
|  |             return md; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static string GetMetaDocumentKey(string appKey, string userId) | ||||
|  |         { | ||||
|  |             return $"{appKey}-{userId}-meta"; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         static string GetDocumentId(string appKey, string userId) | ||||
|  |         { | ||||
|  |             return $"{appKey}-{userId}"; | ||||
|  |         } | ||||
|  |     } | ||||
|  | } | ||||
| @ -0,0 +1,107 @@ | |||||
|  | using Microsoft.AspNetCore.Http; | ||||
|  | using Microsoft.AspNetCore.Mvc; | ||||
|  | using Microsoft.Azure.Functions.Worker; | ||||
|  | using Microsoft.Azure.Functions.Worker.Http; | ||||
|  | using Microsoft.Extensions.Logging; | ||||
|  | using Newtonsoft.Json; | ||||
|  | 
 | ||||
|  | namespace CDP | ||||
|  | { | ||||
|  |     public class GroupsFunctions | ||||
|  |     { | ||||
|  |         private readonly ILogger<GroupsFunctions> _logger; | ||||
|  | 
 | ||||
|  |         public GroupsFunctions(ILogger<GroupsFunctions> logger) | ||||
|  |         { | ||||
|  |             _logger = logger; | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("AddGroup")] | ||||
|  |         public async Task<IActionResult> AddGroup([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Adding a new group invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             AddUserGroupDto addUserGroupDto = JsonConvert.DeserializeObject<AddUserGroupDto>(requestBody); | ||||
|  |             if (addUserGroupDto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(addUserGroupDto.EmailId.ToLower()); | ||||
|  | 
 | ||||
|  |             List<ContactRecord> contacts = new List<ContactRecord>(); | ||||
|  |             foreach (Contact contact in addUserGroupDto.Contacts) | ||||
|  |             { | ||||
|  |                 ContactRecord cr = new ContactRecord(contact.Name, contact.Email, contact.Phone, contact.Address, contact.Company); | ||||
|  |                 contacts.Add(cr); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             Boolean resp = await GroupsDB.AppendGroup(addUserGroupDto.AppKey, userId, addUserGroupDto.GroupName, addUserGroupDto.GroupDescription, contacts); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(resp); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("GetGroups")] | ||||
|  |         public async Task<IActionResult> GetGroups([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Getting groups invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             GetUserGroupDto getUserGroupDto = JsonConvert.DeserializeObject<GetUserGroupDto>(requestBody); | ||||
|  |             if (getUserGroupDto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(getUserGroupDto.EmailId.ToLower()); | ||||
|  | 
 | ||||
|  |             List<GroupsRecord> fr = await GroupsDB.GetUserGroups(getUserGroupDto.AppKey, userId); | ||||
|  |             if (fr == null) | ||||
|  |             { | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "groups not found " + getUserGroupDto.EmailId }); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(fr); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("DeleteGroup")] | ||||
|  |         public async Task<IActionResult> DeleteGroup([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Deleting Group invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             DeleteGroupDto deleteGroupdto = JsonConvert.DeserializeObject<DeleteGroupDto>(requestBody); | ||||
|  |             if (deleteGroupdto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(deleteGroupdto.UserEmail.ToLower()); | ||||
|  |             Boolean resp = await GroupsDB.RemoveGroup(deleteGroupdto.AppKey, userId, deleteGroupdto.GroupId); | ||||
|  |             return new OkObjectResult(resp); | ||||
|  |         } | ||||
|  | 
 | ||||
|  |         [Function("UpdateGroup")] | ||||
|  |         public async Task<IActionResult> UpdateGroup([HttpTrigger(AuthorizationLevel.Function, "get", "post")] HttpRequestData req) | ||||
|  |         { | ||||
|  |             _logger.LogInformation("Updating Group invoked"); | ||||
|  | 
 | ||||
|  |             // Convert the JSON payload to a string
 | ||||
|  |             string requestBody = await new StreamReader(req.Body).ReadToEndAsync(); | ||||
|  |             UpdateGroupDto updateUserGroupDto = JsonConvert.DeserializeObject<UpdateGroupDto>(requestBody); | ||||
|  |             if (updateUserGroupDto == null) | ||||
|  |                 return new BadRequestObjectResult(new { error = true, message = "Parse error." }); | ||||
|  | 
 | ||||
|  |             string userId = Helpers.HashAndShortenText(updateUserGroupDto.UserEmail.ToLower()); | ||||
|  | 
 | ||||
|  |             List<ContactRecord> contacts = new List<ContactRecord>(); | ||||
|  |             foreach (Contact contact in updateUserGroupDto.Contacts) | ||||
|  |             { | ||||
|  |                 ContactRecord cr = new ContactRecord(contact.Name, contact.Email, contact.Phone, contact.Address, contact.Company); | ||||
|  |                 contacts.Add(cr); | ||||
|  |             } | ||||
|  | 
 | ||||
|  |             Boolean resp = await GroupsDB.UpdateGroup(updateUserGroupDto.AppKey, userId, updateUserGroupDto.GroupId, updateUserGroupDto.GroupName, updateUserGroupDto.GroupDescription, contacts); | ||||
|  | 
 | ||||
|  |             return new OkObjectResult(resp); | ||||
|  |         } | ||||
|  |     } | ||||
|  | } | ||||
| @ -1,12 +1,15 @@ | |||||
| { | { | ||||
|     "version": "2.0", |  | ||||
|     "logging": { |  | ||||
|         "applicationInsights": { |  | ||||
|             "samplingSettings": { |  | ||||
|                 "isEnabled": true, |  | ||||
|                 "excludedTypes": "Request" |  | ||||
|             }, |  | ||||
|             "enableLiveMetricsFilters": true |  | ||||
|         } |  | ||||
|  |   "version": "2.0", | ||||
|  |   "logging": { | ||||
|  |     "applicationInsights": { | ||||
|  |       "samplingSettings": { | ||||
|  |         "isEnabled": true, | ||||
|  |         "excludedTypes": "Request" | ||||
|  |       }, | ||||
|  |       "enableLiveMetricsFilters": true | ||||
|     } |     } | ||||
|  |   }, | ||||
|  |   "tracing": { | ||||
|  |     "consoleLevel": "verbose" | ||||
|  |   } | ||||
| } | } | ||||
						Write
						Preview
					
					
					Loading…
					
					Cancel
						Save
					
		Reference in new issue