Simplify processOrders, fixing buffer off-by-one error.
This commit is contained in:
parent
6066be83fe
commit
3e92604d18
2 changed files with 316 additions and 26 deletions
40
lib/index.js
40
lib/index.js
|
@ -43,15 +43,10 @@ const Channel = function(length = 0) {
|
||||||
let buffered = 0;
|
let buffered = 0;
|
||||||
let closed = false;
|
let closed = false;
|
||||||
let lastValue;
|
let lastValue;
|
||||||
let resolvedIndex = 0;
|
|
||||||
const pushes = [];
|
const pushes = [];
|
||||||
const shifts = [];
|
const shifts = [];
|
||||||
|
|
||||||
// Process the push and shift queues like an order book, looking for matches.
|
const matchPushesAndShifts = index => {
|
||||||
const processOrders = () => {
|
|
||||||
const index = { push: 0, shift: 0 };
|
|
||||||
|
|
||||||
// Match pushes and shifts.
|
|
||||||
while (index.push < pushes.length && index.shift < shifts.length) {
|
while (index.push < pushes.length && index.shift < shifts.length) {
|
||||||
const push = pushes[index.push];
|
const push = pushes[index.push];
|
||||||
const shift = shifts[index.shift];
|
const shift = shifts[index.shift];
|
||||||
|
@ -63,32 +58,31 @@ const Channel = function(length = 0) {
|
||||||
} else {
|
} else {
|
||||||
lastValue = push.value;
|
lastValue = push.value;
|
||||||
shift.resolve(lastValue);
|
shift.resolve(lastValue);
|
||||||
buffered = Math.max(0, buffered - 1);
|
|
||||||
index.push++;
|
|
||||||
index.shift++;
|
index.shift++;
|
||||||
|
push.resolve(length);
|
||||||
|
index.push++;
|
||||||
|
buffered = Math.max(0, buffered - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Resolve push promises up to the end of the buffer.
|
// Resolve push promises up to the end of the buffer.
|
||||||
|
const resolveBufferedPushes = index => {
|
||||||
for (
|
for (
|
||||||
;
|
let resolvedIndex = index.push + buffered;
|
||||||
resolvedIndex < index.push ||
|
resolvedIndex < pushes.length && buffered < length;
|
||||||
(resolvedIndex < pushes.length && buffered < length);
|
|
||||||
resolvedIndex++
|
resolvedIndex++
|
||||||
) {
|
) {
|
||||||
const { cancelled, resolve } = pushes[resolvedIndex];
|
const { cancelled, resolve } = pushes[resolvedIndex];
|
||||||
|
|
||||||
if (!cancelled) {
|
if (!cancelled) {
|
||||||
if (resolvedIndex > index.push) {
|
|
||||||
buffered++;
|
buffered++;
|
||||||
}
|
|
||||||
|
|
||||||
resolve(length);
|
resolve(length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// If the channel is closed then resolve 'undefined' to remaining shifts.
|
const resolveClosedShifts = index => {
|
||||||
if (closed) {
|
|
||||||
for (; index.shift < shifts.length; index.shift++) {
|
for (; index.shift < shifts.length; index.shift++) {
|
||||||
const { cancelled, resolve } = shifts[index.shift];
|
const { cancelled, resolve } = shifts[index.shift];
|
||||||
|
|
||||||
|
@ -97,11 +91,21 @@ const Channel = function(length = 0) {
|
||||||
resolve(lastValue);
|
resolve(lastValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Process the push and shift queues like an order book, looking for matches.
|
||||||
|
const processOrders = () => {
|
||||||
|
const index = { push: 0, shift: 0 };
|
||||||
|
matchPushesAndShifts(index);
|
||||||
|
resolveBufferedPushes(index);
|
||||||
|
|
||||||
|
// If the channel is closed then resolve 'undefined' to remaining shifts.
|
||||||
|
if (closed) {
|
||||||
|
resolveClosedShifts(index);
|
||||||
}
|
}
|
||||||
|
|
||||||
pushes.splice(0, index.push);
|
pushes.splice(0, index.push);
|
||||||
shifts.splice(0, index.shift);
|
shifts.splice(0, index.shift);
|
||||||
resolvedIndex -= index.push;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const readOnly = Object.freeze(
|
const readOnly = Object.freeze(
|
||||||
|
@ -317,7 +321,7 @@ const Channel = function(length = 0) {
|
||||||
|
|
||||||
// If value is a promise that rejects, catch it in case there hasn't
|
// If value is a promise that rejects, catch it in case there hasn't
|
||||||
// been a matching shift yet in order to prevent an unhandledRejection
|
// been a matching shift yet in order to prevent an unhandledRejection
|
||||||
// error. Reject it again when there's a shift.
|
// error.
|
||||||
Promise.resolve(value).catch(() => {});
|
Promise.resolve(value).catch(() => {});
|
||||||
|
|
||||||
if (closed) {
|
if (closed) {
|
||||||
|
|
286
test/go/nonblock.js
Normal file
286
test/go/nonblock.js
Normal file
|
@ -0,0 +1,286 @@
|
||||||
|
// run
|
||||||
|
|
||||||
|
// Copyright 2009 The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Test channel operations that test for blocking.
|
||||||
|
// Use several sizes and types of operands.
|
||||||
|
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
const Channel = require(`../../lib`);
|
||||||
|
|
||||||
|
it(`doubleselect`, async function() {
|
||||||
|
const i32receiver = async (c, strobe) => {
|
||||||
|
if ((await c.shift()) != 123) {
|
||||||
|
throw new Error(`i32 value`);
|
||||||
|
}
|
||||||
|
|
||||||
|
await strobe.push(true);
|
||||||
|
};
|
||||||
|
});
|
||||||
|
/*
|
||||||
|
func i32sender(c chan int32, strobe chan bool) {
|
||||||
|
c <- 234
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func i64receiver(c chan int64, strobe chan bool) {
|
||||||
|
if <-c != 123456 {
|
||||||
|
panic("i64 value")
|
||||||
|
}
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func i64sender(c chan int64, strobe chan bool) {
|
||||||
|
c <- 234567
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func breceiver(c chan bool, strobe chan bool) {
|
||||||
|
if !<-c {
|
||||||
|
panic("b value")
|
||||||
|
}
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func bsender(c chan bool, strobe chan bool) {
|
||||||
|
c <- true
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func sreceiver(c chan string, strobe chan bool) {
|
||||||
|
if <-c != "hello" {
|
||||||
|
panic("s value")
|
||||||
|
}
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
func ssender(c chan string, strobe chan bool) {
|
||||||
|
c <- "hello again"
|
||||||
|
strobe <- true
|
||||||
|
}
|
||||||
|
|
||||||
|
var ticker = time.Tick(10 * 1000) // 10 us
|
||||||
|
func sleep() {
|
||||||
|
<-ticker
|
||||||
|
<-ticker
|
||||||
|
runtime.Gosched()
|
||||||
|
runtime.Gosched()
|
||||||
|
runtime.Gosched()
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxTries = 10000 // Up to 100ms per test.
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
var i32 int32
|
||||||
|
var i64 int64
|
||||||
|
var b bool
|
||||||
|
var s string
|
||||||
|
|
||||||
|
var sync = make(chan bool)
|
||||||
|
|
||||||
|
for buffer := 0; buffer < 2; buffer++ {
|
||||||
|
c32 := make(chan int32, buffer)
|
||||||
|
c64 := make(chan int64, buffer)
|
||||||
|
cb := make(chan bool, buffer)
|
||||||
|
cs := make(chan string, buffer)
|
||||||
|
|
||||||
|
select {
|
||||||
|
case i32 = <-c32:
|
||||||
|
panic("blocked i32sender")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case i64 = <-c64:
|
||||||
|
panic("blocked i64sender")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case b = <-cb:
|
||||||
|
panic("blocked bsender")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case s = <-cs:
|
||||||
|
panic("blocked ssender")
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
go i32receiver(c32, sync)
|
||||||
|
try := 0
|
||||||
|
Send32:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case c32 <- 123:
|
||||||
|
break Send32
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
println("i32receiver buffer=", buffer)
|
||||||
|
panic("fail")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<-sync
|
||||||
|
|
||||||
|
go i32sender(c32, sync)
|
||||||
|
if buffer > 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
try = 0
|
||||||
|
Recv32:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case i32 = <-c32:
|
||||||
|
break Recv32
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
println("i32sender buffer=", buffer)
|
||||||
|
panic("fail")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if i32 != 234 {
|
||||||
|
panic("i32sender value")
|
||||||
|
}
|
||||||
|
if buffer == 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
|
||||||
|
go i64receiver(c64, sync)
|
||||||
|
try = 0
|
||||||
|
Send64:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case c64 <- 123456:
|
||||||
|
break Send64
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("i64receiver")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<-sync
|
||||||
|
|
||||||
|
go i64sender(c64, sync)
|
||||||
|
if buffer > 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
try = 0
|
||||||
|
Recv64:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case i64 = <-c64:
|
||||||
|
break Recv64
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("i64sender")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if i64 != 234567 {
|
||||||
|
panic("i64sender value")
|
||||||
|
}
|
||||||
|
if buffer == 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
|
||||||
|
go breceiver(cb, sync)
|
||||||
|
try = 0
|
||||||
|
SendBool:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case cb <- true:
|
||||||
|
break SendBool
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("breceiver")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<-sync
|
||||||
|
|
||||||
|
go bsender(cb, sync)
|
||||||
|
if buffer > 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
try = 0
|
||||||
|
RecvBool:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case b = <-cb:
|
||||||
|
break RecvBool
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("bsender")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !b {
|
||||||
|
panic("bsender value")
|
||||||
|
}
|
||||||
|
if buffer == 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
|
||||||
|
go sreceiver(cs, sync)
|
||||||
|
try = 0
|
||||||
|
SendString:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case cs <- "hello":
|
||||||
|
break SendString
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("sreceiver")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<-sync
|
||||||
|
|
||||||
|
go ssender(cs, sync)
|
||||||
|
if buffer > 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
try = 0
|
||||||
|
RecvString:
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case s = <-cs:
|
||||||
|
break RecvString
|
||||||
|
default:
|
||||||
|
try++
|
||||||
|
if try > maxTries {
|
||||||
|
panic("ssender")
|
||||||
|
}
|
||||||
|
sleep()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if s != "hello again" {
|
||||||
|
panic("ssender value")
|
||||||
|
}
|
||||||
|
if buffer == 0 {
|
||||||
|
<-sync
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
*/
|
Loading…
Add table
Add a link
Reference in a new issue