brianc/node-pg-copy-streams

_final is sometimes called before submit

gabegorelick opened this issue · 4 comments

I haven't been able to get a minimal test case yet, but when submitting multiple COPY FROM queries in parallel on the same connection, I often see the following error:

TypeError: Cannot read property 'stream' of undefined
    at CopyStreamQuery._final (node_modules/pg-copy-streams/copy-from.js:70:21)
    at callFinal (_stream_writable.js:617:10)
    at process._tickCallback (internal/process/next_tick.js:63:19)

I've tracked it down to the fact that _final is called before submit. Since submit is what sets this.connection, if it's not called then this.connection is undefined.

Since node-postgres is queuing these queries, what stops the input stream (the one piped to copy-from) from finishing before our COPY query becomes active?

Relevant code:

submit(connection) {
this.connection = connection
connection.query(this.text)
}

_final(cb) {
this.flush()
const Int32Len = 4
const finBuffer = Buffer.from([code.CopyDone, 0, 0, 0, Int32Len])
this.connection.stream.write(finBuffer)
this.connection = null
this.cb_flush = cb
}

Hello thanks for your report on this.

I know that we delay the callback of _final (cf cb_flush) in order to avoid a problem when pg will dequeue the next query.

now for a very small input source, the whole source could indeed be swallowed by the internal buffers of the writableStream before pg dequeues the COPY query.

I would have thought that the stream beeing corked in the constructor was sufficient to handle this (it is uncorked later when CopyInResponse is received so pg has called submit already)

Your report seems to show that this may not be an effective strategy in all cases.

The documentation indeed states that

The writable.cork() method forces all written data to be buffered in memory. The buffered data will be flushed when either the stream.uncork() or stream.end() methods are called.

so if stream.end() is called on the source before pg could submit the query there might be a problem I suppose we need a test pinpointing this.

We need to keep a _final implementation because it is needed to delay the finish event until the COPY operation is done ; but we could probably improve the sequence of things to handle this corner case.

OK it seems that the delaying of the _write and _writev callbacks works as intended (any first write to write or writev will be acknowledged only after CopyInResponse is received, meaning that pg has submitted the query).

There is only one case where I could reproduce your issue, which is when the source is totally empty. In this case, _write and _writev are never called (thus not delaying the call to _final), and the call to _final breaks.

This breaks _final with or without the query beeing queued by pg.

now my question is : do you thinks it matches the issue you are experiencing ? do you sometimes have empty sources in your sequence of copyFrom operations ?

Could you please try version 5.1.1 that fixes an issue with empty sources

I can confirm that 97febfc fixes my issue.

Thanks @jeromew!