You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			254 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			254 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			TypeScript
		
	
type Logger = typeof window.log | typeof console;
 | 
						|
type LogLevel = keyof Logger;
 | 
						|
 | 
						|
/**
 | 
						|
 * Create a new TimedLog instance. A logging method can be called later to log a message with an elapsed time.
 | 
						|
 *
 | 
						|
 * When an instance of this class is created it will save the current time in itself and use that time to compute the elapsed time when a logging method is called on it.
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = new TimedLog();
 | 
						|
 * timedLog.debug('A message was sent with id 7');
 | 
						|
 * // Output: A message was sent with id 7: 1.923s
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = new TimedLog();
 | 
						|
 * timedLog.debug('A message was sent after {time} with id 7');
 | 
						|
 * // Output: A message was sent after 1.923s with id 7
 | 
						|
 */
 | 
						|
export class TimedLog {
 | 
						|
  private start: number = Date.now();
 | 
						|
  private logger: Logger;
 | 
						|
 | 
						|
  private static timeAppendPrefix = ':';
 | 
						|
  private static millisecondSuffix = 'ms';
 | 
						|
  private static secondSuffix = 's';
 | 
						|
 | 
						|
  constructor(initialLogMessage?: string, initialLogMessageLevel?: keyof Logger) {
 | 
						|
    if (typeof window === 'undefined') {
 | 
						|
      this.logger = console;
 | 
						|
    } else {
 | 
						|
      this.logger = window.log;
 | 
						|
    }
 | 
						|
    if (initialLogMessage) {
 | 
						|
      this.log(initialLogMessageLevel ?? 'debug', initialLogMessage);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Reset the timer to the current time.
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = new TimedLog();
 | 
						|
   * timedLog.debug('A message was sent with id 7');
 | 
						|
   * // Output: A message was sent with id 7: 1.923s
 | 
						|
   * timedLog.resetTimer();
 | 
						|
   * timedLog.debug('A message was sent with id 8');
 | 
						|
   * // Output: A message was sent with id 8: 2.318s
 | 
						|
   */
 | 
						|
  public resetTimer() {
 | 
						|
    this.start = Date.now();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Format the time elapsed since the start of the timer.
 | 
						|
   * @param time The time to format.
 | 
						|
   * @returns The formatted time.
 | 
						|
   */
 | 
						|
  public static formatDistanceToNow(time: number) {
 | 
						|
    const ms = Date.now() - Math.floor(time);
 | 
						|
    const s = Math.floor(ms / 1000);
 | 
						|
    if (s === 0) {
 | 
						|
      return `${ms}${TimedLog.millisecondSuffix}`;
 | 
						|
    }
 | 
						|
 | 
						|
    if (ms === 0) {
 | 
						|
      return `${s}${TimedLog.secondSuffix}`;
 | 
						|
    }
 | 
						|
 | 
						|
    function formatMillisecondsToSeconds(milliseconds: number): string {
 | 
						|
      const seconds = milliseconds / 1000;
 | 
						|
      return seconds.toFixed(3).replace(/\.?0+$/, '');
 | 
						|
    }
 | 
						|
 | 
						|
    return `${formatMillisecondsToSeconds(ms)}${TimedLog.secondSuffix}`;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Format a message with the time elapsed since the start of the timer.
 | 
						|
   * If the message contains a placeholder {*}, the placeholder will be replaced with the time passed.
 | 
						|
   * Otherwise the time passed will be added to the end of the message, separated by a separator or ': ' by default.
 | 
						|
   *
 | 
						|
   * @param data The message to replace the time in.
 | 
						|
   * @returns The message with the time replaced.
 | 
						|
   */
 | 
						|
  private writeTimeToLog(...data: Array<any>) {
 | 
						|
    const time = TimedLog.formatDistanceToNow(this.start);
 | 
						|
 | 
						|
    const includesTemplate = data.some(arg => typeof arg === 'string' && /\{.*\}/.test(arg));
 | 
						|
 | 
						|
    if (!includesTemplate) {
 | 
						|
      return [...data, TimedLog.timeAppendPrefix, time];
 | 
						|
    }
 | 
						|
 | 
						|
    return data.map(arg =>
 | 
						|
      typeof arg === 'string' && /\{.*\}/.test(arg) ? arg.replace(/\{.*\}/g, time) : arg
 | 
						|
    );
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Log a message at the given level.
 | 
						|
   *
 | 
						|
   * @param level The level to log at.
 | 
						|
   * @param message The message to log.
 | 
						|
   */
 | 
						|
  private log(level: LogLevel, ...data: Array<any>) {
 | 
						|
    this.logger[level](...this.writeTimeToLog(...data));
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Log a message at the debug level with the elapsed time.
 | 
						|
   *
 | 
						|
   * If the message contains a placeholder {*}, the placeholder will be replaced with the time passed.
 | 
						|
   * Otherwise the time passed will be added to the end of the message.
 | 
						|
   *
 | 
						|
   * @param message The message to log at the debug level.
 | 
						|
   *
 | 
						|
   * @see {@link initTimedLog} to create a new TimedLog instance.
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.debug('A message was sent with id 7');
 | 
						|
   * // Output: A message was sent with id 7: 1.923s
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.debug('A message was sent after {time} with id 7');
 | 
						|
   * // Output: A message was sent after 1.923s with id 7
 | 
						|
   */
 | 
						|
  public debug(...data: Array<any>) {
 | 
						|
    this.log('debug', ...data);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Log a message at the info level with the elapsed time.
 | 
						|
   *
 | 
						|
   * If the message contains a placeholder {*}, the placeholder will be replaced with the time passed.
 | 
						|
   * Otherwise the time passed will be added to the end of the message.
 | 
						|
   *
 | 
						|
   * @param message The message to log at the debug level.
 | 
						|
   *
 | 
						|
   * @see {@link initTimedLog} to create a new TimedLog instance.
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.info('A message was sent with id 7');
 | 
						|
   * // Output: A message was sent with id 7: 1.923s
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.info('A message was sent after {time} with id 7');
 | 
						|
   * // Output: A message was sent after 1.923s with id 7
 | 
						|
   */
 | 
						|
  public info(...data: Array<any>) {
 | 
						|
    this.log('info', ...data);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Log a message at the warn level with the elapsed time.
 | 
						|
   *
 | 
						|
   * If the message contains a placeholder {*}, the placeholder will be replaced with the time passed.
 | 
						|
   * Otherwise the time passed will be added to the end of the message.
 | 
						|
   *
 | 
						|
   * @param message The message to log at the debug level.
 | 
						|
   *
 | 
						|
   * @see {@link initTimedLog} to create a new TimedLog instance.
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.warn('A message was sent with id 7');
 | 
						|
   * // Output: A message was sent with id 7: 1.923s
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.warn('A message was sent after {time} with id 7');
 | 
						|
   * // Output: A message was sent after 1.923s with id 7
 | 
						|
   */
 | 
						|
  public warn(...data: Array<any>) {
 | 
						|
    this.log('warn', ...data);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Log a message at the error level with the elapsed time.
 | 
						|
   *
 | 
						|
   * If the message contains a placeholder {*}, the placeholder will be replaced with the time passed.
 | 
						|
   * Otherwise the time passed will be added to the end of the message.
 | 
						|
   *
 | 
						|
   * @param message The message to log at the debug level.
 | 
						|
   *
 | 
						|
   * @see {@link initTimedLog} to create a new TimedLog instance.
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.error('A message was sent with id 7');
 | 
						|
   * // Output: A message was sent with id 7: 1.923s
 | 
						|
   *
 | 
						|
   * @example
 | 
						|
   * const timedLog = initTimedLog();
 | 
						|
   * timedLog.error('A message was sent after {time} with id 7');
 | 
						|
   * // Output: A message was sent after 1.923s with id 7
 | 
						|
   */
 | 
						|
  public error(...data: Array<any>) {
 | 
						|
    this.log('error', ...data);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
export type TimedLogInstance = TimedLog;
 | 
						|
 | 
						|
/**
 | 
						|
 * Create a new TimedLog instance. This can be called later to log a message with an elapsed time.
 | 
						|
 * @returns A new TimedLog instance.
 | 
						|
 *
 | 
						|
 * @see {@link TimedLog} for more information on how to use the returned instance.
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = TimedLog();
 | 
						|
 * timer.debug('A message was sent with id 7');
 | 
						|
 * // Output: A message was sent with id 7: 1.923s
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = TimedLog();
 | 
						|
 * timer.info('A message was sent after {time} with id 7');
 | 
						|
 * // Output: A message was sent after 1.923s with id 7
 | 
						|
 */
 | 
						|
export const initTimedLog = () => {
 | 
						|
  return new TimedLog();
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Create a new TimedLog instance. This can be called later to log a message with an elapsed time.
 | 
						|
 * @param initialLogMessage The message to log when the instance is created.
 | 
						|
 * @param initialLogMessageLevel The level to log the initial message at.
 | 
						|
 * @returns A new TimedLog instance.
 | 
						|
 *
 | 
						|
 * @see {@link TimedLog} for more information on how to use the returned instance.
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = TimedLog('A message is being sent with id 7');
 | 
						|
 * // Output: A message is being sent with id 7
 | 
						|
 * timer.debug('A message was sent with id 7');
 | 
						|
 * // Output: A message was sent with id 7: 1.923s
 | 
						|
 *
 | 
						|
 * @example
 | 
						|
 * const timedLog = TimedLog('A message is being sent with id 7', 'info');
 | 
						|
 * // Output: A message is being sent with id 7
 | 
						|
 * timer.info('A message was sent after {time} with id 7');
 | 
						|
 * // Output: A message was sent after 1.923s with id 7
 | 
						|
 */
 | 
						|
export const initTimedLogWithInitialLog = (
 | 
						|
  initialLogMessage: string,
 | 
						|
  initialLogMessageLevel: keyof Logger = 'debug'
 | 
						|
) => {
 | 
						|
  return new TimedLog(initialLogMessage, initialLogMessageLevel);
 | 
						|
};
 |