123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310 |
- const levels = require("./levels");
- const DEFAULT_FORMAT =
- ":remote-addr - -" +
- ' ":method :url HTTP/:http-version"' +
- ' :status :content-length ":referrer"' +
- ' ":user-agent"';
- function getUrl(req) {
- return req.originalUrl || req.url;
- }
- function assembleTokens(req, res, customTokens) {
- const arrayUniqueTokens = array => {
- const a = array.concat();
- for (let i = 0; i < a.length; ++i) {
- for (let j = i + 1; j < a.length; ++j) {
-
-
- if (a[i].token == a[j].token) {
- a.splice(j--, 1);
- }
- }
- }
- return a;
- };
- const defaultTokens = [];
- defaultTokens.push({ token: ":url", replacement: getUrl(req) });
- defaultTokens.push({ token: ":protocol", replacement: req.protocol });
- defaultTokens.push({ token: ":hostname", replacement: req.hostname });
- defaultTokens.push({ token: ":method", replacement: req.method });
- defaultTokens.push({
- token: ":status",
- replacement: res.__statusCode || res.statusCode
- });
- defaultTokens.push({
- token: ":response-time",
- replacement: res.responseTime
- });
- defaultTokens.push({ token: ":date", replacement: new Date().toUTCString() });
- defaultTokens.push({
- token: ":referrer",
- replacement: req.headers.referer || req.headers.referrer || ""
- });
- defaultTokens.push({
- token: ":http-version",
- replacement: `${req.httpVersionMajor}.${req.httpVersionMinor}`
- });
- defaultTokens.push({
- token: ":remote-addr",
- replacement:
- req.headers["x-forwarded-for"] ||
- req.ip ||
- req._remoteAddress ||
- (req.socket &&
- (req.socket.remoteAddress ||
- (req.socket.socket && req.socket.socket.remoteAddress)))
- });
- defaultTokens.push({
- token: ":user-agent",
- replacement: req.headers["user-agent"]
- });
- defaultTokens.push({
- token: ":content-length",
- replacement:
- res.getHeader("content-length") ||
- (res.__headers && res.__headers["Content-Length"]) ||
- "-"
- });
- defaultTokens.push({
- token: /:req\[([^\]]+)]/g,
- replacement(_, field) {
- return req.headers[field.toLowerCase()];
- }
- });
- defaultTokens.push({
- token: /:res\[([^\]]+)]/g,
- replacement(_, field) {
- return (
- res.getHeader(field.toLowerCase()) ||
- (res.__headers && res.__headers[field])
- );
- }
- });
- return arrayUniqueTokens(customTokens.concat(defaultTokens));
- }
- function format(str, tokens) {
- for (let i = 0; i < tokens.length; i++) {
- str = str.replace(tokens[i].token, tokens[i].replacement);
- }
- return str;
- }
- function createNoLogCondition(nolog) {
- let regexp = null;
- if (nolog instanceof RegExp) {
- regexp = nolog;
- }
- if (typeof nolog === "string") {
- regexp = new RegExp(nolog);
- }
- if (Array.isArray(nolog)) {
-
- const regexpsAsStrings = nolog.map(reg => (reg.source ? reg.source : reg));
- regexp = new RegExp(regexpsAsStrings.join("|"));
- }
- return regexp;
- }
- function matchRules(statusCode, currentLevel, ruleSet) {
- let level = currentLevel;
- if (ruleSet) {
- const matchedRule = ruleSet.find(rule => {
- let ruleMatched = false;
- if (rule.from && rule.to) {
- ruleMatched = statusCode >= rule.from && statusCode <= rule.to;
- } else {
- ruleMatched = rule.codes.indexOf(statusCode) !== -1;
- }
- return ruleMatched;
- });
- if (matchedRule) {
- level = levels.getLevel(matchedRule.level, level);
- }
- }
- return level;
- }
- module.exports = function getLogger(logger4js, options) {
- if (typeof options === "string" || typeof options === "function") {
- options = { format: options };
- } else {
- options = options || {};
- }
- const thisLogger = logger4js;
- let level = levels.getLevel(options.level, levels.INFO);
- const fmt = options.format || DEFAULT_FORMAT;
- const nolog = createNoLogCondition(options.nolog);
- return (req, res, next) => {
-
- if (req._logging) return next();
-
- if (nolog && nolog.test(req.originalUrl)) return next();
- if (thisLogger.isLevelEnabled(level) || options.level === "auto") {
- const start = new Date();
- const { writeHead } = res;
-
- req._logging = true;
-
- res.writeHead = (code, headers) => {
- res.writeHead = writeHead;
- res.writeHead(code, headers);
- res.__statusCode = code;
- res.__headers = headers || {};
- };
-
- let finished = false;
- const handler = () => {
- if (finished) {
- return;
- }
- finished = true;
- res.responseTime = new Date() - start;
-
- if (res.statusCode && options.level === "auto") {
- level = levels.INFO;
- if (res.statusCode >= 300) level = levels.WARN;
- if (res.statusCode >= 400) level = levels.ERROR;
- }
- level = matchRules(res.statusCode, level, options.statusRules);
- const combinedTokens = assembleTokens(req, res, options.tokens || []);
- if (options.context) thisLogger.addContext("res", res);
- if (typeof fmt === "function") {
- const line = fmt(req, res, str => format(str, combinedTokens));
- if (line) thisLogger.log(level, line);
- } else {
- thisLogger.log(level, format(fmt, combinedTokens));
- }
- if (options.context) thisLogger.removeContext("res");
- };
- res.on("end", handler);
- res.on("finish", handler);
- res.on("error", handler);
- res.on("close", handler);
- }
-
- return next();
- };
- };
|