LogFileWebSocketListener.cs 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. using MediaBrowser.Common.IO;
  2. using MediaBrowser.Common.Net;
  3. using MediaBrowser.Controller.IO;
  4. using MediaBrowser.Model.Logging;
  5. using System;
  6. using System.Collections.Generic;
  7. using System.IO;
  8. using System.Linq;
  9. using System.Threading.Tasks;
  10. namespace MediaBrowser.Api.WebSocket
  11. {
  12. /// <summary>
  13. /// Class ScheduledTasksWebSocketListener
  14. /// </summary>
  15. public class LogFileWebSocketListener : BasePeriodicWebSocketListener<IEnumerable<string>, LogFileWebSocketState>
  16. {
  17. /// <summary>
  18. /// Gets the name.
  19. /// </summary>
  20. /// <value>The name.</value>
  21. protected override string Name
  22. {
  23. get { return "LogFile"; }
  24. }
  25. /// <summary>
  26. /// The _kernel
  27. /// </summary>
  28. private readonly ILogManager _logManager;
  29. private readonly IFileSystem _fileSystem;
  30. /// <summary>
  31. /// Initializes a new instance of the <see cref="LogFileWebSocketListener" /> class.
  32. /// </summary>
  33. /// <param name="logger">The logger.</param>
  34. /// <param name="logManager">The log manager.</param>
  35. public LogFileWebSocketListener(ILogger logger, ILogManager logManager, IFileSystem fileSystem)
  36. : base(logger)
  37. {
  38. _logManager = logManager;
  39. _fileSystem = fileSystem;
  40. _logManager.LoggerLoaded += kernel_LoggerLoaded;
  41. }
  42. /// <summary>
  43. /// Gets the data to send.
  44. /// </summary>
  45. /// <param name="state">The state.</param>
  46. /// <returns>IEnumerable{System.String}.</returns>
  47. protected override async Task<IEnumerable<string>> GetDataToSend(LogFileWebSocketState state)
  48. {
  49. if (!string.Equals(_logManager.LogFilePath, state.LastLogFilePath))
  50. {
  51. state.LastLogFilePath = _logManager.LogFilePath;
  52. state.StartLine = 0;
  53. }
  54. var lines = await GetLogLines(state.LastLogFilePath, state.StartLine, _fileSystem).ConfigureAwait(false);
  55. state.StartLine += lines.Count;
  56. return lines;
  57. }
  58. /// <summary>
  59. /// Releases unmanaged and - optionally - managed resources.
  60. /// </summary>
  61. /// <param name="dispose"><c>true</c> to release both managed and unmanaged resources; <c>false</c> to release only unmanaged resources.</param>
  62. protected override void Dispose(bool dispose)
  63. {
  64. if (dispose)
  65. {
  66. _logManager.LoggerLoaded -= kernel_LoggerLoaded;
  67. }
  68. base.Dispose(dispose);
  69. }
  70. /// <summary>
  71. /// Handles the LoggerLoaded event of the kernel control.
  72. /// </summary>
  73. /// <param name="sender">The source of the event.</param>
  74. /// <param name="e">The <see cref="EventArgs" /> instance containing the event data.</param>
  75. void kernel_LoggerLoaded(object sender, EventArgs e)
  76. {
  77. // Reset the startline for each connection whenever the logger reloads
  78. lock (ActiveConnections)
  79. {
  80. foreach (var connection in ActiveConnections)
  81. {
  82. connection.Item4.StartLine = 0;
  83. }
  84. }
  85. }
  86. /// <summary>
  87. /// Gets the log lines.
  88. /// </summary>
  89. /// <param name="logFilePath">The log file path.</param>
  90. /// <param name="startLine">The start line.</param>
  91. /// <returns>Task{IEnumerable{System.String}}.</returns>
  92. internal static async Task<List<string>> GetLogLines(string logFilePath, int startLine, IFileSystem fileSystem)
  93. {
  94. var lines = new List<string>();
  95. using (var fs = fileSystem.GetFileStream(logFilePath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite, true))
  96. {
  97. using (var reader = new StreamReader(fs))
  98. {
  99. while (!reader.EndOfStream)
  100. {
  101. var line = await reader.ReadLineAsync().ConfigureAwait(false);
  102. if (line.IndexOf("Info", StringComparison.OrdinalIgnoreCase) != -1 ||
  103. line.IndexOf("Warn", StringComparison.OrdinalIgnoreCase) != -1 ||
  104. line.IndexOf("Error", StringComparison.OrdinalIgnoreCase) != -1)
  105. {
  106. lines.Add(line);
  107. }
  108. }
  109. }
  110. }
  111. if (startLine > 0)
  112. {
  113. lines = lines.Skip(startLine).ToList();
  114. }
  115. return lines;
  116. }
  117. }
  118. /// <summary>
  119. /// Class LogFileWebSocketState
  120. /// </summary>
  121. public class LogFileWebSocketState
  122. {
  123. /// <summary>
  124. /// Gets or sets the last log file path.
  125. /// </summary>
  126. /// <value>The last log file path.</value>
  127. public string LastLogFilePath { get; set; }
  128. /// <summary>
  129. /// Gets or sets the start line.
  130. /// </summary>
  131. /// <value>The start line.</value>
  132. public int StartLine { get; set; }
  133. }
  134. }