log.ts 3.3 KB
Newer Older
liuzhe-lz's avatar
liuzhe-lz committed
1
2
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT license.
Deshui Yu's avatar
Deshui Yu committed
3
4
5
6
7

'use strict';

import * as fs from 'fs';
import { Writable } from 'stream';
8
import * as util from 'util';
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

/* log level constants */

export const DEBUG = 10;
export const INFO = 20;
export const WARNING = 30;
export const ERROR = 40;
export const CRITICAL = 50;

export const TRACE = 1;
export const FATAL = 50;

const levelNames = new Map<number, string>([
    [CRITICAL, 'CRITICAL'],
    [ERROR, 'ERROR'],
    [WARNING, 'WARNING'],
    [INFO, 'INFO'],
    [DEBUG, 'DEBUG'],
    [TRACE, 'TRACE'],
28
]);
Deshui Yu's avatar
Deshui Yu committed
29

30
/* global_ states */
Deshui Yu's avatar
Deshui Yu committed
31

32
33
let logLevel: number = 0;
const loggers = new Map<string, Logger>();
Deshui Yu's avatar
Deshui Yu committed
34

35
/* major api */
Deshui Yu's avatar
Deshui Yu committed
36

37
38
39
40
41
export class Logger {
    private name: string;

    constructor(name: string = 'root') {
        this.name = name;
Deshui Yu's avatar
Deshui Yu committed
42
43
    }

44
45
46
    public trace(...args: any[]): void {
        this.log(TRACE, args);
    }
47

48
49
50
    public debug(...args: any[]): void {
        this.log(DEBUG, args);
    }
SparkSnail's avatar
SparkSnail committed
51

52
53
    public info(...args: any[]): void {
        this.log(INFO, args);
54
55
    }

56
57
    public warning(...args: any[]): void {
        this.log(WARNING, args);
Deshui Yu's avatar
Deshui Yu committed
58
59
    }

60
61
    public error(...args: any[]): void {
        this.log(ERROR, args);
62
63
    }

64
65
    public critical(...args: any[]): void {
        this.log(CRITICAL, args);
Deshui Yu's avatar
Deshui Yu committed
66
67
    }

68
69
    public fatal(...args: any[]): void {
        this.log(FATAL, args);
Deshui Yu's avatar
Deshui Yu committed
70
71
    }

72
    private log(level: number, args: any[]): void {
73
        const logFile: Writable | undefined = (global as any).logFile;
J-shang's avatar
J-shang committed
74
        if (level < logLevel) {
75
            return;
Deshui Yu's avatar
Deshui Yu committed
76
77
        }

78
79
80
81
82
        const zeroPad = (num: number): string => num.toString().padStart(2, '0');
        const now = new Date();
        const date = now.getFullYear() + '-' + zeroPad(now.getMonth() + 1) + '-' + zeroPad(now.getDate());
        const time = zeroPad(now.getHours()) + ':' + zeroPad(now.getMinutes()) + ':' + zeroPad(now.getSeconds());
        const datetime = date + ' ' + time;
83
84
85

        const levelName = levelNames.has(level) ? levelNames.get(level) : level.toString();

86
        const message = args.map(arg => (typeof arg === 'string' ? arg : util.inspect(arg))).join(' ');
87
        
88
        const record = `[${datetime}] ${levelName} (${this.name}) ${message}\n`;
J-shang's avatar
J-shang committed
89
90
91
92
93
94

        if (logFile === undefined) {
            console.log(record);
        } else {
            logFile.write(record);
        }
Deshui Yu's avatar
Deshui Yu committed
95
    }
96
}
Deshui Yu's avatar
Deshui Yu committed
97

98
99
100
101
102
export function getLogger(name: string = 'root'): Logger {
    let logger = loggers.get(name);
    if (logger === undefined) {
        logger = new Logger(name);
        loggers.set(name, logger);
Deshui Yu's avatar
Deshui Yu committed
103
    }
104
105
106
107
108
109
110
111
112
113
114
115
116
    return logger;
}

/* management functions */

export function setLogLevel(levelName: string): void {
    if (levelName) {
        const level = module.exports[levelName.toUpperCase()];
        if (typeof level === 'number') {
            logLevel = level;
        } else {
            console.log('[ERROR] Bad log level:', levelName);
            getLogger('logging').error('Bad log level:', levelName);
SparkSnail's avatar
SparkSnail committed
117
        }
Deshui Yu's avatar
Deshui Yu committed
118
119
120
    }
}

121
export function startLogging(logPath: string): void {
122
    (global as any).logFile = fs.createWriteStream(logPath, {
123
124
125
126
        flags: 'a+',
        encoding: 'utf8',
        autoClose: true
    });
Deshui Yu's avatar
Deshui Yu committed
127
128
}

129
export function stopLogging(): void {
130
131
132
    if ((global as any).logFile !== undefined) {
        (global as any).logFile.end();
        (global as any).logFile = undefined;
133
134
    }
}