logger/lib/loggers/Logger.js

  1. import process from "node:process";
  2. import {inspect} from "node:util";
  3. // Module name must not contain any other characters than alphanumerical and some specials
  4. const rIllegalModuleNameChars = /[^0-9a-zA-Z-_:@./]/i;
  5. /**
  6. * Standard logging module for UI5 Tooling and extensions.
  7. * <br><br>
  8. * Emits <code>ui5.log</code> events on the [<code>process</code>]{@link https://nodejs.org/api/process.html} object,
  9. * which can be handled by dedicated writers,
  10. * like [@ui5/logger/writers/Console]{@link @ui5/logger/writers/Console}.
  11. * <br><br>
  12. * If no listener is attached to an event, messages are written directly to the <code>process.stderr</code> stream.
  13. *
  14. * @public
  15. * @class
  16. * @alias @ui5/logger/Logger
  17. */
  18. class Logger {
  19. /**
  20. * Available log levels, ordered by priority:
  21. * <br>
  22. * <ol>
  23. * <li>silly</li>
  24. * <li>verbose</li>
  25. * <li>perf</li>
  26. * <li>info <i>(default)</i></li>
  27. * <li>warn</li>
  28. * <li>error</li>
  29. * <li>silent</li>
  30. * </ol>
  31. *
  32. * Log level <code>silent</code> is special in the sense that no messages can be submitted with that level.
  33. * It can be used to suppress all logging.
  34. *
  35. * @member {string[]}
  36. * @public
  37. */
  38. static LOG_LEVELS = ["silly", "verbose", "perf", "info", "warn", "error", "silent"];
  39. /**
  40. * Event name used for emitting new log-message event on the
  41. * [<code>process</code>]{@link https://nodejs.org/api/process.html} object
  42. *
  43. * @member {string}
  44. * @public
  45. */
  46. static LOG_EVENT_NAME = "ui5.log";
  47. /**
  48. * Sets the standard log level.
  49. * <br>
  50. * <b>Example:</b> Setting it to <code>perf</code> would suppress all <code>silly</code> and <code>verbose</code>
  51. * logging, and only show <code>perf</code>, <code>info</code>, <code>warn</code> and <code>error</code> logs.
  52. *
  53. * @public
  54. * @param {string} levelName New log level
  55. */
  56. static setLevel(levelName) {
  57. process.env.UI5_LOG_LVL = levelName;
  58. }
  59. /**
  60. * Gets the current log level
  61. *
  62. * @public
  63. * @returns {string} The current log level. Defaults to <code>info</code>
  64. */
  65. static getLevel() {
  66. if (process.env.UI5_LOG_LVL) {
  67. // Check whether set log level is valid
  68. const levelName = process.env.UI5_LOG_LVL;
  69. if (!Logger.LOG_LEVELS.includes(levelName)) {
  70. throw new Error(
  71. `UI5 Logger: Environment variable UI5_LOG_LVL is set to an unknown log level "${levelName}". ` +
  72. `Valid levels are ${Logger.LOG_LEVELS.join(", ")}`);
  73. }
  74. return levelName;
  75. } else {
  76. return "info";
  77. }
  78. }
  79. /**
  80. * Tests whether the provided log level is enabled by the current log level
  81. *
  82. * @public
  83. * @param {string} levelName Log level to test
  84. * @returns {boolean} True if the provided level is enabled
  85. */
  86. static isLevelEnabled(levelName) {
  87. const currIdx = Logger.LOG_LEVELS.indexOf(Logger.getLevel());
  88. const reqIdx = Logger.LOG_LEVELS.indexOf(levelName);
  89. if (reqIdx === -1) {
  90. throw new Error(`Unknown log level "${levelName}"`);
  91. }
  92. return reqIdx >= currIdx;
  93. }
  94. /**
  95. * Formats a given parameter into a string
  96. *
  97. * @param {any} message Single log message parameter passed by a program
  98. * @returns {string} String representation for the given message
  99. */
  100. static _formatMessage(message) {
  101. if (typeof message === "string" || message instanceof String) {
  102. return message;
  103. }
  104. return inspect(message, {
  105. depth: 3,
  106. compact: 2,
  107. });
  108. }
  109. #moduleName;
  110. /**
  111. *
  112. *
  113. * @public
  114. * @param {string} moduleName Identifier for messages created by this logger.
  115. * Example: <code>module:submodule:Class</code>
  116. */
  117. constructor(moduleName) {
  118. if (!moduleName) {
  119. throw new Error("Logger: Missing moduleName parameter");
  120. }
  121. if (rIllegalModuleNameChars.test(moduleName)) {
  122. throw new Error(`Logger: Invalid module name: ${moduleName}`);
  123. }
  124. this.#moduleName = moduleName;
  125. }
  126. /**
  127. * Tests whether the provided log level is enabled by the current log level
  128. *
  129. * @public
  130. * @param {string} levelName Log level to test
  131. * @returns {boolean} True if the provided level is enabled
  132. */
  133. isLevelEnabled(levelName) {
  134. return Logger.isLevelEnabled(levelName);
  135. }
  136. _emit(eventName, payload) {
  137. return process.emit(eventName, payload);
  138. }
  139. _log(level, message) {
  140. if (this.isLevelEnabled(level)) {
  141. process.stderr.write(`[${level}] ${message}\n`);
  142. }
  143. }
  144. _emitOrLog(level, message) {
  145. const hasListeners = this._emit(Logger.LOG_EVENT_NAME, {
  146. level,
  147. message,
  148. moduleName: this.#moduleName,
  149. });
  150. if (!hasListeners) {
  151. this._log(level, `${this.#moduleName}: ${message}`);
  152. }
  153. }
  154. }
  155. /**
  156. * Create a log entry with the <code>silly</code> level
  157. *
  158. * @public
  159. * @name @ui5/logger/Logger#silly
  160. * @function
  161. * @memberof @ui5/logger/Logger
  162. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  163. */
  164. /**
  165. * Create a log entry with the <code>verbose</code> level
  166. *
  167. * @public
  168. * @name @ui5/logger/Logger#verbose
  169. * @function
  170. * @memberof @ui5/logger/Logger
  171. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  172. */
  173. /**
  174. * Create a log entry with the <code>perf</code> level
  175. *
  176. * @public
  177. * @name @ui5/logger/Logger#perf
  178. * @function
  179. * @memberof @ui5/logger/Logger
  180. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  181. */
  182. /**
  183. * Create a log entry with the <code>info</code> level
  184. *
  185. * @public
  186. * @name @ui5/logger/Logger#info
  187. * @function
  188. * @memberof @ui5/logger/Logger
  189. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  190. */
  191. /**
  192. * Create a log entry with the <code>warn</code> level
  193. *
  194. * @public
  195. * @name @ui5/logger/Logger#warn
  196. * @function
  197. * @memberof @ui5/logger/Logger
  198. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  199. */
  200. /**
  201. * Create a log entry with the <code>error</code> level
  202. *
  203. * @public
  204. * @name @ui5/logger/Logger#error
  205. * @function
  206. * @memberof @ui5/logger/Logger
  207. * @param {...any} message Messages to log. An automatic string conversion is applied if necessary
  208. */
  209. Logger.LOG_LEVELS.forEach((logLevel) => {
  210. if (logLevel === "silent") {
  211. // This level is to suppress any logging. Hence we do not provide a dedicated log-function
  212. return;
  213. }
  214. Logger.prototype[logLevel] = function(...args) {
  215. const message = args.map(Logger._formatMessage).join(" ");
  216. this._emitOrLog(logLevel, message);
  217. };
  218. });
  219. export default Logger;