forked from RoccoC/webpack-build-notifier
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
168 lines (156 loc) · 6.84 KB
/
index.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
/**
* @class WebpackBuildNotifierPlugin
* @extends Object
* A Webpack plugin that generates OS notifications for build steps using node-notifier.
*/
var path = require('path');
var os = require('os');
var notifier = require('node-notifier');
var stripAnsi = require('strip-ansi');
var exec = require('child_process').exec;
var WebpackBuildNotifierPlugin = function(cfg) {
cfg = cfg || {};
var defaultIconPath = path.resolve(__dirname, 'icons');
/**
* @cfg {String} [title='Webpack Build']
* The notification title prefix.
*/
this.title = cfg.title || 'Webpack Build';
/**
* @cfg {String} logo
* The absolute path to the project logo to be displayed as a content image in the notification. Optional.
*/
this.logo = cfg.logo;
/**
* @cfg {String} [sound='Submarine']
* The sound to play for notifications. Set to false to play no sound. Valid sounds are listed
* in the node-notifier project: https://github.com/mikaelbr/node-notifier
*/
this.sound = cfg.hasOwnProperty('sound') ? cfg.sound : 'Submarine';
/**
* @cfg {String} [successSound='Submarine']
* The sound to play for success notifications. Defaults to the value of the *sound* configuration option.
* Set to false to play no sound for success notifications. Takes precedence over the *sound* configuration option.
*/
this.successSound = cfg.hasOwnProperty('successSound') ? cfg.successSound : this.sound;
/**
* @cfg {String} [warningSound='Submarine']
* The sound to play for warning notifications. Defaults to the value of the *sound* configuration option.
* Set to false to play no sound for warning notifications. Takes precedence over the *sound* configuration option.
*/
this.warningSound = cfg.hasOwnProperty('warningSound') ? cfg.warningSound : this.sound;
/**
* @cfg {String} [failureSound='Submarine']
* The sound to play for failure notifications. Defaults to the value of the *sound* configuration option.
* Set to false to play no sound for failure notifications. Takes precedence over the *sound* configuration option.
*/
this.failureSound = cfg.hasOwnProperty('failureSound') ? cfg.failureSound : this.sound;
/**
* @cfg {Boolean} [suppressSuccess=false]
* True to suppress the success notifications, otherwise false (default). Note that the success notification will
* always be shown following a failed compilation regardless of this setting.
*/
this.suppressSuccess = cfg.suppressSuccess || false;
/**
* @cfg {Boolean} [suppressWarning=false]
* True to suppress the warning notifications, otherwise false (default).
*/
this.suppressWarning = cfg.suppressWarning || false;
/**
* @cfg {Boolean} [activateTerminalOnError=false]
* True to activate (focus) the terminal window when a compilation error occurs.
* Note that this only works on Mac OSX.
*/
this.activateTerminalOnError = cfg.activateTerminalOnError || false;
/**
* @cfg {String} [successIcon='./icons/success.png']
* The absolute path to the icon to be displayed for success notifications.
*/
this.successIcon = cfg.successIcon || path.join(defaultIconPath, 'success.png');
/**
* @cfg {String} [warningIcon='./icons/warning.png']
* The absolute path to the icon to be displayed for warning notifications.
*/
this.warningIcon = cfg.warningIcon || path.join(defaultIconPath, 'warning.png');
/**
* @cfg {String} [failureIcon='./icons/failure.png']
* The absolute path to the icon to be displayed for failure notifications.
*/
this.failureIcon = cfg.failureIcon || path.join(defaultIconPath, 'failure.png');
/**
* @property {Boolean} buildSuccessful
* Whether or not the last build was successful. Read-only.
*/
this.buildSuccessful = false;
/**
* @property {Function} onClick
* A function called when clicking the notification. By default, it activates the Terminal application.
*/
this.onClick = cfg.onClick || function(notifierObject, options) { this.activateTerminalWindow(); };
/**
* @property {Function} messageFormatter
* A function which returns a formatted notification message. The function is passed two parameters:
* 1) {Object} error/warning - The raw error or warning object.
* 2) {String} filepath - The path to the file containing the error/warning (if available).
*/
this.messageFormatter = cfg.messageFormatter || this.messageFormatter;
// add notification click handler to activate terminal window
notifier.on('click', this.onClick.bind(this));
};
WebpackBuildNotifierPlugin.prototype.messageFormatter = function(error, filepath) {
return filepath + os.EOL + (error.message ? error.message.replace(error.module ? error.module.resource : '', '') : '');
};
WebpackBuildNotifierPlugin.prototype.activateTerminalWindow = function() {
if (os.platform() === 'darwin') {
exec('osascript -e \'tell application "Terminal" to activate\'');
} else if (os.platform() === 'win32') {
// TODO: Windows platform
}
};
WebpackBuildNotifierPlugin.prototype.onCompilationDone = function(results) {
var notify = !this.suppressSuccess,
title = this.title + ' - ',
msg = 'Build successful!',
icon = this.successIcon,
sound = this.successSound;
if (results.hasErrors()) {
var error = results.compilation.errors[0];
notify = true;
title += 'Error';
msg = this.messageFormatter(error, error.module && error.module.rawRequest ? error.module.rawRequest : '');
icon = this.failureIcon;
sound = this.failureSound;
this.buildSuccessful = false;
} else if (!this.suppressWarning && results.hasWarnings()) {
var warning = results.compilation.warnings[0];
notify = true;
title += 'Warning';
msg = this.messageFormatter(warning, warning.module && warning.module.rawRequest ? warning.module.rawRequest : '');
icon = this.warningIcon;
sound = this.warningSound;
this.buildSuccessful = false;
} else {
title += 'Success';
if (!notify && !this.buildSuccessful) {
notify = true; // previous build failed, let's show a notification even if success notifications are suppressed
}
this.buildSuccessful = true;
}
if (notify) {
notifier.notify({
title: title,
message: stripAnsi(msg),
sound: sound,
contentImage: this.logo,
icon: icon,
wait: true
});
}
if (this.activateTerminalOnError && !this.buildSuccessful) {
this.activateTerminalWindow();
}
};
WebpackBuildNotifierPlugin.prototype.apply = function(compiler) {
compiler.plugin('done', this.onCompilationDone.bind(this));
};
module.exports = WebpackBuildNotifierPlugin;