LocalMessageSet.cs 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. using LiteDB;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. namespace NTMiner.Core.Impl {
  6. public class LocalMessageSet : SetBase, ILocalMessageSet {
  7. private readonly LinkedList<ILocalMessage> _records = new LinkedList<ILocalMessage>();
  8. private readonly List<Guid> _dbToRemoveIds = new List<Guid>();
  9. private readonly List<LocalMessageData> _dbToInserts = new List<LocalMessageData>();
  10. private string ConnString {
  11. get {
  12. return $"filename={HomePath.LocalDbFileFullName}";
  13. }
  14. }
  15. public ILocalMessageDtoSet LocalMessageDtoSet { get; private set; } = EmptyLocalMessageDtoSet.Instance;
  16. public LocalMessageSet() {
  17. if (ClientAppType.IsMinerClient) {
  18. LocalMessageDtoSet = new LocalMessageDtoSet();
  19. }
  20. VirtualRoot.BuildCmdPath<AddLocalMessageCommand>(location: this.GetType(), LogEnum.DevConsole, path: message => {
  21. InitOnece();
  22. var data = LocalMessageData.Create(message.Input);
  23. List<ILocalMessage> removeds = new List<ILocalMessage>();
  24. lock (_dbToInserts) {
  25. _records.AddFirst(data);
  26. _dbToInserts.Add(data);
  27. while (_records.Count > NTKeyword.LocalMessageSetCapacity) {
  28. var toRemove = _records.Last.Value;
  29. removeds.Add(toRemove);
  30. _records.RemoveLast();
  31. }
  32. _dbToRemoveIds.AddRange(removeds.Select(a => a.Id));
  33. }
  34. LocalMessageDtoSet.Add(data.ToDto());
  35. VirtualRoot.RaiseEvent(new LocalMessageAddedEvent(message.MessageId, data, removeds));
  36. });
  37. VirtualRoot.BuildCmdPath<ClearLocalMessageSetCommand>(location: this.GetType(), LogEnum.DevConsole, path: message => {
  38. lock (_dbToInserts) {
  39. _records.Clear();
  40. _dbToRemoveIds.Clear();
  41. _dbToInserts.Clear();
  42. }
  43. try {
  44. using (LiteDatabase db = new LiteDatabase(ConnString)) {
  45. db.DropCollection(nameof(LocalMessageData));
  46. }
  47. }
  48. catch (Exception e) {
  49. Logger.ErrorDebugLine(e);
  50. }
  51. VirtualRoot.RaiseEvent(new LocalMessageSetClearedEvent());
  52. });
  53. VirtualRoot.BuildEventPath<Per1MinuteEvent>("周期保存LocalMessage到数据库", LogEnum.DevConsole, this.GetType(), PathPriority.Normal, path: message => {
  54. SaveToDb();
  55. });
  56. VirtualRoot.BuildEventPath<AppExitEvent>("程序退出时保存LocalMessage到数据库", LogEnum.DevConsole, this.GetType(), PathPriority.Normal, path: message => {
  57. SaveToDb();
  58. });
  59. }
  60. private void SaveToDb() {
  61. if (_dbToInserts.Count > 0) {
  62. lock (_dbToInserts) {
  63. if (_dbToInserts.Count > 0) {
  64. List<Guid> toRemoveIds = new List<Guid>();
  65. List<LocalMessageData> toInserts = new List<LocalMessageData>();
  66. toRemoveIds.AddRange(_dbToRemoveIds);
  67. toInserts.AddRange(_dbToInserts);
  68. _dbToRemoveIds.Clear();
  69. _dbToInserts.Clear();
  70. try {
  71. using (LiteDatabase db = new LiteDatabase(ConnString)) {
  72. var col = db.GetCollection<LocalMessageData>();
  73. foreach (Guid id in toRemoveIds) {
  74. col.Delete(id);
  75. }
  76. if (toInserts.Count != 0) {
  77. col.InsertBulk(toInserts);
  78. }
  79. }
  80. }
  81. catch {
  82. }
  83. }
  84. }
  85. }
  86. }
  87. protected override void Init() {
  88. try {
  89. using (LiteDatabase db = new LiteDatabase(ConnString)) {
  90. var col = db.GetCollection<LocalMessageData>();
  91. foreach (var item in col.FindAll().OrderBy(a => a.Timestamp)) {
  92. if (_records.Count < NTKeyword.LocalMessageSetCapacity) {
  93. _records.AddFirst(item);
  94. }
  95. else {
  96. col.Delete(item.Id);
  97. }
  98. }
  99. }
  100. }
  101. catch (Exception e) {
  102. Logger.ErrorDebugLine(e);
  103. try {
  104. using (LiteDatabase db = new LiteDatabase(ConnString)) {
  105. db.DropCollection(nameof(LocalMessageData));
  106. }
  107. }
  108. catch {
  109. }
  110. }
  111. foreach (var item in _records.Take(50).Reverse()) {
  112. LocalMessageDtoSet.Add(item.ToDto());
  113. }
  114. }
  115. public IEnumerable<ILocalMessage> AsEnumerable() {
  116. InitOnece();
  117. return _records;
  118. }
  119. }
  120. }