serverjs/logger.js

/**
* @module
*/

'use strict';

const winston = require('winston');

/** 
* находится ли программа в режиме релиза
* @type {boolean}
*/
const inProd = process.env.PROD;

/**
* Уровни логгера.
* @type {Object}
*/
let levels = {error: 0, warn: 1, notice: 2, info: 3, debug: 4, silly: 5};

/**
* Цвета уровней логгера.
* @type {Object}
*/
let colors = {error: 'red', warn: 'yellow', notice: 'cyan', info: 'green', debug: 'magenta', silly: 'blue'};

/**
* Добавляет нули перед числом до указанного кол-ва.
* @param  {number} num     число
* @param  {number} [len=2] желаемая длина строки длина
* @return {string}         Число в виде строки с желаемой длиной.
*/
function leadWithZeros(num, len){
	len = len || 2;
	return (Array(len).join("0") + num).slice(-len);
}

/**
* Форматирует текущее время.
* @param  {boolean} [full] нужно ли добавляеть дату или только время
* @return {string}      	Дата в формате `[DD-MM-YYt]HH:MM:SS:MMMM`.
*/
function getTimeStamp(full){
	const d = new Date();
	let date = leadWithZeros(d.getHours()) + ':' + leadWithZeros(d.getMinutes()) + ':' +
			   leadWithZeros(d.getSeconds()) + ':' + leadWithZeros(d.getMilliseconds(), 3);
	if(full){
		date = leadWithZeros(d.getDate()) + leadWithZeros(d.getMonth()) + leadWithZeros(d.getYear()) + 't' + date;
	}
	return date;
}

/**
* Создает новый winston логгер.  
* У логгера 6 уровней и функций для вывода в консоль: `error, warn, notice, info, debug, silly`.  
* В консоль будет выводиться переданный уровень, в файл выводятся уровни до и включая debug.
* @param  {object} callingModule    `module` объект модуля, вызывающего эту функцию
* @param  {string} [id]             id нового логгера
* @param  {string} [level='notice'] уровень сообщений консоли
* @return {winston.Logger}       	 Новый логгер.
*/
module.exports = function(callingModule, id, level) {
	let name = callingModule.filename.split(/[/\\]+/).pop().replace('.js', '');
	let filename = './logs/' + name + (id ? '#' + id : '') + '-' + (getTimeStamp(true).replace(/[|&;$%@"<>()+,\/:-]/g,'')) + '.log';
	let transports = [
		new winston.transports.Console(
		{	
			level: level || 'notice',
			label: name + (id ? '#' + id : ''),
			prettyPrint: true,
			colorize: true,
			timestamp: getTimeStamp
		})
	];
	if(!inProd){
		transports.push(new winston.transports.File(
		{
			level: 'debug',
			filename: filename,
			json: false, 
			timestamp: getTimeStamp
		}));
	}
	return new winston.Logger({
		levels: levels,
		colors: colors,
		transports: transports
	});
};