2026-03-10 19:19:23 +00:00
|
|
|
import { LocationSource } from "@aelis/source-location"
|
2026-03-16 01:30:02 +00:00
|
|
|
import { WeatherSource } from "@aelis/source-weatherkit"
|
2026-03-15 22:57:19 +00:00
|
|
|
import { describe, expect, mock, spyOn, test } from "bun:test"
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
import { UserSessionManager } from "./user-session-manager.ts"
|
|
|
|
|
|
2026-03-16 01:30:02 +00:00
|
|
|
const mockWeatherProvider = async () =>
|
|
|
|
|
new WeatherSource({ client: { fetch: async () => ({}) as never } })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
describe("UserSessionManager", () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
test("getOrCreate creates session on first call", async () => {
|
|
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(session).toBeDefined()
|
|
|
|
|
expect(session.engine).toBeDefined()
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("getOrCreate returns same session for same user", async () => {
|
|
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session1 = await manager.getOrCreate("user-1")
|
|
|
|
|
const session2 = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(session1).toBe(session2)
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("getOrCreate returns different sessions for different users", async () => {
|
|
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session1 = await manager.getOrCreate("user-1")
|
|
|
|
|
const session2 = await manager.getOrCreate("user-2")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(session1).not.toBe(session2)
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("each user gets independent source instances", async () => {
|
|
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session1 = await manager.getOrCreate("user-1")
|
|
|
|
|
const session2 = await manager.getOrCreate("user-2")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-10 19:19:23 +00:00
|
|
|
const source1 = session1.getSource<LocationSource>("aelis.location")
|
|
|
|
|
const source2 = session2.getSource<LocationSource>("aelis.location")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(source1).not.toBe(source2)
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("remove destroys session and allows re-creation", async () => {
|
|
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session1 = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
manager.remove("user-1")
|
2026-03-15 22:57:19 +00:00
|
|
|
const session2 = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(session1).not.toBe(session2)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("remove is no-op for unknown user", () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
|
|
|
|
expect(() => manager.remove("unknown")).not.toThrow()
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("accepts function providers", async () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
const result = await session.engine.refresh()
|
|
|
|
|
|
|
|
|
|
expect(result.errors).toHaveLength(0)
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("accepts object providers", async () => {
|
2026-03-05 02:01:30 +00:00
|
|
|
const manager = new UserSessionManager({
|
2026-03-16 01:30:02 +00:00
|
|
|
providers: [async () => new LocationSource(), mockWeatherProvider],
|
2026-03-05 02:01:30 +00:00
|
|
|
})
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-10 19:19:23 +00:00
|
|
|
expect(session.getSource("aelis.weather")).toBeDefined()
|
2026-02-18 00:41:20 +00:00
|
|
|
})
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
test("accepts mixed providers", async () => {
|
2026-03-05 02:01:30 +00:00
|
|
|
const manager = new UserSessionManager({
|
2026-03-16 01:30:02 +00:00
|
|
|
providers: [async () => new LocationSource(), mockWeatherProvider],
|
2026-03-05 02:01:30 +00:00
|
|
|
})
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-10 19:19:23 +00:00
|
|
|
expect(session.getSource("aelis.location")).toBeDefined()
|
|
|
|
|
expect(session.getSource("aelis.weather")).toBeDefined()
|
2026-02-18 00:41:20 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("refresh returns feed result through session", async () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
const result = await session.engine.refresh()
|
|
|
|
|
|
|
|
|
|
expect(result).toHaveProperty("context")
|
|
|
|
|
expect(result).toHaveProperty("items")
|
|
|
|
|
expect(result).toHaveProperty("errors")
|
|
|
|
|
expect(result.context.time).toBeInstanceOf(Date)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("location update via executeAction works", async () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-03-10 19:19:23 +00:00
|
|
|
await session.engine.executeAction("aelis.location", "update-location", {
|
2026-02-18 00:41:20 +00:00
|
|
|
lat: 51.5074,
|
|
|
|
|
lng: -0.1278,
|
|
|
|
|
accuracy: 10,
|
|
|
|
|
timestamp: new Date(),
|
|
|
|
|
})
|
|
|
|
|
|
2026-03-10 19:19:23 +00:00
|
|
|
const source = session.getSource<LocationSource>("aelis.location")
|
2026-02-18 00:41:20 +00:00
|
|
|
expect(source?.lastLocation?.lat).toBe(51.5074)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("subscribe receives updates after location push", async () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
const callback = mock()
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
session.engine.subscribe(callback)
|
|
|
|
|
|
2026-03-10 19:19:23 +00:00
|
|
|
await session.engine.executeAction("aelis.location", "update-location", {
|
2026-02-18 00:41:20 +00:00
|
|
|
lat: 51.5074,
|
|
|
|
|
lng: -0.1278,
|
|
|
|
|
accuracy: 10,
|
|
|
|
|
timestamp: new Date(),
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// Wait for async update propagation
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
|
|
|
|
|
|
|
|
expect(callback).toHaveBeenCalled()
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("remove stops reactive updates", async () => {
|
2026-03-15 22:57:19 +00:00
|
|
|
const manager = new UserSessionManager({ providers: [async () => new LocationSource()] })
|
2026-02-18 00:41:20 +00:00
|
|
|
const callback = mock()
|
|
|
|
|
|
2026-03-15 22:57:19 +00:00
|
|
|
const session = await manager.getOrCreate("user-1")
|
2026-02-18 00:41:20 +00:00
|
|
|
session.engine.subscribe(callback)
|
|
|
|
|
|
|
|
|
|
manager.remove("user-1")
|
|
|
|
|
|
|
|
|
|
// Create new session and push location — old callback should not fire
|
2026-03-15 22:57:19 +00:00
|
|
|
const session2 = await manager.getOrCreate("user-1")
|
2026-03-10 19:19:23 +00:00
|
|
|
await session2.engine.executeAction("aelis.location", "update-location", {
|
2026-02-18 00:41:20 +00:00
|
|
|
lat: 51.5074,
|
|
|
|
|
lng: -0.1278,
|
|
|
|
|
accuracy: 10,
|
|
|
|
|
timestamp: new Date(),
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
|
|
|
|
|
|
|
|
expect(callback).not.toHaveBeenCalled()
|
|
|
|
|
})
|
2026-03-15 22:57:19 +00:00
|
|
|
|
|
|
|
|
test("creates session with successful providers when some fail", async () => {
|
|
|
|
|
const manager = new UserSessionManager({
|
|
|
|
|
providers: [
|
|
|
|
|
async () => new LocationSource(),
|
|
|
|
|
async () => {
|
|
|
|
|
throw new Error("provider failed")
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const spy = spyOn(console, "error").mockImplementation(() => {})
|
|
|
|
|
|
|
|
|
|
const session = await manager.getOrCreate("user-1")
|
|
|
|
|
|
|
|
|
|
expect(session).toBeDefined()
|
|
|
|
|
expect(session.getSource("aelis.location")).toBeDefined()
|
|
|
|
|
expect(spy).toHaveBeenCalled()
|
|
|
|
|
|
|
|
|
|
spy.mockRestore()
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("throws AggregateError when all providers fail", async () => {
|
|
|
|
|
const manager = new UserSessionManager({
|
|
|
|
|
providers: [
|
|
|
|
|
async () => {
|
|
|
|
|
throw new Error("first failed")
|
|
|
|
|
},
|
|
|
|
|
async () => {
|
|
|
|
|
throw new Error("second failed")
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
await expect(manager.getOrCreate("user-1")).rejects.toBeInstanceOf(AggregateError)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("concurrent getOrCreate for same user returns same session", async () => {
|
|
|
|
|
let callCount = 0
|
|
|
|
|
const manager = new UserSessionManager({
|
|
|
|
|
providers: [
|
|
|
|
|
async () => {
|
|
|
|
|
callCount++
|
|
|
|
|
// Simulate async work to widen the race window
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
|
|
|
return new LocationSource()
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const [session1, session2] = await Promise.all([
|
|
|
|
|
manager.getOrCreate("user-1"),
|
|
|
|
|
manager.getOrCreate("user-1"),
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
expect(session1).toBe(session2)
|
|
|
|
|
expect(callCount).toBe(1)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
test("remove during in-flight getOrCreate prevents session from being stored", async () => {
|
|
|
|
|
let resolveProvider: () => void
|
|
|
|
|
const providerGate = new Promise<void>((r) => {
|
|
|
|
|
resolveProvider = r
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const manager = new UserSessionManager({
|
|
|
|
|
providers: [
|
|
|
|
|
async () => {
|
|
|
|
|
await providerGate
|
|
|
|
|
return new LocationSource()
|
|
|
|
|
},
|
|
|
|
|
],
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const sessionPromise = manager.getOrCreate("user-1")
|
|
|
|
|
|
|
|
|
|
// remove() while provider is still resolving
|
|
|
|
|
manager.remove("user-1")
|
|
|
|
|
|
|
|
|
|
// Let the provider finish
|
|
|
|
|
resolveProvider!()
|
|
|
|
|
|
|
|
|
|
await expect(sessionPromise).rejects.toThrow("removed during creation")
|
|
|
|
|
|
|
|
|
|
// A fresh getOrCreate should produce a new session, not the cancelled one
|
|
|
|
|
const freshSession = await manager.getOrCreate("user-1")
|
|
|
|
|
expect(freshSession).toBeDefined()
|
|
|
|
|
expect(freshSession.engine).toBeDefined()
|
|
|
|
|
})
|
2026-02-18 00:41:20 +00:00
|
|
|
})
|