如何关闭通过 TransformStream 传输的 Web 串行端口?

问题描述 投票:0回答:2

我有一个 Web 串行端口,我想从中读取一些数据。我想使用

TransformStream
来进行一些处理(例如,将字节解码为字符串、分离出逻辑消息等),方法是使用
pipeThrough
来转换数据。然而,一旦我这样做了,我就无法再通过调用
reader.releaseLock()
来释放端口上的锁定。我在这里做错了什么?

此代码按照我的预期工作(在安全上下文中在浏览器中运行,无需依赖项):

async serialTestWorking() {
  const port = await navigator.serial.requestPort();

  await port.open({baudRate: 9600});
  console.log("Is locked after open?", port.readable.locked);
  // Prints "Is locked after open? false"

  let reader = port.readable.getReader();
  console.log("Is locked after getReader?", port.readable.locked);
  // Prints "Is locked after getReader? true"

  reader.releaseLock();
  console.log("Is locked after releaseLock?", port.readable.locked);
  // Prints "Is locked after releaseLock? false"

  await port.close();
  console.log("Port closed");
  // Prints "Port closed"
}

但是,如果我使用

pipeThrough
通过无所事事的
TransformStream
发送输出,那么一切都会崩溃。
releaseLock
时锁未释放,最终
close
无法工作。

async serialTestBroken() {
  const port = await navigator.serial.requestPort();

  await port.open({baudRate: 9600});
  console.log("Is locked after open?", port.readable.locked);
  // Prints "Is locked after open? false"

  let reader = port.readable.pipeThrough(new TransformStream()).getReader();
  console.log("Is locked after getReader?", port.readable.locked);
  // Prints "Is locked after getReader? true"

  reader.releaseLock();
  console.log("Is locked after releaseLock?", port.readable.locked);
  // Prints "Is locked after releaseLock? true"

  await port.close();
  console.log("Port closed");
  // Doesn't make it to the log line
  //throws "TypeError: Failed to execute 'close' on 'SerialPort': Cannot cancel a locked stream"
}

我在这里做错了什么?释放

TransformStream
上的锁真的不会传播到上游吗?我是否必须跟踪管道中每个变压器的实例,以便确保将它们全部解锁?

streams 规范表示管道在管道操作期间锁定可读和可写流。

管道锁定可读和可写流,防止它们在管道操作期间被操纵。这允许实现执行重要的优化,例如直接将数据从底层源传输到底层接收器,同时绕过许多中间队列。

是否有其他方式我必须表明“管道操作”已完成?

javascript asynchronous stream web-serial-api
2个回答
9
投票

https://web.dev/serial/#close-port 中所述,如果

port.close()
readable
成员已解锁,
writable
会关闭串行端口,这意味着
releaseLock()
已被调用各自的读者和作者。

使用转换流时关闭串行端口有点复杂。拨打

reader.cancel()
,然后拨打
writer.close()
port.close()
。这会将错误通过转换流传播到底层串行端口。由于错误传播不会立即发生,因此您需要使用之前创建的
readableStreamClosed
writableStreamClosed
Promise 来检测
port.readable
port.writable
何时解锁。取消读取器会导致流中止;这就是为什么您必须捕获并忽略由此产生的错误。

const textDecoder = new TextDecoderStream();
const readableStreamClosed = port.readable.pipeTo(textDecoder.writable);
const reader = textDecoder.readable.getReader();

// Listen to data coming from the serial device.
while (true) {
  const { value, done } = await reader.read();
  if (done) {
    reader.releaseLock();
    break;
  }
  // value is a string.
  console.log(value);
}
// Later...

const textEncoder = new TextEncoderStream();
const writer = textEncoder.writable.getWriter();
const writableStreamClosed = textEncoder.readable.pipeTo(port.writable);

reader.cancel();
await readableStreamClosed.catch(() => { /* Ignore the error */ });

writer.close();
await writableStreamClosed;

await port.close();

0
投票

对于可能关心的人来说,我也发现了这个解决方案,它一直有效,直到不再有效。

await thing.catch(Object)
作为无操作是准确的(同时也很尴尬),但总的来说,经过 1 小时的混乱,我得出的结论是,关闭端口并释放所有内容的最简单方法是传递 AbortController.signal

pipeTo 确实接受第二个

options
参数,该参数又接受
signal
属性,因此这就是我最终所做的,并最终正确释放并关闭端口:

const writable = new WritableStream({ ... });

// pass { signal: aborter.signal } or just aborter
const aborter = new AbortController;
const readerClosed = port.readable.pipeTo(writable, aborter);

// later on ...
aborter.abort(); // that's it ...
// the rest of the dance
writer.close();
await writerClosed;
await readerClosed.catch(Object);
await port.close();

我希望这个提示能够帮助那些像我今天早上一样疯狂的人。

© www.soinside.com 2019 - 2024. All rights reserved.