private async Task <bool> BackupToCloud() { StringBuilder sb = new StringBuilder(); StringBuilder sbFailures = new StringBuilder(); List <EarnedGratuity> lstUsersWithCloudBackup = EarnedGratuity.GratuitiesForUser(string.Empty, Gratuity.GratuityTypes.CloudBackup); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersWithCloudBackup.RemoveAll(eg => eg.Username.CompareCurrentCultureIgnoreCase(UserRestriction) != 0); } foreach (EarnedGratuity eg in lstUsersWithCloudBackup) { StorageID sid = StorageID.None; if (eg.UserProfile != null && ((sid = eg.UserProfile.BestCloudStorage) != StorageID.None) && eg.CurrentStatus != EarnedGratuity.EarnedGratuityStatus.Expired) { try { Profile pf = eg.UserProfile; LogbookBackup lb = new LogbookBackup(pf); switch (sid) { case StorageID.Dropbox: await BackupDropbox(lb, pf, sb, sbFailures).ConfigureAwait(false); break; case StorageID.OneDrive: await BackupOneDrive(lb, pf, sb, sbFailures).ConfigureAwait(false); break; case StorageID.GoogleDrive: await BackupGoogleDrive(lb, pf, sb, sbFailures).ConfigureAwait(false); break; case StorageID.iCloud: break; default: break; } } catch (Exception ex) when(!(ex is OutOfMemoryException)) { string szError = String.Format(CultureInfo.CurrentCulture, "eg user={0}{1}\r\n\r\n{2}\r\n\r\n{3}", eg == null ? "NULL eg!" : eg.Username, (eg != null && eg.UserProfile == null) ? " NULL PROFILE" : string.Empty, sbFailures.ToString(), sb.ToString()); util.NotifyAdminException("ERROR running nightly backup", new MyFlightbookException(szError, ex)); } } } ; util.NotifyAdminEvent("Dropbox report", sbFailures.ToString() + sb.ToString(), ProfileRoles.maskCanReport); return(true); }
private void InitDonations() { List <Gratuity> lstKnownGratuities = new List <Gratuity>(Gratuity.KnownGratuities); lstKnownGratuities.Sort((g1, g2) => { return(g1.Threshold.CompareTo(g2.Threshold)); }); rptAvailableGratuities.DataSource = lstKnownGratuities; rptAvailableGratuities.DataBind(); pnlPaypalCanceled.Visible = util.GetStringParam(Request, "pp").CompareCurrentCultureIgnoreCase("canceled") == 0; pnlPaypalSuccess.Visible = util.GetStringParam(Request, "pp").CompareCurrentCultureIgnoreCase("success") == 0; lblDonatePrompt.Text = Branding.ReBrand(Resources.LocalizedText.DonatePrompt); gvDonations.DataSource = Payment.RecordsForUser(User.Identity.Name); gvDonations.DataBind(); List <EarnedGratuity> lst = EarnedGratuity.GratuitiesForUser(User.Identity.Name, Gratuity.GratuityTypes.Unknown); lst.RemoveAll(eg => eg.CurrentStatus == EarnedGratuity.EarnedGratuityStatus.Expired); if (pnlEarnedGratuities.Visible = (lst.Count > 0)) { rptEarnedGratuities.DataSource = lst; rptEarnedGratuities.DataBind(); } }
private async Task <bool> BackupToCloud() { System.Text.StringBuilder sb = new System.Text.StringBuilder(); System.Text.StringBuilder sbFailures = new System.Text.StringBuilder(); List <EarnedGratuity> lstUsersWithCloudBackup = EarnedGratuity.GratuitiesForUser(string.Empty, Gratuity.GratuityTypes.CloudBackup); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersWithCloudBackup.RemoveAll(eg => eg.Username.CompareCurrentCultureIgnoreCase(UserRestriction) != 0); } foreach (EarnedGratuity eg in lstUsersWithCloudBackup) { StorageID sid = StorageID.None; if (eg.UserProfile != null && ((sid = eg.UserProfile.BestCloudStorage) != StorageID.None) && eg.CurrentStatus != EarnedGratuity.EarnedGratuityStatus.Expired) { try { Profile pf = eg.UserProfile; LogbookBackup lb = new LogbookBackup(pf); switch (sid) { case StorageID.Dropbox: { try { MFBDropbox.TokenStatus ts = await new MFBDropbox().ValidateDropboxToken(pf, true, true); if (ts == MFBDropbox.TokenStatus.None) { continue; } Dropbox.Api.Files.FileMetadata result = null; result = await lb.BackupToDropbox(Branding.CurrentBrand); sb.AppendFormat(CultureInfo.CurrentCulture, "Dropbox: user {0} ", pf.UserName); if (ts == MFBDropbox.TokenStatus.oAuth1) { sb.Append("Token UPDATED from oauth1! "); } sb.AppendFormat(CultureInfo.CurrentCulture, "Logbook backed up for user {0}...", pf.UserName); System.Threading.Thread.Sleep(0); result = await lb.BackupImagesToDropbox(Branding.CurrentBrand); System.Threading.Thread.Sleep(0); sb.AppendFormat(CultureInfo.CurrentCulture, "and images backed up for user {0}.\r\n \r\n", pf.UserName); } catch (Dropbox.Api.ApiException <Dropbox.Api.Files.UploadError> ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (Dropbox.Api.ApiException<Dropbox.Api.Files.UploadError) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); string szMessage = (ex.ErrorResponse.IsPath && ex.ErrorResponse.AsPath != null && ex.ErrorResponse.AsPath.Value.Reason.IsInsufficientSpace) ? Resources.LocalizedText.DropboxErrorOutOfSpace : ex.Message; util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, szMessage, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (Dropbox.Api.ApiException <Dropbox.Api.Auth.TokenFromOAuth1Error> ex) { // De-register dropbox. pf.DropboxAccessToken = string.Empty; pf.FCommit(); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (TokenFromOAuth1Error, token removed) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, Resources.LocalizedText.DropboxErrorDeAuthorized), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (Dropbox.Api.AuthException ex) { // De-register dropbox. pf.DropboxAccessToken = string.Empty; pf.FCommit(); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (AuthException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, Resources.LocalizedText.DropboxErrorDeAuthorized), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (Dropbox.Api.BadInputException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (BadInputException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), false, false); } catch (Dropbox.Api.HttpException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (HttpException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (Dropbox.Api.AccessException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (AccessException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (Dropbox.Api.DropboxException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (Base dropbox exception) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (UnauthorizedAccessException ex) { // De-register dropbox. pf.DropboxAccessToken = string.Empty; pf.FCommit(); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (UnauthorizedAccess) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, Resources.LocalizedText.DropboxErrorDeAuthorized), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (MyFlightbookException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (MyFlightbookException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.DropboxFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.DropboxFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (System.IO.FileNotFoundException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user: FileNotFoundException, no notification sent {0}: {1} {2}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message); } catch (Exception ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Dropbox FAILED for user (Unknown Exception), no notification sent {0}: {1} {2}\r\n\r\n{3}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message, ex.StackTrace); if (ex.InnerException != null) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "Inner exception: {0}\r\n{1}", ex.InnerException.Message, ex.InnerException.StackTrace); } } } break; case StorageID.OneDrive: { try { if (pf.OneDriveAccessToken == null) { throw new UnauthorizedAccessException(); } Microsoft.OneDrive.Sdk.Item item = null; OneDrive od = new OneDrive(pf.OneDriveAccessToken); item = await lb.BackupToOneDrive(od); sb.AppendFormat(CultureInfo.CurrentCulture, "OneDrive: user {0} ", pf.UserName); sb.AppendFormat(CultureInfo.CurrentCulture, "Logbook backed up for user {0}...", pf.UserName); System.Threading.Thread.Sleep(0); item = await lb.BackupImagesToOneDrive(od, Branding.CurrentBrand); System.Threading.Thread.Sleep(0); sb.AppendFormat(CultureInfo.CurrentCulture, "and images backed up for user {0}.\r\n \r\n", pf.UserName); // if we are here we were successful, so save the updated refresh token if (String.Compare(pf.OneDriveAccessToken.RefreshToken, od.AuthState.RefreshToken, StringComparison.Ordinal) != 0) { pf.OneDriveAccessToken.RefreshToken = od.AuthState.RefreshToken; pf.FCommit(); } } catch (Microsoft.OneDrive.Sdk.OneDriveException ex) { string szMessage = OneDrive.MessageForException(ex); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "OneDrive FAILED for user (OneDriveException) {0}: {1}\r\n\r\n", pf.UserName, szMessage + " " + ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.OneDriveFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.OneDriveFailure, pf.UserFullName, szMessage, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (MyFlightbookException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "OneDrive FAILED for user (MyFlightbookException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.OneDriveFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.OneDriveFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (UnauthorizedAccessException ex) { // De-register oneDrive. pf.OneDriveAccessToken = null; pf.FCommit(); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "OneDrive FAILED for user (UnauthorizedAccess) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.OneDriveFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.OneDriveFailure, pf.UserFullName, ex.Message, Resources.LocalizedText.DropboxErrorDeAuthorized), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (System.IO.FileNotFoundException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "OneDrive FAILED for user: FileNotFoundException, no notification sent {0}: {1} {2}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message); } catch (Exception ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "OneDrive FAILED for user (Unknown Exception), no notification sent {0}: {1} {2}\r\n\r\n{3}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message, ex.StackTrace); } } break; case StorageID.GoogleDrive: { try { if (pf.GoogleDriveAccessToken == null) { throw new UnauthorizedAccessException(); } GoogleDrive gd = new GoogleDrive(pf.GoogleDriveAccessToken); bool fRefreshed = await gd.RefreshAccessToken(); sb.AppendFormat(CultureInfo.CurrentCulture, "GoogleDrive: user {0} ", pf.UserName); IReadOnlyDictionary <string, string> meta = await lb.BackupToGoogleDrive(gd, Branding.CurrentBrand); if (meta != null) { sb.AppendFormat(CultureInfo.CurrentCulture, "Logbook backed up for user {0}...", pf.UserName); } System.Threading.Thread.Sleep(0); meta = await lb.BackupImagesToGoogleDrive(gd, Branding.CurrentBrand); System.Threading.Thread.Sleep(0); if (meta != null) { sb.AppendFormat(CultureInfo.CurrentCulture, "and images backed up for user {0}.\r\n \r\n", pf.UserName); } // if we are here we were successful, so save the updated refresh token if (fRefreshed) { pf.FCommit(); } } catch (MyFlightbookException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "GoogleDrive FAILED for user (MyFlightbookException) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.GoogleDriveFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.GoogleDriveFailure, pf.UserFullName, ex.Message, string.Empty), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (UnauthorizedAccessException ex) { // De-register GoogleDrive. pf.GoogleDriveAccessToken = null; pf.FCommit(); sbFailures.AppendFormat(CultureInfo.CurrentCulture, "GoogleDrive FAILED for user (UnauthorizedAccess) {0}: {1}\r\n\r\n", pf.UserName, ex.Message); util.NotifyUser(Branding.ReBrand(Resources.EmailTemplates.GoogleDriveFailureSubject, ActiveBrand), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.EmailTemplates.GoogleDriveFailure, pf.UserFullName, ex.Message, Resources.LocalizedText.DropboxErrorDeAuthorized), ActiveBrand), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), true, false); } catch (System.IO.FileNotFoundException ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "GoogleDrive FAILED for user: FileNotFoundException, no notification sent {0}: {1} {2}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message); } catch (Exception ex) { sbFailures.AppendFormat(CultureInfo.CurrentCulture, "GoogleDrive FAILED for user (Unknown Exception), no notification sent {0}: {1} {2}\r\n\r\n", pf.UserName, ex.GetType().ToString(), ex.Message); } } break; case StorageID.iCloud: break; default: break; } } catch (Exception ex) { string szError = String.Format(CultureInfo.CurrentCulture, "eg user={0}{1}\r\n\r\n{2}\r\n\r\n{3}", eg == null ? "NULL eg!" : eg.Username, (eg != null && eg.UserProfile == null) ? " NULL PROFILE" : string.Empty, sbFailures.ToString(), sb.ToString()); util.NotifyAdminException("ERROR running nightly backup", new Exception(szError, ex)); } } } ; util.NotifyAdminEvent("Dropbox report", sbFailures.ToString() + sb.ToString(), ProfileRoles.maskCanReport); return(true); }
/// <summary> /// Sends the nightly/monthly emails for users that have requested it. /// </summary> private void SendNightlyEmails() { // Find all users who have expiring currencies - they may trigger an early email. List <EarnedGratuity> lstUsersWithExpiringCurrencies = EarnedGratuity.GratuitiesForUser(null, Gratuity.GratuityTypes.CurrencyAlerts); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersWithExpiringCurrencies.RemoveAll(eg => eg.Username.CompareCurrentCultureIgnoreCase(UserRestriction) != 0); } // get the list of people who have a subscription OTHER than simple monthly List <Profile> lstUsersToSend = new List <Profile>(Profile.UsersWithSubscriptions(~EmailSubscription.FlagForType(SubscriptionType.MonthlyTotals), DateTime.Now.AddDays(-7))); // And the list of people who have a subscription to expiring currencies List <Profile> lstUsersSubscribedForExpiration = new List <Profile>(Profile.UsersWithSubscriptions(EmailSubscription.FlagForType(SubscriptionType.Expiration), DateTime.Now.AddDays(-7))); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersToSend.RemoveAll(pf => pf.UserName.CompareOrdinalIgnoreCase(UserRestriction) != 0); } // See who has expiring currencies that require notification. foreach (EarnedGratuity eg in lstUsersWithExpiringCurrencies) { // Skip over anybody that's not also subscribed for expiration notices Profile pf = lstUsersSubscribedForExpiration.Find(profile => profile.UserName.CompareCurrentCultureIgnoreCase(eg.Username) == 0); if (pf == null) { continue; } IEnumerable <FlightCurrency.CurrencyStatusItem> expiringCurrencies = FlightCurrency.CurrencyStatusItem.CheckForExpiringCurrencies(eg.State, eg.Username, out string newState); if (newState.CompareOrdinal(eg.State) != 0) { eg.State = newState; eg.Commit(); } if (expiringCurrencies.Count() > 0) { if (SendMailForUser(pf, Resources.Profile.EmailCurrencyExpiringMailSubject, string.Empty)) { // Don't send the weekly mail, since we just pre-empted it. lstUsersToSend.RemoveAll(p => pf.UserName.CompareCurrentCultureIgnoreCase(eg.Username) == 0); pf.LastEmailDate = DateTime.Now; pf.FCommit(); } } } foreach (Profile pf in lstUsersToSend) { if (SendMailForUser(pf, Resources.Profile.EmailWeeklyMailSubject, String.Empty)) { pf.LastEmailDate = DateTime.Now; pf.FCommit(); } } // Now do the monthly/annual emails if (DateTime.Now.Day == 1) { lstUsersToSend = new List <Profile>(Profile.UsersWithSubscriptions(EmailSubscription.FlagForType(SubscriptionType.MonthlyTotals), DateTime.Now.AddDays(1))); // We don't update the last-email sent on this because this email is asynchronous - i.e., not dependent on any other mail that was sent. foreach (Profile pf in lstUsersToSend) { SendMailForUser(pf, String.Format(CultureInfo.CurrentCulture, Resources.Profile.EmailMonthlySubject, DateTime.Now.AddMonths(-1).ToString("MMMM", CultureInfo.InvariantCulture)), "monthly"); } } }
private async Task <bool> BackupToCloud() { string szBackups = Path.Combine(Path.GetTempPath(), String.Format(CultureInfo.InvariantCulture, "{0}-CloudBackup.txt", DateTime.Now.ToString("yyyy-MM-dd", CultureInfo.InvariantCulture))); string szBackupFailures = Path.Combine(Path.GetTempPath(), String.Format(CultureInfo.InvariantCulture, "{0}-CloudBackup-errors.txt", DateTime.Now.ToString("yyyy-MM-dd", CultureInfo.InvariantCulture))); if (File.Exists(szBackups)) { File.Delete(szBackups); } if (File.Exists(szBackupFailures)) { File.Delete(szBackupFailures); } List <EarnedGratuity> lstUsersWithCloudBackup = EarnedGratuity.GratuitiesForUser(string.Empty, Gratuity.GratuityTypes.CloudBackup); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersWithCloudBackup.RemoveAll(eg => eg.Username.CompareCurrentCultureIgnoreCase(UserRestriction) != 0); } foreach (EarnedGratuity eg in lstUsersWithCloudBackup) { StringBuilder sb = new StringBuilder(); StringBuilder sbFailures = new StringBuilder(); StorageID sid = StorageID.None; if (eg.UserProfile != null && ((sid = eg.UserProfile.BestCloudStorage) != StorageID.None) && eg.CurrentStatus != EarnedGratuity.EarnedGratuityStatus.Expired) { try { Profile pf = eg.UserProfile; LogbookBackup lb = new LogbookBackup(pf); EventRecorder.LogCall("Nightly run: backing up to {storage} for user {user}", sid.ToString(), eg.Username); switch (sid) { case StorageID.Dropbox: await BackupDropbox(lb, pf, sb, sbFailures).ConfigureAwait(false); break; case StorageID.OneDrive: await BackupOneDrive(lb, pf, sb, sbFailures).ConfigureAwait(false); break; case StorageID.GoogleDrive: await BackupGoogleDrive(lb, pf, sb, sbFailures).ConfigureAwait(false); break; default: break; } } catch (Exception ex) when(!(ex is OutOfMemoryException)) { string szError = String.Format(CultureInfo.CurrentCulture, "eg user={0}{1}\r\n\r\n{2}\r\n\r\n{3}", eg == null ? "NULL eg!" : eg.Username, (eg != null && eg.UserProfile == null) ? " NULL PROFILE" : string.Empty, sbFailures.ToString(), sb.ToString()); util.NotifyAdminException("ERROR running nightly backup", new MyFlightbookException(szError, ex)); } finally { lock (lockObj) { using (StreamWriter swSuccess = new StreamWriter(szBackups, true, Encoding.UTF8)) using (StreamWriter swFailure = new StreamWriter(szBackupFailures, true, Encoding.UTF8)) { // Regardless of what happens, write this to the file (in case something dies). string sz = sb.ToString(); if (!String.IsNullOrWhiteSpace(sz)) { swSuccess.WriteLine(sz.Trim()); } sz = sbFailures.ToString(); if (!String.IsNullOrWhiteSpace(sz)) { swFailure.WriteLine(sz.Trim()); } swSuccess.Flush(); swFailure.Flush(); } } } } } using (StreamReader srSuccess = new StreamReader(szBackups, Encoding.UTF8)) using (StreamReader srFailure = new StreamReader(szBackupFailures, Encoding.UTF8)) { util.NotifyAdminEvent("Cloud Backup report", srFailure.ReadToEnd() + srSuccess.ReadToEnd(), ProfileRoles.maskCanReport); } // if we are here, we should be able to delete the temp files try { File.Delete(szBackups); File.Delete(szBackupFailures); } finally { } return(true); }
/// <summary> /// Sends the nightly/monthly emails for users that have requested it. /// </summary> private void SendNightlyEmails() { // Find all users who qualify for expiring currency notifications - they may trigger an early email. List <EarnedGratuity> lstUsersWithExpiringCurrencies = EarnedGratuity.GratuitiesForUser(null, Gratuity.GratuityTypes.CurrencyAlerts); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersWithExpiringCurrencies.RemoveAll(eg => eg.Username.CompareCurrentCultureIgnoreCase(UserRestriction) != 0); } // And the list of people who have a subscription to expiring currencies List <Profile> lstUsersSubscribedForExpiration = new List <Profile>(Profile.UsersWithSubscriptions(EmailSubscription.FlagForType(SubscriptionType.Expiration), DateTime.Now.AddDays(-7))); // get the list of people who have a subscription OTHER than simple monthly or expiration List <Profile> lstUsersToSend = new List <Profile>(Profile.UsersWithSubscriptions(~(EmailSubscription.FlagForType(SubscriptionType.MonthlyTotals) | EmailSubscription.FlagForType(SubscriptionType.Expiration)), DateTime.Now.AddDays(-7))); if (!String.IsNullOrEmpty(UserRestriction)) { lstUsersToSend.RemoveAll(pf => pf.UserName.CompareOrdinalIgnoreCase(UserRestriction) != 0); } // See who has expiring currencies that require notification. foreach (EarnedGratuity eg in lstUsersWithExpiringCurrencies) { // Skip over anybody that's not also subscribed for expiration notices Profile pf = lstUsersSubscribedForExpiration.Find(profile => profile.UserName.CompareCurrentCultureIgnoreCase(eg.Username) == 0); if (pf == null) { continue; } IEnumerable <CurrencyStatusItem> expiringCurrencies = CurrencyStatusItem.CheckForExpiringCurrencies(eg.State, eg.Username, out string newState); if (newState.CompareOrdinal(eg.State) != 0) { eg.State = newState; eg.Commit(); } if (expiringCurrencies.Any()) { if (SendMailForUser(pf, Resources.Profile.EmailCurrencyExpiringMailSubject, string.Empty)) { // Don't send the weekly mail, since we just pre-empted it. lstUsersToSend.RemoveAll(p => pf.UserName.CompareCurrentCultureIgnoreCase(eg.Username) == 0); pf.LastEmailDate = DateTime.Now; pf.FCommit(); } } } foreach (Profile pf in lstUsersToSend) { if (SendMailForUser(pf, Resources.Profile.EmailWeeklyMailSubject, String.Empty)) { pf.LastEmailDate = DateTime.Now; pf.FCommit(); } } // Now do the monthly/annual emails if (DateTime.Now.Day == 1) { lstUsersToSend = new List <Profile>(Profile.UsersWithSubscriptions(EmailSubscription.FlagForType(SubscriptionType.MonthlyTotals), DateTime.Now.AddDays(1))); // We don't update the last-email sent on this because this email is asynchronous - i.e., not dependent on any other mail that was sent. foreach (Profile pf in lstUsersToSend) { SendMailForUser(pf, String.Format(CultureInfo.CurrentCulture, Resources.Profile.EmailMonthlySubject, DateTime.Now.AddMonths(-1).ToString("MMMM", CultureInfo.InvariantCulture)), "monthly"); } } // Do a pending flights reminder - every week or so if (DateTime.Now.DayOfYear % 7 == 6) { IDictionary <string, int> usersWithPendingFlights = PendingFlight.UsersWithLotsOfPendingFlights(30); foreach (string szUser in usersWithPendingFlights.Keys) { Profile pf = Profile.GetUser(szUser); util.NotifyUser(Branding.ReBrand(Resources.Profile.PendingFlightsReminderSubject), Branding.ReBrand(String.Format(CultureInfo.CurrentCulture, Resources.Profile.PendingFlightsReminder, pf.UserFullName, usersWithPendingFlights[szUser])), new System.Net.Mail.MailAddress(pf.Email, pf.UserFullName), false, true); } } }