Search code examples
javascriptgoogle-chromeserial-portwebapi

How can I interrupt a Reader when it "hangs" (need a timeout on Reader.read() )


This question is related to a situation that occurred when using Chrome Serial API but probably could be relevant to any ReadableStream. I studied the documentation and probably missed some feature or pattern.

A simple program is running in Chrome browser, accessing CW keyer (based on Arduino, but this is not important).

The application sends a command to the keyer and expects two binary bytes or a string as a response (particular format depends on the command sent and is not important).

In case that the serial device (not the USB/serial adapter, but the Arduino) misses the command for whatever reason, response is never sent and the function expectResponse() below will never return any data, nor will it throw any exception. As a result, the Reader remains locked, the ReadableStream therefore cannot be closed and as a consequence, the serial port cannot be closed either.

Also, depending on the application structure, in case of other command sent successfully to the keyer, it may be not possible to read the second response because the first reader blocks the stream and until it is released, new Reader cannot be created.


async function expectResponse( serialPort ) {
   const reader = serialPort.readable.getReader() ;
   let { value, done } = await reader.read() ; // this never returns because no data arrive, not possible to "break"
}

async function disconnect( serialPort ) {
   // ... some cleanup ...
   // naive attempt to unlock ReadableStream before closing 
   await serialPort.readable.getReader().releaseLock() // this will throw exception - cannot create  new reader while another one is still active and locks the stream
   // ...
   await serialPort.close(); // this will throw exception - cannot close port because readable stream is locked
}

serialPort is the object returned by navigator.serial.requestPort()

I am convinced I must have missed something important in API docs (ReadableStream or Reader API, not Serial API), but I did not find solution.

P.S. in the real app serialPort is a global variable, but it does not matter, does it?


Solution

  • I don't think ReadableStream has timeouts built in.

    I'd use Promise.race with the other promise being your timeout:

    let { value, done } = await Promise.race([
        reader.read(),
        new Promise((_, reject) => setTimeout(reject, TIMEOUT, new Error("timeout")))
    ]);
    

    (You'd probably put that new Promise code in a utility function.)

    Promise.race watches the promises race, settling its promise based on the first settlement it sees of the promises in the array you give it. So if read's promise is fulfilled (or rejected) before the timeout promise rejects, read's settlement dictates the settlement of the race promise. Otherwise, the race promise settles based on the settlement (in this case, rejection) of the timeout promise.