/NetOffice/NamedPipes/DebugConsole.cs

# · C# · 372 lines · 229 code · 39 blank · 104 comment · 34 complexity · dc05cab044fb7f06980518a3a0a6290a MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.ComponentModel;
  5. using NetOffice.NamedPipes;
  6. namespace NetOffice
  7. {
  8. /// <summary>
  9. /// Operation mode for DebugConsole
  10. /// </summary>
  11. public enum ConsoleMode
  12. {
  13. /// <summary>
  14. /// debug log are not enabled
  15. /// </summary>
  16. None = 0,
  17. /// <summary>
  18. /// debug log was redirected to System.Console
  19. /// </summary>
  20. Console = 1,
  21. /// <summary>
  22. /// debug log append to a logfile
  23. /// </summary>
  24. LogFile = 2,
  25. /// <summary>
  26. /// hold all debug and exceptions logs in a internal string list
  27. /// </summary>
  28. MemoryList = 3,
  29. /// <summary>
  30. /// debug log was redirected to System.Diagnostics.Trace
  31. /// </summary>
  32. Trace = 4
  33. }
  34. /// <summary>
  35. /// Shared output connection technique
  36. /// </summary>
  37. public enum SharedOutputMode
  38. {
  39. /// <summary>
  40. /// IPC named pipes
  41. /// </summary>
  42. LocalNamedPipes = 0
  43. }
  44. /// <summary>
  45. /// offers various debug, log and diagnostic functionality
  46. /// </summary>
  47. public static class DebugConsole
  48. {
  49. private static object _sharedLock = new object();
  50. private static List<string> _messageList = new List<string>();
  51. /// <summary>
  52. /// append current time information in messages
  53. /// </summary>
  54. public static bool AppendTimeInfoEnabled { get; set; }
  55. /// <summary>
  56. /// operation mode
  57. /// </summary>
  58. public static ConsoleMode Mode { get; set; }
  59. /// <summary>
  60. /// send a all messages to a named pipe. Use the NOTools.ConsoleMonitor to observe the console
  61. /// </summary>
  62. public static bool EnableSharedOutput { get; set; }
  63. /// <summary>
  64. /// Specify the shared output connection technique (currently ipc named pipes only. for future use to enable network and db logging)
  65. /// </summary>
  66. public static SharedOutputMode SharedOutputMode { get; set; }
  67. /// <summary>
  68. /// PipeConnection to NOTools.ConsoleMonitor
  69. /// </summary>
  70. private static PipeClient Pipe { get; set; }
  71. /// <summary>
  72. /// name full file path and name of a logfile, must be set if Mode == LogFile
  73. /// </summary>
  74. public static string FileName { get; set; }
  75. /// <summary>
  76. /// returns all collected messages if Mode == MemoryList
  77. /// </summary>
  78. public static string[] Messages { get { return _messageList.ToArray(); } }
  79. /// <summary>
  80. /// clears message buffer
  81. /// </summary>
  82. public static void ClearMessagesList()
  83. {
  84. _messageList.Clear();
  85. }
  86. /// <summary>
  87. /// write log message
  88. /// </summary>
  89. /// <param name="message"></param>
  90. /// <param name="args"></param>
  91. public static void WriteLine(string message, params object[] args)
  92. {
  93. string output = message;
  94. int i = 0;
  95. foreach (object item in args)
  96. {
  97. string replaceValue = "";
  98. if (null != item)
  99. replaceValue = item.ToString();
  100. output = output.Replace("{" + i.ToString() + "}", replaceValue);
  101. i++;
  102. }
  103. if (ConsoleMode.Console == Mode || ConsoleMode.Trace == Mode)
  104. output = "NetOffice: " + output;
  105. if (AppendTimeInfoEnabled)
  106. output = DateTime.Now.ToLongTimeString() + " - " + message;
  107. switch (Mode)
  108. {
  109. case ConsoleMode.Console:
  110. Console.WriteLine(output);
  111. break;
  112. case ConsoleMode.Trace:
  113. System.Diagnostics.Trace.WriteLine(output);
  114. break;
  115. case ConsoleMode.LogFile:
  116. AppendToLogFile(output);
  117. break;
  118. case ConsoleMode.MemoryList:
  119. _messageList.Add(output);
  120. break;
  121. case ConsoleMode.None:
  122. // do nothing
  123. break;
  124. default:
  125. throw new ArgumentOutOfRangeException("Unkown Log Mode.");
  126. }
  127. InternalSendNamedPipeMessage(output, null);
  128. }
  129. /// <summary>
  130. /// write log message
  131. /// </summary>
  132. /// <param name="message"></param>
  133. public static void WriteLine(string message)
  134. {
  135. string output = message;
  136. if (ConsoleMode.Console == Mode || ConsoleMode.Trace == Mode)
  137. output = "NetOffice: " + output;
  138. if (AppendTimeInfoEnabled)
  139. output = DateTime.Now.ToLongTimeString() + " - " + message;
  140. switch (Mode)
  141. {
  142. case ConsoleMode.Console:
  143. Console.WriteLine(output);
  144. break;
  145. case ConsoleMode.Trace:
  146. System.Diagnostics.Trace.WriteLine(output);
  147. break;
  148. case ConsoleMode.LogFile:
  149. AppendToLogFile(output);
  150. break;
  151. case ConsoleMode.MemoryList:
  152. _messageList.Add(output);
  153. break;
  154. case ConsoleMode.None:
  155. // do nothing
  156. break;
  157. default:
  158. throw new ArgumentOutOfRangeException("Unkown Log Mode.");
  159. }
  160. InternalSendNamedPipeMessage(output, null);
  161. }
  162. /// <summary>
  163. /// write exception log message
  164. /// </summary>
  165. /// <param name="exception"></param>
  166. public static void WriteException(Exception exception)
  167. {
  168. string message = CreateExecptionLog(exception);
  169. WriteLine(message);
  170. }
  171. /// <summary>
  172. /// Send a message to the NOTools.Console monitor pipe
  173. /// </summary>
  174. /// <param name="console">name for the console(must exclude the '?' char) or null for default console</param>
  175. /// <param name="message">the given message as any</param>
  176. /// <returns>entry id for the log message if arrived, otherwise null</returns>
  177. public static string SendPipeConsoleMessage(string console, string message)
  178. {
  179. try
  180. {
  181. lock (_sharedLock)
  182. {
  183. if (null == Pipe)
  184. Pipe = new PipeClient();
  185. return Pipe.SendConsoleMessage(console, message, null);
  186. }
  187. }
  188. catch (Exception exception)
  189. {
  190. EnableSharedOutput = false;
  191. WriteException(exception);
  192. return null;
  193. }
  194. }
  195. /// <summary>
  196. /// Send a message to the NOTools.Console monitor pipe
  197. /// </summary>
  198. /// <param name="console">name for the console(must exclude the '?' char) or null for default console</param>
  199. /// <param name="message">the given message as any</param>
  200. /// <param name="parentEntryID">parent message id. the console monitor can show a hierarchy with these info</param>
  201. /// <returns>entry id for the log message if arrived, otherwise null</returns>
  202. public static string SendPipeConsoleMessage(string console, string message, string parentEntryID)
  203. {
  204. try
  205. {
  206. lock (_sharedLock)
  207. {
  208. if (null == Pipe)
  209. Pipe = new PipeClient();
  210. return Pipe.SendConsoleMessage(console, message, parentEntryID);
  211. }
  212. }
  213. catch (Exception exception)
  214. {
  215. EnableSharedOutput = false;
  216. WriteException(exception);
  217. return null;
  218. }
  219. }
  220. /// <summary>
  221. /// Send a channel message to the NOTools.Console monitor pipe
  222. /// </summary>
  223. /// <param name="channel">channel id string. the argument must exclude the '?' character</param>
  224. /// <param name="message">the given message as any</param>
  225. /// <returns>entry id for the log message if arrived, otherwise null</returns>
  226. public static string SendPipeChannelMessage(string channel, string message)
  227. {
  228. try
  229. {
  230. lock (_sharedLock)
  231. {
  232. if (null == Pipe)
  233. Pipe = new PipeClient();
  234. return Pipe.SendChannelMessage(channel, message);
  235. }
  236. }
  237. catch (Exception exception)
  238. {
  239. EnableSharedOutput = false;
  240. WriteException(exception);
  241. return null;
  242. }
  243. }
  244. /// Send a message to the NOTools.Console monitor pipe
  245. /// </summary>
  246. /// <param name="message">the given message as any</param>
  247. /// <returns>true if send</returns>
  248. internal static string InternalSendNamedPipeMessage(string message, string parentEntryID)
  249. {
  250. try
  251. {
  252. if (!EnableSharedOutput)
  253. return null;
  254. lock (_sharedLock)
  255. {
  256. if (null == Pipe)
  257. Pipe = new PipeClient();
  258. return Pipe.SendConsoleMessage(null, message, parentEntryID);
  259. }
  260. }
  261. catch (Exception exception)
  262. {
  263. EnableSharedOutput = false;
  264. WriteException(exception);
  265. return null;
  266. }
  267. }
  268. /// <summary>
  269. /// Send a channel message to the NOTools.Console monitor pipe
  270. /// </summary>
  271. /// <param name="channel">channel id string. the argument must exclude the '?' character</param>
  272. /// <param name="message">the given message as any</param>
  273. /// <returns>true if send</returns>
  274. internal static string InternalSendNamedPipeChannelMessage(string channel, string message)
  275. {
  276. try
  277. {
  278. if (!EnableSharedOutput)
  279. return null;
  280. lock (_sharedLock)
  281. {
  282. if (null == Pipe)
  283. Pipe = new PipeClient();
  284. return Pipe.SendChannelMessage(channel, message);
  285. }
  286. }
  287. catch (Exception exception)
  288. {
  289. EnableSharedOutput = false;
  290. WriteException(exception);
  291. return null;
  292. }
  293. }
  294. /// <summary>
  295. /// append message to logfile
  296. /// </summary>
  297. /// <param name="message"></param>
  298. private static void AppendToLogFile(string message)
  299. {
  300. if (null == FileName)
  301. throw new NetOfficeException("FileName not set.");
  302. System.IO.File.AppendAllText(FileName, message + Environment.NewLine, Encoding.UTF8);
  303. }
  304. /// <summary>
  305. /// convert an exception to a string
  306. /// </summary>
  307. /// <param name="exception"></param>
  308. /// <returns></returns>
  309. private static string CreateExecptionLog(Exception exception)
  310. {
  311. string result = "";
  312. Exception ex = exception;
  313. while (ex != null)
  314. {
  315. string type = ex.GetType().Name;
  316. string message = ex.Message;
  317. string target = "<Empty>";
  318. if (null != ex.TargetSite)
  319. target = ex.TargetSite.ToString();
  320. string trace = "<Empty>";
  321. if (null != ex.StackTrace)
  322. trace = ex.StackTrace.ToString();
  323. result += "Type:" + type + Environment.NewLine;
  324. result += "Message:" + message + Environment.NewLine;
  325. result += "Target:" + target + Environment.NewLine;
  326. result += "Stack:" + trace + Environment.NewLine;
  327. result += Environment.NewLine;
  328. ex = ex.InnerException;
  329. }
  330. return result;
  331. }
  332. }
  333. }