aspnetcore/clients/ts/signalr/tests/ServerSentEventsTransport.t...

194 lines
6.9 KiB
TypeScript

// Copyright (c) .NET Foundation. All rights reserved.
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
import { TransferFormat } from "../src/ITransport";
import { HttpClient, HttpRequest } from "../src/HttpClient";
import { ILogger } from "../src/ILogger";
import { ServerSentEventsTransport } from "../src/ServerSentEventsTransport";
import { VerifyLogger } from "./Common";
import { TestEventSource, TestMessageEvent } from "./TestEventSource";
import { TestHttpClient } from "./TestHttpClient";
import { registerUnhandledRejectionHandler } from "./Utils";
registerUnhandledRejectionHandler();
describe("ServerSentEventsTransport", () => {
it("does not allow non-text formats", async () => {
await VerifyLogger.run(async (logger) => {
const sse = new ServerSentEventsTransport(new TestHttpClient(), undefined, logger, true, TestEventSource);
await expect(sse.connect("", TransferFormat.Binary))
.rejects
.toEqual(new Error("The Server-Sent Events transport only supports the 'Text' transfer format"));
});
});
it("connect waits for EventSource to be connected", async () => {
await VerifyLogger.run(async (logger) => {
const sse = new ServerSentEventsTransport(new TestHttpClient(), undefined, logger, true, TestEventSource);
let connectComplete: boolean = false;
const connectPromise = (async () => {
await sse.connect("http://example.com", TransferFormat.Text);
connectComplete = true;
})();
await TestEventSource.eventSource.openSet;
expect(connectComplete).toBe(false);
TestEventSource.eventSource.onopen(new TestMessageEvent());
await connectPromise;
expect(connectComplete).toBe(true);
});
});
[["http://example.com", "http://example.com?access_token=secretToken"],
["http://example.com?value=null", "http://example.com?value=null&access_token=secretToken"]]
.forEach(([input, expected]) => {
it(`appends access_token to url ${input}`, async () => {
await VerifyLogger.run(async (logger) => {
await createAndStartSSE(logger, input, () => "secretToken");
expect(TestEventSource.eventSource.url).toBe(expected);
});
});
});
it("sets Authorization header on sends", async () => {
await VerifyLogger.run(async (logger) => {
let request: HttpRequest;
const httpClient = new TestHttpClient().on((r) => {
request = r;
return "";
});
const sse = await createAndStartSSE(logger, "http://example.com", () => "secretToken", httpClient);
await sse.send("");
expect(request!.headers!.Authorization).toBe("Bearer secretToken");
expect(request!.url).toBe("http://example.com");
});
});
it("can send data", async () => {
await VerifyLogger.run(async (logger) => {
let request: HttpRequest;
const httpClient = new TestHttpClient().on((r) => {
request = r;
return "";
});
const sse = await createAndStartSSE(logger, "http://example.com", undefined, httpClient);
await sse.send("send data");
expect(request!.content).toBe("send data");
});
});
it("can receive data", async () => {
await VerifyLogger.run(async (logger) => {
const sse = await createAndStartSSE(logger);
let received: string | ArrayBuffer;
sse.onreceive = (data) => {
received = data;
};
const message = new TestMessageEvent();
message.data = "receive data";
TestEventSource.eventSource.onmessage(message);
expect(typeof received!).toBe("string");
expect(received!).toBe("receive data");
});
});
it("stop closes EventSource and calls onclose", async () => {
await VerifyLogger.run(async (logger) => {
const sse = await createAndStartSSE(logger);
let closeCalled: boolean = false;
sse.onclose = () => {
closeCalled = true;
};
await sse.stop();
expect(closeCalled).toBe(true);
expect(TestEventSource.eventSource.closed).toBe(true);
});
});
it("can close from EventSource error", async () => {
await VerifyLogger.run(async (logger) => {
const sse = await createAndStartSSE(logger);
let closeCalled: boolean = false;
let error: Error | undefined;
sse.onclose = (e) => {
closeCalled = true;
error = e;
};
const errorEvent = new TestMessageEvent();
errorEvent.data = "error";
TestEventSource.eventSource.onerror(errorEvent);
expect(closeCalled).toBe(true);
expect(TestEventSource.eventSource.closed).toBe(true);
expect(error).toEqual(new Error("error"));
});
});
it("send throws if not connected", async () => {
await VerifyLogger.run(async (logger) => {
const sse = new ServerSentEventsTransport(new TestHttpClient(), undefined, logger, true, TestEventSource);
await expect(sse.send(""))
.rejects
.toEqual(new Error("Cannot send until the transport is connected"));
});
});
it("closes on error from receive", async () => {
await VerifyLogger.run(async (logger) => {
const sse = await createAndStartSSE(logger);
sse.onreceive = () => {
throw new Error("error parsing");
};
let closeCalled: boolean = false;
let error: Error | undefined;
sse.onclose = (e) => {
closeCalled = true;
error = e;
};
const errorEvent = new TestMessageEvent();
errorEvent.data = "some data";
TestEventSource.eventSource.onmessage(errorEvent);
expect(closeCalled).toBe(true);
expect(TestEventSource.eventSource.closed).toBe(true);
expect(error).toEqual(new Error("error parsing"));
});
});
});
async function createAndStartSSE(logger: ILogger, url?: string, accessTokenFactory?: (() => string | Promise<string>), httpClient?: HttpClient): Promise<ServerSentEventsTransport> {
const sse = new ServerSentEventsTransport(httpClient || new TestHttpClient(), accessTokenFactory, logger, true, TestEventSource);
const connectPromise = sse.connect(url || "http://example.com", TransferFormat.Text);
await TestEventSource.eventSource.openSet;
TestEventSource.eventSource.onopen(new TestMessageEvent());
await connectPromise;
return sse;
}