Skip to content
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
47 changes: 32 additions & 15 deletions packages/client/lib/client/commands-queue.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,17 +27,19 @@ export interface CommandToWrite extends CommandWaitingForReply {
signal: AbortSignal;
listener: () => unknown;
} | undefined;
timeout: {
signal: AbortSignal;
listener: () => unknown;
} | undefined;
}

interface CommandWaitingForReply {
resolve(reply?: unknown): void;
reject(err: unknown): void;
channelsCounter: number | undefined;
typeMapping: TypeMapping | undefined;
node?: DoublyLinkedNode<CommandWaitingForReply>
timeout: {
signal: AbortSignal;
toBeSent: boolean;
listener: () => unknown;
} | undefined;
}

export type OnShardedChannelMoved = (channel: string, listeners: ChannelListeners) => void;
Expand All @@ -54,7 +56,7 @@ export default class RedisCommandsQueue {
readonly #respVersion;
readonly #maxLength;
readonly #toWrite = new DoublyLinkedList<CommandToWrite>();
readonly #waitingForReply = new SinglyLinkedList<CommandWaitingForReply>();
readonly #waitingForReply = new DoublyLinkedList<CommandWaitingForReply>();
readonly #onShardedChannelMoved;
#chainInExecution: symbol | undefined;
readonly decoder;
Expand All @@ -78,11 +80,21 @@ export default class RedisCommandsQueue {
}

#onReply(reply: ReplyUnion) {
this.#waitingForReply.shift()!.resolve(reply);
const node = this.#waitingForReply.shift()!;
if (node.timeout) {
RedisCommandsQueue.#removeTimeoutListener(node)
node.timeout = undefined;
}
node.resolve(reply);
}

#onErrorReply(err: ErrorReply) {
this.#waitingForReply.shift()!.reject(err);
const node = this.#waitingForReply.shift()!;
if (node.timeout) {
RedisCommandsQueue.#removeTimeoutListener(node)
node.timeout = undefined;
}
node.reject(err);
}

#onPush(push: Array<any>) {
Expand Down Expand Up @@ -156,7 +168,6 @@ export default class RedisCommandsQueue {
}

return new Promise((resolve, reject) => {
let node: DoublyLinkedNode<CommandToWrite>;
const value: CommandToWrite = {
args,
chainId: options?.chainId,
Expand All @@ -173,9 +184,16 @@ export default class RedisCommandsQueue {
const signal = AbortSignal.timeout(timeout);
value.timeout = {
signal,
toBeSent: true,
listener: () => {
this.#toWrite.remove(node);
value.reject(new TimeoutError());
const reject = value.reject;
if (value.timeout!.toBeSent) {
this.#toWrite.remove(value.node as DoublyLinkedNode<CommandToWrite>);
} else {
value.resolve = () => {};
value.reject = () => {};
}
reject(new TimeoutError());
}
};
signal.addEventListener('abort', value.timeout.listener, { once: true });
Expand All @@ -186,14 +204,14 @@ export default class RedisCommandsQueue {
value.abort = {
signal,
listener: () => {
this.#toWrite.remove(node);
this.#toWrite.remove(value.node as DoublyLinkedNode<CommandToWrite>);
value.reject(new AbortError());
}
};
signal.addEventListener('abort', value.abort.listener, { once: true });
}

node = this.#toWrite.add(value, options?.asap);
value.node = this.#toWrite.add(value, options?.asap);
});
}

Expand Down Expand Up @@ -408,8 +426,7 @@ export default class RedisCommandsQueue {
toSend.abort = undefined;
}
if (toSend.timeout) {
RedisCommandsQueue.#removeTimeoutListener(toSend);
toSend.timeout = undefined;
toSend.timeout.toBeSent = false;
}
this.#chainInExecution = toSend.chainId;
toSend.chainId = undefined;
Expand All @@ -431,7 +448,7 @@ export default class RedisCommandsQueue {
command.abort!.signal.removeEventListener('abort', command.abort!.listener);
}

static #removeTimeoutListener(command: CommandToWrite) {
static #removeTimeoutListener(command: CommandToWrite | CommandWaitingForReply) {
command.timeout!.signal.removeEventListener('abort', command.timeout!.listener);
}

Expand Down