1 |
|
2 |
|
3 |
|
4 |
|
5 |
|
6 |
|
7 |
|
8 |
|
9 |
|
10 |
|
11 |
|
12 |
|
13 |
|
14 |
|
15 |
|
16 |
|
17 |
|
18 |
|
19 |
|
20 |
|
21 | var assert = require('assert');
|
22 | var Writable = require('readable-stream/writable');
|
23 |
|
24 | var Logger = require('../logger.js');
|
25 |
|
26 |
|
27 | var startRSS = process.memoryUsage().rss;
|
28 |
|
29 | console.log('# Logger supports back pressure');
|
30 |
|
31 | var backend = {
|
32 | createStream: function () {
|
33 | return LeakStream();
|
34 | }
|
35 | };
|
36 |
|
37 | var logger = Logger({
|
38 | meta: {},
|
39 | backends: {
|
40 | myBackend: backend
|
41 | },
|
42 | levels: {
|
43 | info: {
|
44 | backends: ['myBackend'],
|
45 | level: 30
|
46 | }
|
47 | }
|
48 | });
|
49 |
|
50 | var onceA = checkedWrite(logger);
|
51 |
|
52 | eqaulTAP(onceA.after.stream.length, 1000,
|
53 | 'onceA stream length is 1000');
|
54 | eqaulTAP(onceA.after.stream.buffer, 999,
|
55 | 'onceA stream buffer is 999');
|
56 |
|
57 | var deltaA = memoryGrowth(onceA);
|
58 | eqaulTAP(deltaA > 3, true,
|
59 | 'expected deltaA to be greater then 5 but found ' + deltaA);
|
60 |
|
61 | setTimeout(function () {
|
62 | var onceB = checkedWrite(logger);
|
63 |
|
64 | eqaulTAP(onceB.after.stream.length, 1000,
|
65 | 'onceB stream length is 1000');
|
66 | eqaulTAP(onceB.after.stream.buffer, 999,
|
67 | 'onceB stream buffer is 999');
|
68 |
|
69 | var deltaB = memoryGrowth(onceB);
|
70 |
|
71 | eqaulTAP(deltaB < 1, true,
|
72 | 'expected deltaB to be less then 1 but found ' + deltaB);
|
73 |
|
74 | setTimeout(function () {
|
75 | var onceC = checkedWrite(logger);
|
76 |
|
77 | eqaulTAP(onceC.after.stream.length, 1000,
|
78 | 'onceC stream length is 1000');
|
79 | eqaulTAP(onceC.after.stream.buffer, 999,
|
80 | 'onceC stream length is 999');
|
81 |
|
82 | var deltaC = memoryGrowth(onceC);
|
83 |
|
84 | eqaulTAP(deltaC < 0.5, true,
|
85 | 'expected deltaC to be less then 1 but found ' +
|
86 | deltaC);
|
87 |
|
88 | logger.destroy();
|
89 | }, 50);
|
90 | }, 50);
|
91 |
|
92 | function memoryGrowth(x) {
|
93 | return (
|
94 | (x.after.mem.rss - startRSS) /
|
95 | (x.before.mem.rss - startRSS)
|
96 | ) - 1;
|
97 | }
|
98 |
|
99 | function checkedWrite(logger) {
|
100 | var LOOP = 2e4;
|
101 |
|
102 | var before = inspect(logger);
|
103 |
|
104 | for (var i = 0; i < LOOP; i++) {
|
105 | logger.info('some message', {
|
106 | random: 'junk'
|
107 | });
|
108 | }
|
109 |
|
110 | return {
|
111 | before: before,
|
112 | after: inspect(logger)
|
113 | };
|
114 | }
|
115 |
|
116 | function LeakStream() {
|
117 | var s = new Writable({
|
118 | objectMode: true,
|
119 | highWaterMark: 1000
|
120 | });
|
121 | s._write = function leak(chunk, enc, cb) {
|
122 |
|
123 |
|
124 |
|
125 | };
|
126 | return s;
|
127 | }
|
128 |
|
129 | function inspect(logger) {
|
130 | var state = logger.streams.myBackend._writableState;
|
131 |
|
132 | return {
|
133 | mem: process.memoryUsage(),
|
134 | stream: {
|
135 | length: state.length,
|
136 | buffer: state.buffer.length
|
137 |
|
138 | }
|
139 | };
|
140 | }
|
141 |
|
142 | function eqaulTAP(a, b, message) {
|
143 | try {
|
144 | assert.equal(a, b, message);
|
145 | console.log('ok ' + message);
|
146 | } catch (err) {
|
147 | console.log('not ok ' + message);
|
148 | throw err;
|
149 | }
|
150 | }
|