Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Thrown readable stream error while uncorking #6154

Closed
jfhbrook opened this issue Apr 11, 2016 · 4 comments
Closed

Thrown readable stream error while uncorking #6154

jfhbrook opened this issue Apr 11, 2016 · 4 comments
Labels
confirmed-bug Issues with confirmed bugs. stream Issues and PRs related to the stream subsystem.

Comments

@jfhbrook
Copy link
Contributor

  • Version: 4.4.1 (currently upgrading to 4.4.2 to see what's good)
  • Platform: Linux ip-10-90-38-23 3.10.0-123.8.1.el7.x86_64 deps: update openssl to 1.0.1j #1 SMP Mon Sep 22 19:06:58 UTC 2014 x86_64 x86_64 x86_64 GNU/Linux
  • Subsystem: _stream_writable.js

I have some production services that are throwing a pretty crazy error:

TypeError: Cannot set property 'entry' of null
    at clearBuffer (_stream_writable.js:379:18)
    at Socket.Writable.uncork (_stream_writable.js:238:7)
    at RedisClient.uncork (/cn/runtime/epi-services/nodejs/releases/current/node_modules/@condenast/tsugu-service/node_modules/redis/index.js:361:25)
    at Multi.exec_transaction (/cn/runtime/epi-services/nodejs/releases/current/node_modules/@condenast/tsugu-service/node_modules/redis/index.js:1171:18)

For some background: We're using redis as a caching layer, and this code is getting triggered while writing to redis with its 'multi' helper. We're using v2.4.2 of the redis module but I'm gonna upgrade to 2.5.3 just in case it helps.

As far as I can tell, what's happening here is that the redis client creates a net stream and at various points in time might call uncork on it, including inside the multi exec handler. For some reason, only sometimes, when this happens, trying to clear the buffer throws because state.corkedRequestsFree is null.

I did look at net and Duplex code paths, as well as the redis code, and didn't see anything touching internal stream state, but maybe I missed something?

Unfortunately, I don't have a reproducing case beyond the production code. I tried triggering this with a naive fuzzer on a writable stream, no dice.

@mcollina tagging you because we talked about this on irc a little bit, and because you have the git blame for CorkedRequest.

Thanks!

EDIT: Also created an issue on the redis project just in case

@jfhbrook jfhbrook changed the title Thrown error while uncorking Thrown readable stream error while uncorking Apr 11, 2016
@mscdex mscdex added the stream Issues and PRs related to the stream subsystem. label Apr 11, 2016
@mcollina
Copy link
Member

I am sorry this happened. This commit has been around in node v5 and readable-stream for months. Unfortunately, nobody catched this very particular behavior before it was backported.

I can reproduce it with the following two scripts, the server and the client:

'use strict'

const net = require('net')
let chunks = 0
net.createServer((stream) => {
  stream.on('data', () => chunks++)
}).listen(4242)

setInterval(() => {
  console.log('received', chunks, 'chunks')
  chunks = 0
}, 1000)
'use strict'

const net = require('net')
const stream = net.connect(4242)
const buf = new Buffer('42')

function doWrite () {
  console.log('doWrite')
  let tooMuch = 0
  while (tooMuch < 10) {
    stream.cork()
    stream.write(buf)
    stream.write(buf) || tooMuch++ // we need two writes to trigger the bug
    stream.uncork()
  }
  console.log('break', tooMuch)
}

stream.setNoDelay(true)
stream.on('drain', doWrite)

doWrite()

The behavior of this script is to leverage the OS tcp buffering, and write synchronously. This come from this line: https://github.com/nodejs/node/blob/master/lib/net.js#L690. The normal output of this script is:

$ node bug-client.js
doWrite
break 10
doWrite

As you can immagine, this script keeps writing in a completely synchronous manner. This technique is used to demonstrate the bug. My example is a corner case which will lead (after a very long period) as a memory leak, because the buffer will never be collected (a reference to the callback will be held in the process.nextTick queue):

process.nextTick(afterWrite, stream, state, finished, cb);
.

I am already working on a fix that does not lead to a slowdown of any sort.

I missed this because I have always tested cork/uncork behavior following this pattern:

node/lib/_http_outgoing.js

Lines 467 to 470 in be68b68

if (this.connection && !this.connection.corked) {
this.connection.cork();
process.nextTick(connectionCorkNT, this.connection);
}
. Which does not trigger the issue.I am sorry this happened. This commit has been around in node v5 and readable-stream for months. Unfortunately, nobody catched this very particular behavior before it was backported.

@mcollina
Copy link
Member

cc @nodejs/streams

mcollina added a commit to mcollina/node that referenced this issue Apr 12, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
nodejs@89aeab9
(nodejs#4354).

Fixes: nodejs#6154.
@Fishrock123 Fishrock123 added the confirmed-bug Issues with confirmed bugs. label Apr 12, 2016
@jfhbrook
Copy link
Contributor Author

@mcollina thanks for the quick work on this!

@mcollina
Copy link
Member

No worries, I'm responsible for this, and I have bandwidth... as of yesterday :D.

MylesBorins pushed a commit that referenced this issue Apr 19, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
89aeab9
(#4354).

Fixes: #6154
PR-URL: #6164
Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com>
Reviewed-By: Mathias Buus <mathiasbuus@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
MylesBorins pushed a commit that referenced this issue Apr 20, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
89aeab9
(#4354).

Fixes: #6154
PR-URL: #6164
Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com>
Reviewed-By: Mathias Buus <mathiasbuus@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
MylesBorins pushed a commit that referenced this issue Apr 20, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
89aeab9
(#4354).

Fixes: #6154
PR-URL: #6164
Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com>
Reviewed-By: Mathias Buus <mathiasbuus@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
jasnell pushed a commit that referenced this issue Apr 26, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
89aeab9
(#4354).

Fixes: #6154
PR-URL: #6164
Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com>
Reviewed-By: Mathias Buus <mathiasbuus@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
MylesBorins pushed a commit that referenced this issue Apr 30, 2016
net streams can request multiple chunks to be written in a synchronous
fashion. If this is combined with cork/uncork, en error is currently
thrown because of a regression introduced in:
89aeab9
(#4354).

Fixes: #6154
PR-URL: #6164
Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com>
Reviewed-By: Mathias Buus <mathiasbuus@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
confirmed-bug Issues with confirmed bugs. stream Issues and PRs related to the stream subsystem.
Projects
None yet
Development

No branches or pull requests

4 participants