using System; using System.Linq; using System.Threading.Tasks; using Microsoft.EntityFrameworkCore; using AyaNova.Util; using AyaNova.Api.ControllerHelpers; using AyaNova.Models; using Newtonsoft.Json.Linq; using System.Collections.Generic; using System.Text.RegularExpressions; namespace AyaNova.Biz { internal class TranslationBiz : BizObject { public bool SeedOrImportRelaxedRulesMode { get; set; } internal TranslationBiz(AyContext dbcontext, long currentUserId, long userTranslationId, AuthorizationRoles userRoles) { ct = dbcontext; UserId = currentUserId; UserTranslationId = userTranslationId; CurrentUserRoles = userRoles; BizType = AyaType.Translation; SeedOrImportRelaxedRulesMode = false;//default } internal static TranslationBiz GetBiz(AyContext ct, Microsoft.AspNetCore.Http.HttpContext httpContext = null) { if (httpContext != null) return new TranslationBiz(ct, UserIdFromContext.Id(httpContext.Items), UserTranslationIdFromContext.Id(httpContext.Items), UserRolesFromContext.Roles(httpContext.Items)); else return new TranslationBiz(ct, 1, ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID, AuthorizationRoles.BizAdminFull); } //////////////////////////////////////////////////////////////////////////////////////////////// //DUPLICATE - only way to create a new translation // internal async Task DuplicateAsync(NameIdItem inObj) { //make sure sourceid exists if (!await TranslationExistsAsync(inObj.Id)) AddError(ApiErrorCode.VALIDATION_INVALID_VALUE, "Id", "Source translation id does not exist"); //Ensure name is unique and not too long and not empty await ValidateAsync(inObj.Name, true); if (HasErrors) return null; //fetch the existing translation for duplication var SourceTranslation = await ct.Translation.Include(x => x.TranslationItems).SingleOrDefaultAsync(m => m.Id == inObj.Id); //replicate the source to a new dest and save Translation NewTranslation = new Translation(); NewTranslation.Name = inObj.Name; NewTranslation.Stock = false; NewTranslation.CjkIndex = false; foreach (TranslationItem i in SourceTranslation.TranslationItems) { NewTranslation.TranslationItems.Add(new TranslationItem() { Key = i.Key, Display = i.Display }); } //Add it to the context so the controller can save it await ct.Translation.AddAsync(NewTranslation); await ct.SaveChangesAsync(); //Log await EventLogProcessor.LogEventToDatabaseAsync(new Event(UserId, NewTranslation.Id, AyaType.Translation, AyaEvent.Created), ct); return NewTranslation; } //////////////////////////////////////////////////////////////////////////////////////////////// /// GET //Get entire translation internal async Task GetAsync(long fetchId) { //This is simple so nothing more here, but often will be copying to a different output object or some other ops return await ct.Translation.Include(x => x.TranslationItems).SingleOrDefaultAsync(m => m.Id == fetchId); } //get list (simple non-paged) internal async Task> GetTranslationListAsync() { List l = new List(); l = await ct.Translation .AsNoTracking() .OrderBy(m => m.Name) .Select(m => new NameIdItem() { Id = m.Id, Name = m.Name }).ToListAsync(); return l; } #if (DEBUG) internal async Task TranslationKeyCoverageAsync() { AyaNova.Api.Controllers.TranslationController.TranslationCoverageInfo L = new AyaNova.Api.Controllers.TranslationController.TranslationCoverageInfo(); L.RequestedKeys = ServerBootConfig.TranslationKeysRequested; L.RequestedKeys.Sort(); var AllKeys = await GetKeyListAsync(); foreach (string StockKey in AllKeys) { if (!L.RequestedKeys.Contains(StockKey)) { L.NotRequestedKeys.Add(StockKey); } } L.NotRequestedKeys.Sort(); L.RequestedKeyCount = L.RequestedKeys.Count; L.NotRequestedKeyCount = L.NotRequestedKeys.Count; return L; } //Track requests for keys so we can determine which are being used and which are not //TODO: Ideally this should be paired with tests that either directly request each key that are def. being used //or the UI needs to be tested in a way that triggers every key to be used even errors etc internal static void TrackRequestedKey(string key) { if (!ServerBootConfig.TranslationKeysRequested.Contains(key)) ServerBootConfig.TranslationKeysRequested.Add(key); } internal static void TrackRequestedKey(List keys) { foreach (string Key in keys) { if (!ServerBootConfig.TranslationKeysRequested.Contains(Key)) ServerBootConfig.TranslationKeysRequested.Add(Key); } } #endif //Get the keys for a list of keys provided internal async Task>> GetSubsetAsync(List param) { #if (DEBUG) TrackRequestedKey(param); #endif var ret = await ct.TranslationItem.Where(x => x.TranslationId == UserTranslationId && param.Contains(x.Key)).ToDictionaryAsync(x => x.Key, x => x.Display); return ret.ToList(); } //Get the keys for a list of keys provided, static format for calling from other internal classes internal static async Task> GetSubsetStaticAsync(List param, long translationId) { #if (DEBUG) TrackRequestedKey(param); #endif AyContext ct = ServiceProviderProvider.DBContext; if (!await ct.Translation.AnyAsync(e => e.Id == translationId)) translationId = ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID; var ret = await ct.TranslationItem.Where(x => x.TranslationId == translationId && param.Contains(x.Key)).AsNoTracking().ToDictionaryAsync(x => x.Key, x => x.Display); return ret; } //Get the CJKIndex value for the translation specified internal static async Task GetCJKIndexAsync(long translationId, AyContext ct = null) { if (ct == null) ct = ServiceProviderProvider.DBContext; var ret = await ct.Translation.Where(x => x.Id == translationId).AsNoTracking().Select(m => m.CjkIndex).SingleOrDefaultAsync(); return ret; } /// /// Get the value of the key provided in the default translation chosen /// /// /// internal static async Task GetDefaultTranslationAsync(string key) { if (string.IsNullOrWhiteSpace(key)) return "ERROR: GetDefaultTranslation NO KEY VALUE SPECIFIED"; #if (DEBUG) TrackRequestedKey(key); #endif AyContext ct = ServiceProviderProvider.DBContext; return await ct.TranslationItem.Where(m => m.TranslationId == ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID && m.Key == key).Select(m => m.Display).AsNoTracking().FirstOrDefaultAsync(); } //Get all stock keys that are valid (used for key coverage reporting) internal static async Task> GetKeyListAsync() { AyContext ct = ServiceProviderProvider.DBContext; return await ct.TranslationItem.Where(m => m.TranslationId == 1).OrderBy(m => m.Key).Select(m => m.Key).AsNoTracking().ToListAsync(); } //////////////////////////////////////////////////////////////////////////////////////////////// //UPDATE // internal async Task PutTranslationItemDisplayTextAsync(TranslationItem dbObj, NewTextIdConcurrencyTokenItem inObj, Translation dbParent) { if (dbParent.Stock == true) { AddError(ApiErrorCode.INVALID_OPERATION, "object", "TranslationItem is from a Stock translation and cannot be modified"); return false; } //Replace the db object with the PUT object //CopyObject.Copy(inObj, dbObj, "Id"); dbObj.Display = inObj.NewText; //Set "original" value of concurrency token to input token //this will allow EF to check it out ct.Entry(dbObj).OriginalValues["Concurrency"] = inObj.Concurrency; //Only thing to validate is if it has data at all in it if (string.IsNullOrWhiteSpace(inObj.NewText)) AddError(ApiErrorCode.VALIDATION_REQUIRED, "Display (NewText)"); if (HasErrors) return false; await ct.SaveChangesAsync(); //Log await EventLogProcessor.LogEventToDatabaseAsync(new Event(UserId, dbParent.Id, AyaType.Translation, AyaEvent.Modified), ct); return true; } internal async Task PutTranslationItemsDisplayTextAsync(List inObj, Translation dbParent) { if (dbParent.Stock == true) { AddError(ApiErrorCode.INVALID_OPERATION, "object", "TranslationItem is from a Stock translation and cannot be modified"); return false; } foreach (NewTextIdConcurrencyTokenItem tit in inObj) { var titem = await ct.TranslationItem.SingleOrDefaultAsync(m => m.Id == tit.Id); if (titem == null) { AddError(ApiErrorCode.NOT_FOUND, $"Translation item ID {tit.Id}"); return false; } //Replace the db object with the PUT object //CopyObject.Copy(inObj, dbObj, "Id"); titem.Display = tit.NewText; //Set "original" value of concurrency token to input token //this will allow EF to check it out ct.Entry(titem).OriginalValues["Concurrency"] = tit.Concurrency; //Only thing to validate is if it has data at all in it if (string.IsNullOrWhiteSpace(tit.NewText)) AddError(ApiErrorCode.VALIDATION_REQUIRED, $"Display (NewText) for Id: {tit.Id}"); } if (HasErrors) return false; await ct.SaveChangesAsync(); //Log await EventLogProcessor.LogEventToDatabaseAsync(new Event(UserId, dbParent.Id, AyaType.Translation, AyaEvent.Modified), ct); return true; } internal async Task PutTranslationNameAsync(Translation dbObj, NewTextIdConcurrencyTokenItem inObj) { if (dbObj.Stock == true) { AddError(ApiErrorCode.INVALID_OPERATION, "object", "Translation is a Stock translation and cannot be modified"); return false; } dbObj.Name = inObj.NewText; //Set "original" value of concurrency token to input token //this will allow EF to check it out ct.Entry(dbObj).OriginalValues["Concurrency"] = inObj.Concurrency; await ValidateAsync(dbObj.Name, false); if (HasErrors) return false; await ct.SaveChangesAsync(); //Log await EventLogProcessor.LogEventToDatabaseAsync(new Event(UserId, dbObj.Id, AyaType.Translation, AyaEvent.Modified), ct); return true; } //////////////////////////////////////////////////////////////////////////////////////////////// //DELETE // internal async Task DeleteAsync(Translation dbObj) { //Determine if the object can be deleted, do the deletion tentatively await ValidateCanDeleteAsync(dbObj); if (HasErrors) return false; ct.Translation.Remove(dbObj); await ct.SaveChangesAsync(); //Log await EventLogProcessor.DeleteObjectLogAsync(UserId, AyaType.Translation, dbObj.Id, dbObj.Name, ct); return true; } //////////////////////////////////////////////////////////////////////////////////////////////// //VALIDATION // //Can save or update? private async Task ValidateAsync(string inObjName, bool isNew) { //run validation and biz rules //Name required if (string.IsNullOrWhiteSpace(inObjName)) AddError(ApiErrorCode.VALIDATION_REQUIRED, "Name"); //Name must be less than 255 characters if (inObjName.Length > 255) AddError(ApiErrorCode.VALIDATION_LENGTH_EXCEEDED, "Name", "255 char max"); //Name must be unique if (await ct.Translation.AnyAsync(m => m.Name == inObjName)) AddError(ApiErrorCode.VALIDATION_NOT_UNIQUE, "Name"); return; } //Can delete? private async Task ValidateCanDeleteAsync(Translation inObj) { //Decided to short circuit these; if there is one issue then return immediately (fail fast rule) //Ensure it's not a stock translation if (inObj.Stock == true) { AddError(ApiErrorCode.INVALID_OPERATION, "object", "Translation is a Stock translation and cannot be deleted"); return; } if (inObj.Id == ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID) { AddError(ApiErrorCode.INVALID_OPERATION, "object", "Translation is set as the default server translation (AYANOVA_DEFAULT_LANGUAGE_ID) and can not be deleted"); return; } //See if any users exist with this translation selected in which case it's not deleteable if (await ct.UserOptions.AnyAsync(e => e.TranslationId == inObj.Id)) { AddError(ApiErrorCode.VALIDATION_REFERENTIAL_INTEGRITY, "object", "Can't be deleted in use by one or more Users"); return; } } //////////////////////////////////////////////////////////////////////////////////////////////// //UTILITIES // public async Task TranslationNameToIdAsync(string translationName) { var v = await ct.Translation.AsNoTracking().FirstOrDefaultAsync(c => c.Name == translationName); if (v == null) return 0; return v.Id; } public static async Task TranslationNameToIdStaticAsync(string translationName, AyContext ct = null) { if (ct == null) { ct = ServiceProviderProvider.DBContext; } var v = await ct.Translation.AsNoTracking().FirstOrDefaultAsync(c => c.Name == translationName); if (v == null) return ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID; return v.Id; } public async Task TranslationExistsAsync(string translationName) { return await ct.Translation.AnyAsync(c => c.Name == translationName); } public async Task TranslationExistsAsync(long id) { return await ct.Translation.AnyAsync(e => e.Id == id); } //this is only called by Search.cs to cache a local cjk and stopwords, no one else calls it currently public static async Task ReturnSpecifiedTranslationIdIfExistsOrDefaultTranslationId(long id, AyContext ct) { if (!await ct.Translation.AnyAsync(e => e.Id == id)) return ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID; return id; } public async Task TranslationItemExistsAsync(long id) { return await ct.TranslationItem.AnyAsync(e => e.Id == id); } /// /// Ensure stock Translations and setup defaults /// Called by boot preflight check code AFTER it has already ensured the translation is a two letter code if stock one was chosen /// public async Task ValidateTranslationsAsync() { //Ensure default translations are present and that there is a server default translation that exists if (!await TranslationExistsAsync("en")) { throw new System.Exception($"E1015: stock translation English (en) not found in database!"); } if (!await TranslationExistsAsync("es")) { throw new System.Exception($"E1015: stock translation Spanish (es) not found in database!"); } if (!await TranslationExistsAsync("de")) { throw new System.Exception($"E1015: stock translation German (de) not found in database!"); } if (!await TranslationExistsAsync("fr")) { throw new System.Exception($"E1015: stock translation French (fr) not found in database!"); } //Ensure chosen default translation exists switch (ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION) { case "en": case "es": case "de": case "fr": break; default: if (!await TranslationExistsAsync(ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION)) { throw new System.Exception($"E1015: stock translation {ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION} not found in database!"); } break; } //Put the default translation ID number into the ServerBootConfig for later use ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION_ID = await TranslationNameToIdAsync(ServerBootConfig.AYANOVA_DEFAULT_TRANSLATION); } ///////////////////////////////////////////////////////////////////// }//eoc // }//eons