forked from x821938/node-red-contrib-flogger
-
Notifications
You must be signed in to change notification settings - Fork 1
/
flogger.js
220 lines (194 loc) · 7.5 KB
/
flogger.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
const fs = require('fs')
const moment = require('moment')
const mustache=require('mustache')
const validLoglevels = ["ERROR", "WARN", "INFO", "DEBUG", "TRACE"]
const rotate = require('log-rotate');
module.exports = function(RED) {
function FloggerNode(n) {
RED.nodes.createNode(this,n)
this.loglevel = n.loglevel
this.logfile = n.logfile
this.inputchoice = n.inputchoice
this.inputobject = n.inputobject
this.inputobjectType = n.inputobjectType
this.inputmoustache = n.inputmoustache
this.logconfig = RED.nodes.getNode(n.logconfig)
this.sendpane = n.sendpane
var node = this
node.on('input', function(msg) {
loglevel = node.loglevel
if (validLoglevels.includes(msg.loglevel)) { // Override loglevel with msg.loglevel
loglevel = msg.loglevel
}
logTimeStamp = GetLogTime(node)
logmessage = ConstructLogMessage(node, msg)
if (node.sendpane) { // User wants the logentry also in the debug pane of the webinterface
node.warn(loglevel + " [" + logmessage.var + "] " + logmessage.msg)
}
if (node.logconfig.logstyle == "plain") {
timeStamp = (node.logconfig.stamp === 'none' ? '' : logTimeStamp + ' ')
level = (loglevel === '' ? '' : loglevel + ' ')
topic = (node.logconfig.logtopic === true && msg.topic ? msg.topic : '')
source = (node.logconfig.logsource === true ? logmessage.var : '')
topicOrSource = (topic !== '' || source !== '')
topicAndSource = (topic !== '' && source !== '')
bracket1 = (topicOrSource ? '[' : '')
bracket2 = (topicOrSource ? '] ' : '')
separator = (topicAndSource ? ':' : '')
logline = timeStamp + level + bracket1 + topic + separator + source + bracket2 + logmessage.msg + '\n'
} else {
logline = GetJSONMsg(logTimeStamp, loglevel, logmessage.var, logmessage.raw)
}
logfile = node.logfile
if (msg.logfile) logfile = msg.logfile // logfile override via msg object if provided
LogRotate(node, logfile, logline.length, msg.rotateNow)
WriteMsgToFile(node, logfile, logline, logTimeStamp)
node.send(msg); // pass through original message
})
}
RED.nodes.registerType("flogger",FloggerNode)
function FloggerConfigNode(n) {
RED.nodes.createNode(this,n)
this.logdir = n.logdir
this.logname = n.logname
this.stamp = n.stamp
this.stampFormat = n.stampFormat;
this.logstyle = n.logstyle
this.logtopic = n.logtopic
this.logsource = n.logsource
this.logrotate = n.logrotate
this.logcompress = n.logcompress
this.logrotatecount = n.logrotatecount
this.logsize = n.logsize
}
RED.nodes.registerType("config-log",FloggerConfigNode)
/* Input: variable
If input is an object it will return a json version of the object. Otherwise just the object will be returned
*/
function VarToString(v) {
if (typeof v == 'object') {
vs = JSON.stringify(v)
} else {
vs = v
}
return vs
}
/* Input logtimestamp, loglevel, variable (the object logged) and raw message object to be logged.
Will return a string with the entire json formattet line to be logged
*/
function GetJSONMsg(logTimeStamp, loglevel, msgvar, messageRaw) {
logobject = {}
logobject.time = logTimeStamp
logobject.level = loglevel
logobject.var = msgvar
logobject.message = messageRaw
logline = JSON.stringify(logobject) + "\n"
return (logline)
}
/* Input: node object and nodered message object
Output object:
.msg: The entire log message that should be logged to the file as a string
.var: A text representation of the object logged. EG: msg.payload, flow.var, global.var, msg or moustache
.raw: The object that should be logged but returned as an object.
*/
function ConstructLogMessage(node, msg) {
if (node.inputchoice == "object" && node.inputobject.length>0 ) {
message = "Please choose a flow or global name!"
if ( node.inputobjectType == "msg") {
if (node.inputobject) {
messageRaw = eval("msg." + node.inputobject)
message = VarToString(messageRaw)
messageVar = "msg." + node.inputobject
} else {
messageRaw = msg
message = VarToString(messageRaw)
messageVar = "msg"
}
} else if (node.inputobjectType == "flow" ) {
flowvar = node.inputobject
messageRaw = node.context().flow.get(flowvar)
message = VarToString(messageRaw)
messageVar = "flow." + flowvar
} else if (node.inputobjectType == "global") {
globalvar = node.inputobject
messageRaw = node.context().global.get(globalvar)
message = VarToString(messageRaw)
messageVar = "global." + globalvar
}
} else if (node.inputchoice == "fullmsg") {
messageRaw = msg
message = VarToString(messageRaw)
messageVar = "msg"
} else if (node.inputchoice == "moustache") {
mustache.escape = function(text) {return text;}
messageRaw = mustache.render(node.inputmoustache, msg)
message = messageRaw
messageVar = "moustache"
}
return {msg: message, var: messageVar, raw: messageRaw}
}
/* Input: node object
Depending on node.logconfig.stamp the corresponding log time is returned in the selected format.
Allowed time formats: none, utc, local
*/
function GetLogTime(node) {
now = new Date();
const format = node.logconfig.stampFormat||"YYYY/MM/DD HH:mm:ss";
if (node.logconfig.stamp == "none") {
logtime = ""
} else if (node.logconfig.stamp == "utc") {
logtime = moment(now).utc().format(format) + "Z"
} else if (node.logconfig.stamp == "local") {
logtime = moment(now).parseZone().local().format(format)
}
return (logtime)
}
/* Input: node object, filename (not with path), the message that should be logged to the file, and the logtime
If no file is provided a warning will be logged to debugpane.
If the file is succesfully written the node status will be show write time. If not it will show
"cant write file" and a warning will be logged to debugpane
*/
function WriteMsgToFile(node, filename, msg, logtime) {
if (logfile) {
fullpath = node.logconfig.logdir + "/" + logfile
try {
fs.appendFileSync(fullpath, logline)
if (node.logconfig.stamp != "none") {
node.status({shape: "ring", fill: "green", text: logtime})
}
} catch (err) {
node.status({shape: "ring", fill: "red", text: "Cant write file!"})
node.error("Can't write file: " + err, msg);
}
} else {
node.status({shape: "ring", fill: "red", text: "Missing logfile!"})
node.warn("Nothing got logged because you didn't provide a logfile in configuration and has not been overridden from msg.logfile")
}
}
/* Input: node object, filename with the full path to the logfile and no. of bytes that is
going to be added to the logfile. This is to prevent it of tipping over the size.
If the logfile exists and it will exceed the size set in node.logconfig.logrotate, it will be rotated
There will be maximum node.logconfig.rotatecount files in the directory.
If compression is set in node.logconfig.compress then the rotated files will be gzipped
*/
function LogRotate(node, filename, addlength, rotateNow) {
if (node.logconfig.logrotate && filename) {
fullpath = node.logconfig.logdir + "/" + filename
if (fs.existsSync(fullpath)) {
stats = fs.statSync(fullpath)
fileSizeInBytes = stats["size"]
if ((rotateNow === true) || (fileSizeInBytes + addlength + 1 >= node.logconfig.logsize * 1000 )) {
rotate(fullpath, { count: node.logconfig.logrotatecount, compress: (node.logconfig.logcompress==true)}, function(err) {
if (err) {
node.warn("Could not rotate logfiles: " + err)
} else {
if (!fs.existsSync(fullpath)) {
fs.appendFileSync(fullpath,''); // Create a new empty file if there is none
}
}
})
}
}
}
}
}