Skip to content

Commit

Permalink
await websocket connection for flag check
Browse files Browse the repository at this point in the history
  • Loading branch information
bpapillon committed Oct 25, 2024
1 parent 4dba429 commit ecc0c61
Show file tree
Hide file tree
Showing 5 changed files with 378 additions and 40 deletions.
8 changes: 4 additions & 4 deletions js/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ schematic.identify({
schematic.track({ event: "query" });

// Check a flag
schematic.checkFlag("some-flag-key");
await schematic.checkFlag("some-flag-key");
```

By default, `checkFlag` will perform a network request to get the flag value for this user. If you'd like to check all flags at once in order to minimize network requests, you can use `checkFlags`:
Expand All @@ -61,22 +61,22 @@ schematic.identify({
},
});

schematic.checkFlags();
await schematic.checkFlags();
```

Alternatively, you can run in websocket mode, which will keep a persistent connection open to the Schematic service and receive flag updates in real time:

```typescript
import { Schematic } from "@schematichq/schematic-js";

const schematic = new Schematic("your-api-key", { useWebSocket: true});
const schematic = new Schematic("your-api-key", { useWebSocket: true });

schematic.identify({
keys: { id: "my-user-id" },
company: { keys: { id: "my-company-id" } },
});

schematic.checkFlag("some-flag-key");
await schematic.checkFlag("some-flag-key");
```

## License
Expand Down
4 changes: 3 additions & 1 deletion js/jest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@ module.exports = {
preset: "ts-jest/presets/js-with-ts-esm",
testEnvironment: "jsdom",
transform: {
'^.+\\.(ts|tsx)?$': 'ts-jest',
"^.+\\.(ts|tsx)?$": "ts-jest",
},
};

global.WebSocket = require("mock-socket").WebSocket;
3 changes: 2 additions & 1 deletion js/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@schematichq/schematic-js",
"version": "1.0.2",
"version": "1.0.3",
"main": "dist/schematic.cjs.js",
"module": "dist/schematic.esm.js",
"types": "dist/schematic.d.ts",
Expand Down Expand Up @@ -44,6 +44,7 @@
"jest-environment-jsdom": "^29.7.0",
"jest-esbuild": "^0.3.0",
"jest-fetch-mock": "^3.0.3",
"mock-socket": "^9.3.1",
"prettier": "^3.3.3",
"ts-jest": "^29.2.5",
"typescript": "^5.6.2"
Expand Down
252 changes: 252 additions & 0 deletions js/src/index.spec.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { Schematic } from "./index";
import { Server as WebSocketServer } from "mock-socket";

const mockFetch = jest.fn();
global.fetch = mockFetch;
Expand Down Expand Up @@ -351,3 +352,254 @@ describe("Schematic", () => {
});
});
});

describe("Schematic WebSocket", () => {
let schematic: Schematic;
let mockServer: WebSocketServer;
const TEST_WS_URL = "ws://localhost:1234";
const FULL_WS_URL = `${TEST_WS_URL}/flags/bootstrap`;

beforeEach(() => {
mockServer?.stop();
mockServer = new WebSocketServer(FULL_WS_URL);
schematic = new Schematic("API_KEY", {
useWebSocket: true,
webSocketUrl: TEST_WS_URL,
});
});

afterEach(async () => {
await schematic.cleanup();
mockServer.stop();
await new Promise((resolve) => setTimeout(resolve, 100));
mockFetch.mockClear();
});

it("should wait for WebSocket connection and initial message before returning flag value", async () => {
const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

mockServer.on("connection", (socket) => {
socket.on("message", (data) => {
const parsedData = JSON.parse(data.toString());
expect(parsedData).toEqual({
apiKey: "API_KEY",
data: context,
});

setTimeout(() => {
socket.send(
JSON.stringify({
flags: [
{
flag: "TEST_FLAG",
value: true,
},
],
}),
);
}, 100);
});
});

const flagValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});

expect(flagValue).toBe(true);
});

it("should handle connection closing and reopening", async () => {
const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

let connectionCount = 0;

mockServer.on("connection", (socket) => {
connectionCount++;
socket.on("message", () => {
socket.send(
JSON.stringify({
flags: [
{
flag: "TEST_FLAG",
value: true,
},
],
}),
);
});
});

const firstCheckResult = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});
expect(firstCheckResult).toBe(true);
expect(connectionCount).toBe(1);

mockServer.stop();
await schematic.cleanup();
await new Promise((resolve) => setTimeout(resolve, 500));

mockServer = new WebSocketServer(FULL_WS_URL);
mockServer.on("connection", (socket) => {
connectionCount++;
socket.on("message", () => {
socket.send(
JSON.stringify({
flags: [
{
flag: "TEST_FLAG",
value: true,
},
],
}),
);
});
});

const secondCheckResult = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});

expect(secondCheckResult).toBe(true);
expect(connectionCount).toBe(2);
}, 15000);

it("should fall back to REST API if WebSocket connection fails", async () => {
mockServer.stop();
await new Promise((resolve) => setTimeout(resolve, 100));

const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

// successful API response
mockFetch.mockImplementationOnce(() =>
Promise.resolve({
ok: true,
json: () =>
Promise.resolve({
data: {
value: true,
flag: "TEST_FLAG",
companyId: context.company?.companyId,
userId: context.user?.userId,
},
}),
}),
);

const flagValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});

expect(flagValue).toBe(true);
expect(mockFetch).toHaveBeenCalledTimes(1);
});

it("should return fallback value if REST API call fails", async () => {
mockServer.stop();
await new Promise((resolve) => setTimeout(resolve, 100));

const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

// API response with server error
mockFetch.mockImplementationOnce(() =>
Promise.resolve({
ok: false,
status: 500,
statusText: "Internal Server Error",
}),
);

const flagValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: true,
});

expect(flagValue).toBe(true); // fallback value
expect(mockFetch).toHaveBeenCalledTimes(1);
});

it("should return fallback value if REST API call throws", async () => {
mockServer.stop();
await new Promise((resolve) => setTimeout(resolve, 100));

const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

// network error
mockFetch.mockImplementationOnce(() =>
Promise.reject(new Error("Network error")),
);

const flagValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: true,
});

expect(flagValue).toBe(true); // fallback value
expect(mockFetch).toHaveBeenCalledTimes(1);
});

it("should use cached values for subsequent checks", async () => {
const context = {
company: { companyId: "456" },
user: { userId: "123" },
};

let messageCount = 0;
mockServer.on("connection", (socket) => {
socket.on("message", () => {
messageCount++;
socket.send(
JSON.stringify({
flags: [
{
flag: "TEST_FLAG",
value: true,
},
],
}),
);
});
});

const firstValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});

const secondValue = await schematic.checkFlag({
key: "TEST_FLAG",
context,
fallback: false,
});

expect(firstValue).toBe(true);
expect(secondValue).toBe(true);
expect(messageCount).toBe(1);
});
});
Loading

0 comments on commit ecc0c61

Please sign in to comment.