aa602938a432600842d9016cb23fcfb7b999a18b.svn-base
3.8 KB
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
/*
* loggly.js: Transport for logginh to remote Loggly API
*
* (C) 2010 Charlie Robbins
* MIT LICENCE
*
*/
var events = require('events'),
loggly = require('loggly'),
util = require('util'),
async = require('async'),
common = require('../common'),
Transport = require('./transport').Transport;
//
// ### function Loggly (options)
// #### @options {Object} Options for this instance.
// Constructor function for the Loggly transport object responsible
// for persisting log messages and metadata to Loggly; 'LaaS'.
//
var Loggly = exports.Loggly = function (options) {
Transport.call(this, options);
function valid() {
return options.inputToken
|| options.inputName && options.auth
|| options.inputName && options.inputs && options.inputs[options.inputName]
|| options.id && options.inputs && options.inputs[options.id];
}
if (!options.subdomain) {
throw new Error('Loggly Subdomain is required');
}
if (!valid()) {
throw new Error('Target input token or name is required.');
}
this.name = 'loggly';
this.logBuffer = [];
this.client = loggly.createClient({
subdomain: options.subdomain,
auth: options.auth || null,
json: options.json || false
});
if (options.inputToken) {
this.inputToken = options.inputToken;
this.ready = true;
}
else if (options.inputs && (options.inputs[options.inputName]
|| options.inputs[options.id])) {
this.inputToken = options.inputs[options.inputName] || options.inputs[options.id];
this.ready = true;
}
else if (options.inputName) {
this.ready = false;
this.inputName = options.inputName;
var self = this;
this.client.getInput(this.inputName, function (err, input) {
if (err) {
throw err;
}
self.inputToken = input.input_token;
self.ready = true;
});
}
};
//
// Inherit from `winston.Transport`.
//
util.inherits(Loggly, Transport);
//
// Expose the name of this Transport on the prototype
//
Loggly.prototype.name = 'loggly';
//
// ### function log (level, msg, [meta], callback)
// #### @level {string} Level at which to log the message.
// #### @msg {string} Message to log
// #### @meta {Object} **Optional** Additional metadata to attach
// #### @callback {function} Continuation to respond to when complete.
// Core logging method exposed to Winston. Metadata is optional.
//
Loggly.prototype.log = function (level, msg, meta, callback) {
if (this.silent) {
return callback(null, true);
}
var self = this,
message = common.clone(meta || {});
message.level = level;
message.message = msg;
if (!this.ready) {
//
// If we haven't gotten the input token yet
// add this message to the log buffer.
//
this.logBuffer.push(message);
}
else if (this.ready && this.logBuffer.length > 0) {
//
// Otherwise if we have buffered messages
// add this message to the buffer and flush them.
//
this.logBuffer.push(message);
this.flush();
}
else {
//
// Otherwise just log the message as normal
//
this.client.log(this.inputToken, message, function () {
self.emit('logged');
});
}
callback(null, true);
};
//
// ### function flush ()
// Flushes any buffered messages to the current `stream`
// used by this instance.
//
Loggly.prototype.flush = function () {
var self = this;
function logMsg (msg, next) {
self.client.log(self.inputToken, msg, function (err) {
if (err) {
self.emit('error', err);
}
next();
});
}
//
// Initiate calls to loggly for each message in the buffer
//
async.forEach(this.logBuffer, logMsg, function () {
self.emit('logged');
});
process.nextTick(function () {
//
// Then quickly truncate the list
//
self.logBuffer.length = 0;
});
};