mirror of
https://github.com/kennethnym/aris.git
synced 2026-04-12 21:01:19 +01:00
Compare commits
5 Commits
kn/per-use
...
fix/upsert
| Author | SHA1 | Date | |
|---|---|---|---|
|
dcabc4fbdb
|
|||
| b5236e0e52 | |||
| 0a8243c55b | |||
| 400055ab8c | |||
| 98ce546eff |
3
.vscode/settings.json
vendored
Normal file
3
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
{
|
||||||
|
"js/ts.experimental.useTsgo": true
|
||||||
|
}
|
||||||
@@ -34,7 +34,7 @@
|
|||||||
"@types/react": "^19.2.5",
|
"@types/react": "^19.2.5",
|
||||||
"@types/react-dom": "^19.2.3",
|
"@types/react-dom": "^19.2.3",
|
||||||
"@vitejs/plugin-react": "^5.1.1",
|
"@vitejs/plugin-react": "^5.1.1",
|
||||||
"typescript": "~5.9.3",
|
"typescript": "^6",
|
||||||
"vite": "^7.2.4"
|
"vite": "^7.2.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,7 +20,13 @@ import {
|
|||||||
import { Separator } from "@/components/ui/separator"
|
import { Separator } from "@/components/ui/separator"
|
||||||
import { Switch } from "@/components/ui/switch"
|
import { Switch } from "@/components/ui/switch"
|
||||||
import { Tooltip, TooltipContent, TooltipTrigger } from "@/components/ui/tooltip"
|
import { Tooltip, TooltipContent, TooltipTrigger } from "@/components/ui/tooltip"
|
||||||
import { fetchSourceConfig, pushLocation, replaceSource, updateProviderConfig } from "@/lib/api"
|
import {
|
||||||
|
fetchSourceConfig,
|
||||||
|
pushLocation,
|
||||||
|
replaceSource,
|
||||||
|
updateProviderConfig,
|
||||||
|
updateSourceCredentials,
|
||||||
|
} from "@/lib/api"
|
||||||
|
|
||||||
interface SourceConfigPanelProps {
|
interface SourceConfigPanelProps {
|
||||||
source: SourceDefinition
|
source: SourceDefinition
|
||||||
@@ -83,7 +89,11 @@ export function SourceConfigPanel({ source, onUpdate }: SourceConfigPanelProps)
|
|||||||
(v) => typeof v === "string" && v.length > 0,
|
(v) => typeof v === "string" && v.length > 0,
|
||||||
)
|
)
|
||||||
if (hasCredentials) {
|
if (hasCredentials) {
|
||||||
promises.push(updateProviderConfig(source.id, { credentials: credentialFields }))
|
if (source.perUserCredentials) {
|
||||||
|
promises.push(updateSourceCredentials(source.id, credentialFields))
|
||||||
|
} else {
|
||||||
|
promises.push(updateProviderConfig(source.id, { credentials: credentialFields }))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await Promise.all(promises)
|
await Promise.all(promises)
|
||||||
|
|||||||
@@ -23,6 +23,8 @@ export interface SourceDefinition {
|
|||||||
name: string
|
name: string
|
||||||
description: string
|
description: string
|
||||||
alwaysEnabled?: boolean
|
alwaysEnabled?: boolean
|
||||||
|
/** When true, secret fields are stored as per-user credentials via /api/sources/:id/credentials. */
|
||||||
|
perUserCredentials?: boolean
|
||||||
fields: Record<string, ConfigFieldDef>
|
fields: Record<string, ConfigFieldDef>
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -78,6 +80,44 @@ const sourceDefinitions: SourceDefinition[] = [
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
id: "aelis.caldav",
|
||||||
|
name: "CalDAV",
|
||||||
|
description: "Calendar events from any CalDAV server (Nextcloud, Radicale, Baikal, etc.).",
|
||||||
|
perUserCredentials: true,
|
||||||
|
fields: {
|
||||||
|
serverUrl: {
|
||||||
|
type: "string",
|
||||||
|
label: "Server URL",
|
||||||
|
required: true,
|
||||||
|
secret: false,
|
||||||
|
description: "CalDAV server URL (e.g. https://nextcloud.example.com/remote.php/dav)",
|
||||||
|
},
|
||||||
|
username: {
|
||||||
|
type: "string",
|
||||||
|
label: "Username",
|
||||||
|
required: true,
|
||||||
|
secret: false,
|
||||||
|
},
|
||||||
|
password: {
|
||||||
|
type: "string",
|
||||||
|
label: "Password",
|
||||||
|
required: true,
|
||||||
|
secret: true,
|
||||||
|
},
|
||||||
|
lookAheadDays: {
|
||||||
|
type: "number",
|
||||||
|
label: "Look-ahead Days",
|
||||||
|
defaultValue: 0,
|
||||||
|
description: "Number of additional days beyond today to fetch events for",
|
||||||
|
},
|
||||||
|
timeZone: {
|
||||||
|
type: "string",
|
||||||
|
label: "Timezone",
|
||||||
|
description: "IANA timezone for determining \"today\" (e.g. Europe/London). Defaults to UTC.",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
id: "aelis.tfl",
|
id: "aelis.tfl",
|
||||||
name: "TfL",
|
name: "TfL",
|
||||||
@@ -164,6 +204,22 @@ export async function updateProviderConfig(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function updateSourceCredentials(
|
||||||
|
sourceId: string,
|
||||||
|
credentials: Record<string, unknown>,
|
||||||
|
): Promise<void> {
|
||||||
|
const res = await fetch(`${serverBase()}/sources/${sourceId}/credentials`, {
|
||||||
|
method: "PUT",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
credentials: "include",
|
||||||
|
body: JSON.stringify(credentials),
|
||||||
|
})
|
||||||
|
if (!res.ok) {
|
||||||
|
const data = (await res.json()) as { error?: string }
|
||||||
|
throw new Error(data.error ?? `Failed to update credentials: ${res.status}`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export interface LocationInput {
|
export interface LocationInput {
|
||||||
lat: number
|
lat: number
|
||||||
lng: number
|
lng: number
|
||||||
|
|||||||
@@ -3,12 +3,11 @@
|
|||||||
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
|
||||||
"target": "ES2022",
|
"target": "ES2022",
|
||||||
"useDefineForClassFields": true,
|
"useDefineForClassFields": true,
|
||||||
"lib": ["ES2022", "DOM", "DOM.Iterable"],
|
"lib": ["ES2022", "DOM"],
|
||||||
"module": "ESNext",
|
"module": "ESNext",
|
||||||
"types": ["vite/client"],
|
"types": ["vite/client"],
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
|
|
||||||
/* Bundler mode */
|
|
||||||
"moduleResolution": "bundler",
|
"moduleResolution": "bundler",
|
||||||
"allowImportingTsExtensions": true,
|
"allowImportingTsExtensions": true,
|
||||||
"verbatimModuleSyntax": true,
|
"verbatimModuleSyntax": true,
|
||||||
@@ -16,14 +15,12 @@
|
|||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
"jsx": "react-jsx",
|
"jsx": "react-jsx",
|
||||||
|
|
||||||
/* Linting */
|
|
||||||
"strict": true,
|
"strict": true,
|
||||||
"noUnusedLocals": true,
|
"noUnusedLocals": true,
|
||||||
"noUnusedParameters": true,
|
"noUnusedParameters": true,
|
||||||
"erasableSyntaxOnly": true,
|
"erasableSyntaxOnly": true,
|
||||||
"noFallthroughCasesInSwitch": true,
|
"noFallthroughCasesInSwitch": true,
|
||||||
"noUncheckedSideEffectImports": true,
|
"noUncheckedSideEffectImports": true,
|
||||||
"baseUrl": ".",
|
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["./src/*"]
|
"@/*": ["./src/*"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
"files": [],
|
"files": [],
|
||||||
"references": [{ "path": "./tsconfig.app.json" }, { "path": "./tsconfig.node.json" }],
|
"references": [{ "path": "./tsconfig.app.json" }, { "path": "./tsconfig.node.json" }],
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"baseUrl": ".",
|
|
||||||
"paths": {
|
"paths": {
|
||||||
"@/*": ["./src/*"]
|
"@/*": ["./src/*"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,14 +7,12 @@
|
|||||||
"types": ["node"],
|
"types": ["node"],
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
|
|
||||||
/* Bundler mode */
|
|
||||||
"moduleResolution": "bundler",
|
"moduleResolution": "bundler",
|
||||||
"allowImportingTsExtensions": true,
|
"allowImportingTsExtensions": true,
|
||||||
"verbatimModuleSyntax": true,
|
"verbatimModuleSyntax": true,
|
||||||
"moduleDetection": "force",
|
"moduleDetection": "force",
|
||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
|
|
||||||
/* Linting */
|
|
||||||
"strict": true,
|
"strict": true,
|
||||||
"noUnusedLocals": true,
|
"noUnusedLocals": true,
|
||||||
"noUnusedParameters": true,
|
"noUnusedParameters": true,
|
||||||
|
|||||||
85
apps/aelis-backend/src/caldav/provider.test.ts
Normal file
85
apps/aelis-backend/src/caldav/provider.test.ts
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
import { describe, expect, test } from "bun:test"
|
||||||
|
|
||||||
|
import { CalDavSourceProvider } from "./provider.ts"
|
||||||
|
|
||||||
|
describe("CalDavSourceProvider", () => {
|
||||||
|
const provider = new CalDavSourceProvider()
|
||||||
|
|
||||||
|
test("sourceId is aelis.caldav", () => {
|
||||||
|
expect(provider.sourceId).toBe("aelis.caldav")
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when credentials are null", async () => {
|
||||||
|
const config = { serverUrl: "https://caldav.icloud.com", username: "user@icloud.com" }
|
||||||
|
await expect(provider.feedSourceForUser("user-1", config, null)).rejects.toThrow(
|
||||||
|
"No CalDAV credentials configured",
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when credentials are missing password", async () => {
|
||||||
|
const config = { serverUrl: "https://caldav.icloud.com", username: "user@icloud.com" }
|
||||||
|
await expect(provider.feedSourceForUser("user-1", config, {})).rejects.toThrow(
|
||||||
|
"password must be a string",
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when config is missing serverUrl", async () => {
|
||||||
|
const credentials = { password: "app-specific-password" }
|
||||||
|
await expect(
|
||||||
|
provider.feedSourceForUser("user-1", { username: "user@icloud.com" }, credentials),
|
||||||
|
).rejects.toThrow("Invalid CalDAV config")
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when config is missing username", async () => {
|
||||||
|
const credentials = { password: "app-specific-password" }
|
||||||
|
await expect(
|
||||||
|
provider.feedSourceForUser("user-1", { serverUrl: "https://caldav.icloud.com" }, credentials),
|
||||||
|
).rejects.toThrow("Invalid CalDAV config")
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when config has extra keys", async () => {
|
||||||
|
const config = {
|
||||||
|
serverUrl: "https://caldav.icloud.com",
|
||||||
|
username: "user@icloud.com",
|
||||||
|
extra: true,
|
||||||
|
}
|
||||||
|
const credentials = { password: "app-specific-password" }
|
||||||
|
await expect(provider.feedSourceForUser("user-1", config, credentials)).rejects.toThrow(
|
||||||
|
"Invalid CalDAV config",
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws when credentials have extra keys", async () => {
|
||||||
|
const config = { serverUrl: "https://caldav.icloud.com", username: "user@icloud.com" }
|
||||||
|
const credentials = { password: "app-specific-password", extra: true }
|
||||||
|
await expect(provider.feedSourceForUser("user-1", config, credentials)).rejects.toThrow(
|
||||||
|
"extra must be removed",
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
test("returns CalDavSource with valid config and credentials", async () => {
|
||||||
|
const config = {
|
||||||
|
serverUrl: "https://caldav.icloud.com",
|
||||||
|
username: "user@icloud.com",
|
||||||
|
lookAheadDays: 3,
|
||||||
|
timeZone: "Europe/London",
|
||||||
|
}
|
||||||
|
const credentials = { password: "app-specific-password" }
|
||||||
|
|
||||||
|
const source = await provider.feedSourceForUser("user-1", config, credentials)
|
||||||
|
expect(source).toBeDefined()
|
||||||
|
expect(source.id).toBe("aelis.caldav")
|
||||||
|
})
|
||||||
|
|
||||||
|
test("returns CalDavSource with minimal config", async () => {
|
||||||
|
const config = {
|
||||||
|
serverUrl: "https://caldav.icloud.com",
|
||||||
|
username: "user@icloud.com",
|
||||||
|
}
|
||||||
|
const credentials = { password: "app-specific-password" }
|
||||||
|
|
||||||
|
const source = await provider.feedSourceForUser("user-1", config, credentials)
|
||||||
|
expect(source).toBeDefined()
|
||||||
|
expect(source.id).toBe("aelis.caldav")
|
||||||
|
})
|
||||||
|
})
|
||||||
53
apps/aelis-backend/src/caldav/provider.ts
Normal file
53
apps/aelis-backend/src/caldav/provider.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { CalDavSource } from "@aelis/source-caldav"
|
||||||
|
import { type } from "arktype"
|
||||||
|
|
||||||
|
import type { FeedSourceProvider } from "../session/feed-source-provider.ts"
|
||||||
|
|
||||||
|
import { InvalidSourceCredentialsError } from "../sources/errors.ts"
|
||||||
|
|
||||||
|
const caldavConfig = type({
|
||||||
|
"+": "reject",
|
||||||
|
serverUrl: "string",
|
||||||
|
username: "string",
|
||||||
|
"lookAheadDays?": "number",
|
||||||
|
"timeZone?": "string",
|
||||||
|
})
|
||||||
|
|
||||||
|
const caldavCredentials = type({
|
||||||
|
"+": "reject",
|
||||||
|
password: "string",
|
||||||
|
})
|
||||||
|
|
||||||
|
export class CalDavSourceProvider implements FeedSourceProvider {
|
||||||
|
readonly sourceId = "aelis.caldav"
|
||||||
|
readonly configSchema = caldavConfig
|
||||||
|
|
||||||
|
async feedSourceForUser(
|
||||||
|
_userId: string,
|
||||||
|
config: unknown,
|
||||||
|
credentials: unknown,
|
||||||
|
): Promise<CalDavSource> {
|
||||||
|
const parsed = caldavConfig(config)
|
||||||
|
if (parsed instanceof type.errors) {
|
||||||
|
throw new Error(`Invalid CalDAV config: ${parsed.summary}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!credentials) {
|
||||||
|
throw new InvalidSourceCredentialsError("aelis.caldav", "No CalDAV credentials configured")
|
||||||
|
}
|
||||||
|
|
||||||
|
const creds = caldavCredentials(credentials)
|
||||||
|
if (creds instanceof type.errors) {
|
||||||
|
throw new InvalidSourceCredentialsError("aelis.caldav", creds.summary)
|
||||||
|
}
|
||||||
|
|
||||||
|
return new CalDavSource({
|
||||||
|
serverUrl: parsed.serverUrl,
|
||||||
|
authMethod: "basic",
|
||||||
|
username: parsed.username,
|
||||||
|
password: creds.password,
|
||||||
|
lookAheadDays: parsed.lookAheadDays,
|
||||||
|
timeZone: parsed.timeZone,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ import { createRequireAdmin } from "./auth/admin-middleware.ts"
|
|||||||
import { registerAuthHandlers } from "./auth/http.ts"
|
import { registerAuthHandlers } from "./auth/http.ts"
|
||||||
import { createAuth } from "./auth/index.ts"
|
import { createAuth } from "./auth/index.ts"
|
||||||
import { createRequireSession } from "./auth/session-middleware.ts"
|
import { createRequireSession } from "./auth/session-middleware.ts"
|
||||||
|
import { CalDavSourceProvider } from "./caldav/provider.ts"
|
||||||
import { createDatabase } from "./db/index.ts"
|
import { createDatabase } from "./db/index.ts"
|
||||||
import { registerFeedHttpHandlers } from "./engine/http.ts"
|
import { registerFeedHttpHandlers } from "./engine/http.ts"
|
||||||
import { createFeedEnhancer } from "./enhancement/enhance-feed.ts"
|
import { createFeedEnhancer } from "./enhancement/enhance-feed.ts"
|
||||||
@@ -48,6 +49,7 @@ function main() {
|
|||||||
const sessionManager = new UserSessionManager({
|
const sessionManager = new UserSessionManager({
|
||||||
db,
|
db,
|
||||||
providers: [
|
providers: [
|
||||||
|
new CalDavSourceProvider(),
|
||||||
new LocationSourceProvider(),
|
new LocationSourceProvider(),
|
||||||
new WeatherSourceProvider({
|
new WeatherSourceProvider({
|
||||||
credentials: {
|
credentials: {
|
||||||
|
|||||||
@@ -44,6 +44,15 @@ function getEnabledSourceIds(userId: string): string[] {
|
|||||||
*/
|
*/
|
||||||
let mockFindResult: unknown | undefined
|
let mockFindResult: unknown | undefined
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Spy for `upsertConfig` calls. Tests can inspect calls via
|
||||||
|
* `mockUpsertConfigCalls`.
|
||||||
|
*/
|
||||||
|
const mockUpsertConfigCalls: Array<{
|
||||||
|
sourceId: string
|
||||||
|
data: { enabled: boolean; config: unknown }
|
||||||
|
}> = []
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Spy for `updateCredentials` calls. Tests can inspect calls via
|
* Spy for `updateCredentials` calls. Tests can inspect calls via
|
||||||
* `mockUpdateCredentialsCalls` or override behavior.
|
* `mockUpdateCredentialsCalls` or override behavior.
|
||||||
@@ -81,6 +90,9 @@ mock.module("../sources/user-sources.ts", () => ({
|
|||||||
updatedAt: now,
|
updatedAt: now,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
async upsertConfig(sourceId: string, data: { enabled: boolean; config: unknown }) {
|
||||||
|
mockUpsertConfigCalls.push({ sourceId, data })
|
||||||
|
},
|
||||||
async updateCredentials(sourceId: string, credentials: Buffer) {
|
async updateCredentials(sourceId: string, credentials: Buffer) {
|
||||||
if (mockUpdateCredentialsError) {
|
if (mockUpdateCredentialsError) {
|
||||||
throw mockUpdateCredentialsError
|
throw mockUpdateCredentialsError
|
||||||
@@ -138,6 +150,7 @@ const weatherProvider: FeedSourceProvider = {
|
|||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
enabledByUser.clear()
|
enabledByUser.clear()
|
||||||
mockFindResult = undefined
|
mockFindResult = undefined
|
||||||
|
mockUpsertConfigCalls.length = 0
|
||||||
mockUpdateCredentialsCalls.length = 0
|
mockUpdateCredentialsCalls.length = 0
|
||||||
mockUpdateCredentialsError = null
|
mockUpdateCredentialsError = null
|
||||||
})
|
})
|
||||||
@@ -824,3 +837,65 @@ describe("UserSessionManager.updateSourceCredentials", () => {
|
|||||||
expect(factory).not.toHaveBeenCalled()
|
expect(factory).not.toHaveBeenCalled()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe("UserSessionManager.upsertSourceConfig", () => {
|
||||||
|
test("persists config to DB even when feedSourceForUser throws", async () => {
|
||||||
|
setEnabledSources(["test"])
|
||||||
|
let callCount = 0
|
||||||
|
const factory = mock(async (_userId: string, _config: unknown, _credentials: unknown) => {
|
||||||
|
callCount++
|
||||||
|
// Succeed on first call (session creation), throw on second (upsert refresh)
|
||||||
|
if (callCount > 1) {
|
||||||
|
throw new InvalidSourceCredentialsError("test", "credentials required")
|
||||||
|
}
|
||||||
|
return createStubSource("test")
|
||||||
|
})
|
||||||
|
const provider: FeedSourceProvider = { sourceId: "test", feedSourceForUser: factory }
|
||||||
|
const manager = new UserSessionManager({ db: fakeDb, providers: [provider] })
|
||||||
|
|
||||||
|
// Create a session so the session-refresh path is exercised
|
||||||
|
await manager.getOrCreate("user-1")
|
||||||
|
|
||||||
|
const spy = spyOn(console, "warn").mockImplementation(() => {})
|
||||||
|
|
||||||
|
// upsertSourceConfig with no existing credentials — provider will throw
|
||||||
|
await manager.upsertSourceConfig("user-1", "test", {
|
||||||
|
enabled: true,
|
||||||
|
config: { url: "https://example.com" },
|
||||||
|
})
|
||||||
|
|
||||||
|
// Config should still have been persisted to DB
|
||||||
|
expect(mockUpsertConfigCalls).toHaveLength(1)
|
||||||
|
expect(mockUpsertConfigCalls[0]!.sourceId).toBe("test")
|
||||||
|
expect(mockUpsertConfigCalls[0]!.data.enabled).toBe(true)
|
||||||
|
|
||||||
|
// The error should have been logged, not thrown
|
||||||
|
expect(spy).toHaveBeenCalled()
|
||||||
|
|
||||||
|
spy.mockRestore()
|
||||||
|
})
|
||||||
|
|
||||||
|
test("adds source to session when feedSourceForUser succeeds", async () => {
|
||||||
|
setEnabledSources(["test"])
|
||||||
|
const factory = mock(async () => createStubSource("test"))
|
||||||
|
const provider: FeedSourceProvider = { sourceId: "test", feedSourceForUser: factory }
|
||||||
|
const manager = new UserSessionManager({ db: fakeDb, providers: [provider] })
|
||||||
|
|
||||||
|
const session = await manager.getOrCreate("user-1")
|
||||||
|
await manager.upsertSourceConfig("user-1", "test", { enabled: true })
|
||||||
|
|
||||||
|
// Config persisted
|
||||||
|
expect(mockUpsertConfigCalls).toHaveLength(1)
|
||||||
|
// Source should be in the session (feedSourceForUser succeeded)
|
||||||
|
expect(session.getSource("test")).toBeDefined()
|
||||||
|
})
|
||||||
|
|
||||||
|
test("throws SourceNotFoundError for unknown provider", async () => {
|
||||||
|
setEnabledSources([])
|
||||||
|
const manager = new UserSessionManager({ db: fakeDb, providers: [] })
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
manager.upsertSourceConfig("user-1", "unknown", { enabled: true }),
|
||||||
|
).rejects.toBeInstanceOf(SourceNotFoundError)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|||||||
@@ -210,11 +210,22 @@ export class UserSessionManager {
|
|||||||
const credentials = existingRow?.credentials
|
const credentials = existingRow?.credentials
|
||||||
? this.decryptCredentials(existingRow.credentials)
|
? this.decryptCredentials(existingRow.credentials)
|
||||||
: null
|
: null
|
||||||
const source = await provider.feedSourceForUser(userId, config, credentials)
|
try {
|
||||||
if (session.hasSource(sourceId)) {
|
const source = await provider.feedSourceForUser(userId, config, credentials)
|
||||||
session.replaceSource(sourceId, source)
|
if (session.hasSource(sourceId)) {
|
||||||
} else {
|
session.replaceSource(sourceId, source)
|
||||||
session.addSource(source)
|
} else {
|
||||||
|
session.addSource(source)
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
// Provider may fail when credentials are not yet available (e.g. new
|
||||||
|
// source added before updateSourceCredentials is called). The config
|
||||||
|
// is already persisted above; updateSourceCredentials will add the
|
||||||
|
// source to the session later.
|
||||||
|
console.warn(
|
||||||
|
`[UserSessionManager] feedSourceForUser("${sourceId}") failed during upsert for user ${userId}, skipping session update:`,
|
||||||
|
err,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -55,6 +55,6 @@
|
|||||||
"eas-cli": "^18.0.1",
|
"eas-cli": "^18.0.1",
|
||||||
"eslint": "^9.25.0",
|
"eslint": "^9.25.0",
|
||||||
"eslint-config-expo": "~10.0.0",
|
"eslint-config-expo": "~10.0.0",
|
||||||
"typescript": "~5.9.2"
|
"typescript": "^6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,7 +30,7 @@
|
|||||||
"@types/react": "^19.2.7",
|
"@types/react": "^19.2.7",
|
||||||
"@types/react-dom": "^19.2.3",
|
"@types/react-dom": "^19.2.3",
|
||||||
"tailwindcss": "^4.1.13",
|
"tailwindcss": "^4.1.13",
|
||||||
"typescript": "^5.9.2",
|
"typescript": "^6",
|
||||||
"vite": "^7.1.7",
|
"vite": "^7.1.7",
|
||||||
"vite-tsconfig-paths": "^5.1.4"
|
"vite-tsconfig-paths": "^5.1.4"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,16 @@
|
|||||||
{
|
{
|
||||||
"include": ["**/*", "**/.server/**/*", "**/.client/**/*", ".react-router/types/**/*"],
|
"include": ["**/*", "**/.server/**/*", "**/.client/**/*", ".react-router/types/**/*"],
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"lib": ["DOM", "DOM.Iterable", "ES2022"],
|
"lib": ["DOM", "ES2022"],
|
||||||
"types": ["node", "vite/client"],
|
"types": ["node", "vite/client"],
|
||||||
"target": "ES2022",
|
"target": "ES2022",
|
||||||
"module": "ES2022",
|
"module": "ES2022",
|
||||||
"moduleResolution": "bundler",
|
"moduleResolution": "bundler",
|
||||||
"jsx": "react-jsx",
|
"jsx": "react-jsx",
|
||||||
"rootDirs": [".", "./.react-router/types"],
|
"rootDirs": [".", "./.react-router/types"],
|
||||||
"baseUrl": ".",
|
|
||||||
"paths": {
|
"paths": {
|
||||||
"~/*": ["./app/*"]
|
"~/*": ["./app/*"]
|
||||||
},
|
},
|
||||||
"esModuleInterop": true,
|
|
||||||
"verbatimModuleSyntax": true,
|
"verbatimModuleSyntax": true,
|
||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
"resolveJsonModule": true,
|
"resolveJsonModule": true,
|
||||||
|
|||||||
31
bun.lock
31
bun.lock
@@ -8,11 +8,12 @@
|
|||||||
"@json-render/core": "^0.12.1",
|
"@json-render/core": "^0.12.1",
|
||||||
"@nym.sh/jrx": "^0.2.0",
|
"@nym.sh/jrx": "^0.2.0",
|
||||||
"@types/bun": "latest",
|
"@types/bun": "latest",
|
||||||
|
"@typescript/native-preview": "^7.0.0-dev.20260412.1",
|
||||||
"oxfmt": "^0.24.0",
|
"oxfmt": "^0.24.0",
|
||||||
"oxlint": "^1.39.0",
|
"oxlint": "^1.39.0",
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"typescript": "^5",
|
"typescript": "^6",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"apps/admin-dashboard": {
|
"apps/admin-dashboard": {
|
||||||
@@ -41,7 +42,7 @@
|
|||||||
"@types/react": "^19.2.5",
|
"@types/react": "^19.2.5",
|
||||||
"@types/react-dom": "^19.2.3",
|
"@types/react-dom": "^19.2.3",
|
||||||
"@vitejs/plugin-react": "^5.1.1",
|
"@vitejs/plugin-react": "^5.1.1",
|
||||||
"typescript": "~5.9.3",
|
"typescript": "^6",
|
||||||
"vite": "^7.2.4",
|
"vite": "^7.2.4",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -111,7 +112,7 @@
|
|||||||
"eas-cli": "^18.0.1",
|
"eas-cli": "^18.0.1",
|
||||||
"eslint": "^9.25.0",
|
"eslint": "^9.25.0",
|
||||||
"eslint-config-expo": "~10.0.0",
|
"eslint-config-expo": "~10.0.0",
|
||||||
"typescript": "~5.9.2",
|
"typescript": "^6",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"apps/waitlist-website": {
|
"apps/waitlist-website": {
|
||||||
@@ -138,7 +139,7 @@
|
|||||||
"@types/react": "^19.2.7",
|
"@types/react": "^19.2.7",
|
||||||
"@types/react-dom": "^19.2.3",
|
"@types/react-dom": "^19.2.3",
|
||||||
"tailwindcss": "^4.1.13",
|
"tailwindcss": "^4.1.13",
|
||||||
"typescript": "^5.9.2",
|
"typescript": "^6",
|
||||||
"vite": "^7.1.7",
|
"vite": "^7.1.7",
|
||||||
"vite-tsconfig-paths": "^5.1.4",
|
"vite-tsconfig-paths": "^5.1.4",
|
||||||
},
|
},
|
||||||
@@ -1452,6 +1453,22 @@
|
|||||||
|
|
||||||
"@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.57.0", "", { "dependencies": { "@typescript-eslint/types": "8.57.0", "eslint-visitor-keys": "^5.0.0" } }, "sha512-zm6xx8UT/Xy2oSr2ZXD0pZo7Jx2XsCoID2IUh9YSTFRu7z+WdwYTRk6LhUftm1crwqbuoF6I8zAFeCMw0YjwDg=="],
|
"@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.57.0", "", { "dependencies": { "@typescript-eslint/types": "8.57.0", "eslint-visitor-keys": "^5.0.0" } }, "sha512-zm6xx8UT/Xy2oSr2ZXD0pZo7Jx2XsCoID2IUh9YSTFRu7z+WdwYTRk6LhUftm1crwqbuoF6I8zAFeCMw0YjwDg=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview": ["@typescript/native-preview@7.0.0-dev.20260412.1", "", { "optionalDependencies": { "@typescript/native-preview-darwin-arm64": "7.0.0-dev.20260412.1", "@typescript/native-preview-darwin-x64": "7.0.0-dev.20260412.1", "@typescript/native-preview-linux-arm": "7.0.0-dev.20260412.1", "@typescript/native-preview-linux-arm64": "7.0.0-dev.20260412.1", "@typescript/native-preview-linux-x64": "7.0.0-dev.20260412.1", "@typescript/native-preview-win32-arm64": "7.0.0-dev.20260412.1", "@typescript/native-preview-win32-x64": "7.0.0-dev.20260412.1" }, "bin": { "tsgo": "bin/tsgo.js" } }, "sha512-tDw3XZt2BkjAlt/MJmnFGmbe9lgKmc5wezmrMoBIEvJcqz+/KVpVBVvjbkZoaiABnJmuG3G3b6IUFrEveTw6UQ=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-darwin-arm64": ["@typescript/native-preview-darwin-arm64@7.0.0-dev.20260412.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-sSkFG+hjtRWffg6FddF3dEkk7N3TRMEqfiUpixwcWhXgyocMdPw8wutTvQRBxQdgxeL9y01M2SO8A8YPPiEgVg=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-darwin-x64": ["@typescript/native-preview-darwin-x64@7.0.0-dev.20260412.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-m2BTeaLkrHEEDg0D9snigddy01qTY+wgx+W+GpXAfx36PPvW4xWuGXNVWfSaB8bqAC9C8NeLnT/C9/G/rJ5v2w=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-linux-arm": ["@typescript/native-preview-linux-arm@7.0.0-dev.20260412.1", "", { "os": "linux", "cpu": "arm" }, "sha512-wDLekbfsfmKMWORg7CTnEnpKj8oXpU/6AEBrtVN9CEUCiQAe6yH878nZHhJNzWQXHtrtFf3lY49Yplqmdxja3w=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-linux-arm64": ["@typescript/native-preview-linux-arm64@7.0.0-dev.20260412.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-JAdsG6MlVV1hoAUKPy8zxAL7xLeNxz8JgCbLCJVqW8EyH29R9FD4cFTqr7CSIRTNUEDzDTrgnXUyoRtDe1gr+w=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-linux-x64": ["@typescript/native-preview-linux-x64@7.0.0-dev.20260412.1", "", { "os": "linux", "cpu": "x64" }, "sha512-gYgppiQIqid3jZ7D8THh4k3Q+4bwidrQH6SL9Xgbk1qfP6/jwv8twuPqDOfZ+cK2OD55lQHp15fOh2lMNAC40Q=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-win32-arm64": ["@typescript/native-preview-win32-arm64@7.0.0-dev.20260412.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-TOh7rH5H3jisHJqRXJSjmUGMzcbNBocS/hufhXPQIv+g3pdG5IKZoSnv3SV62I5d12FFDSS5KQon5MQPnOKAHg=="],
|
||||||
|
|
||||||
|
"@typescript/native-preview-win32-x64": ["@typescript/native-preview-win32-x64@7.0.0-dev.20260412.1", "", { "os": "win32", "cpu": "x64" }, "sha512-u+70wL89wspN1wKoX6FVNUATRGCG3BpleByP3H/UqOZvlwuMm8N7Gy8hEbM0U8bDyAuyP/daUfTBVkqXjjv9mA=="],
|
||||||
|
|
||||||
"@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
|
"@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
|
||||||
|
|
||||||
"@unrs/resolver-binding-android-arm-eabi": ["@unrs/resolver-binding-android-arm-eabi@1.11.1", "", { "os": "android", "cpu": "arm" }, "sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw=="],
|
"@unrs/resolver-binding-android-arm-eabi": ["@unrs/resolver-binding-android-arm-eabi@1.11.1", "", { "os": "android", "cpu": "arm" }, "sha512-ppLRUgHVaGRWUx0R0Ut06Mjo9gBaBkg3v/8AxusGLhsIotbBLuRk51rAzqLC8gq6NyyAojEXglNjzf6R948DNw=="],
|
||||||
@@ -3428,7 +3445,7 @@
|
|||||||
|
|
||||||
"typed-array-length": ["typed-array-length@1.0.7", "", { "dependencies": { "call-bind": "^1.0.7", "for-each": "^0.3.3", "gopd": "^1.0.1", "is-typed-array": "^1.1.13", "possible-typed-array-names": "^1.0.0", "reflect.getprototypeof": "^1.0.6" } }, "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg=="],
|
"typed-array-length": ["typed-array-length@1.0.7", "", { "dependencies": { "call-bind": "^1.0.7", "for-each": "^0.3.3", "gopd": "^1.0.1", "is-typed-array": "^1.1.13", "possible-typed-array-names": "^1.0.0", "reflect.getprototypeof": "^1.0.6" } }, "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg=="],
|
||||||
|
|
||||||
"typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
|
"typescript": ["typescript@6.0.2", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-bGdAIrZ0wiGDo5l8c++HWtbaNCWTS4UTv7RaTH/ThVIgjkveJt83m74bBHMJkuCbslY8ixgLBVZJIOiQlQTjfQ=="],
|
||||||
|
|
||||||
"ua-parser-js": ["ua-parser-js@1.0.41", "", { "bin": { "ua-parser-js": "script/cli.js" } }, "sha512-LbBDqdIC5s8iROCUjMbW1f5dJQTEFB1+KO9ogbvlb3nm9n4YHa5p4KTvFPWvh2Hs8gZMBuiB1/8+pdfe/tDPug=="],
|
"ua-parser-js": ["ua-parser-js@1.0.41", "", { "bin": { "ua-parser-js": "script/cli.js" } }, "sha512-LbBDqdIC5s8iROCUjMbW1f5dJQTEFB1+KO9ogbvlb3nm9n4YHa5p4KTvFPWvh2Hs8gZMBuiB1/8+pdfe/tDPug=="],
|
||||||
|
|
||||||
@@ -3904,8 +3921,6 @@
|
|||||||
|
|
||||||
"accepts/negotiator": ["negotiator@0.6.3", "", {}, "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg=="],
|
"accepts/negotiator": ["negotiator@0.6.3", "", {}, "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg=="],
|
||||||
|
|
||||||
"aelis-client/@tanstack/react-query": ["@tanstack/react-query@5.90.21", "", { "dependencies": { "@tanstack/query-core": "5.90.20" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-0Lu6y5t+tvlTJMTO7oh5NSpJfpg/5D41LlThfepTixPYkJ0sE2Jj0m0f6yYqujBwIXlId87e234+MxG3D3g7kg=="],
|
|
||||||
|
|
||||||
"aelis-client/@types/react": ["@types/react@19.1.17", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-Qec1E3mhALmaspIrhWt9jkQMNdw6bReVu64mjvhbhq2NFPftLPVr+l1SZgmw/66WwBNpDh7ao5AT6gF5v41PFA=="],
|
"aelis-client/@types/react": ["@types/react@19.1.17", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-Qec1E3mhALmaspIrhWt9jkQMNdw6bReVu64mjvhbhq2NFPftLPVr+l1SZgmw/66WwBNpDh7ao5AT6gF5v41PFA=="],
|
||||||
|
|
||||||
"aelis-client/react": ["react@19.1.0", "", {}, "sha512-FS+XFBNvn3GTAWq26joslQgWNoFu08F4kl0J4CgdNKADkdSGXQyTCnKteIAJy96Br6YbpEU1LSzV5dYtjMkMDg=="],
|
"aelis-client/react": ["react@19.1.0", "", {}, "sha512-FS+XFBNvn3GTAWq26joslQgWNoFu08F4kl0J4CgdNKADkdSGXQyTCnKteIAJy96Br6YbpEU1LSzV5dYtjMkMDg=="],
|
||||||
@@ -4544,8 +4559,6 @@
|
|||||||
|
|
||||||
"@typescript-eslint/typescript-estree/minimatch/brace-expansion": ["brace-expansion@5.0.4", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-h+DEnpVvxmfVefa4jFbCf5HdH5YMDXRsmKflpf1pILZWRFlTbJpxeU55nJl4Smt5HQaGzg1o6RHFPJaOqnmBDg=="],
|
"@typescript-eslint/typescript-estree/minimatch/brace-expansion": ["brace-expansion@5.0.4", "", { "dependencies": { "balanced-match": "^4.0.2" } }, "sha512-h+DEnpVvxmfVefa4jFbCf5HdH5YMDXRsmKflpf1pILZWRFlTbJpxeU55nJl4Smt5HQaGzg1o6RHFPJaOqnmBDg=="],
|
||||||
|
|
||||||
"aelis-client/@tanstack/react-query/@tanstack/query-core": ["@tanstack/query-core@5.90.20", "", {}, "sha512-OMD2HLpNouXEfZJWcKeVKUgQ5n+n3A2JFmBaScpNDUqSrQSjiveC7dKMe53uJUg1nDG16ttFPz2xfilz6i2uVg=="],
|
|
||||||
|
|
||||||
"aelis-client/react-dom/scheduler": ["scheduler@0.26.0", "", {}, "sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA=="],
|
"aelis-client/react-dom/scheduler": ["scheduler@0.26.0", "", {}, "sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA=="],
|
||||||
|
|
||||||
"better-opn/open/define-lazy-prop": ["define-lazy-prop@2.0.0", "", {}, "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og=="],
|
"better-opn/open/define-lazy-prop": ["define-lazy-prop@2.0.0", "", {}, "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og=="],
|
||||||
|
|||||||
@@ -17,10 +17,11 @@
|
|||||||
"@json-render/core": "^0.12.1",
|
"@json-render/core": "^0.12.1",
|
||||||
"@nym.sh/jrx": "^0.2.0",
|
"@nym.sh/jrx": "^0.2.0",
|
||||||
"@types/bun": "latest",
|
"@types/bun": "latest",
|
||||||
|
"@typescript/native-preview": "^7.0.0-dev.20260412.1",
|
||||||
"oxfmt": "^0.24.0",
|
"oxfmt": "^0.24.0",
|
||||||
"oxlint": "^1.39.0"
|
"oxlint": "^1.39.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"typescript": "^5"
|
"typescript": "^6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
// Environment setup & latest features
|
|
||||||
"lib": ["ESNext"],
|
"lib": ["ESNext"],
|
||||||
"target": "ESNext",
|
"target": "ESNext",
|
||||||
"module": "Preserve",
|
"module": "Preserve",
|
||||||
@@ -8,20 +7,19 @@
|
|||||||
"jsx": "react-jsx",
|
"jsx": "react-jsx",
|
||||||
"allowJs": true,
|
"allowJs": true,
|
||||||
|
|
||||||
// Bundler mode
|
|
||||||
"moduleResolution": "bundler",
|
"moduleResolution": "bundler",
|
||||||
"allowImportingTsExtensions": true,
|
"allowImportingTsExtensions": true,
|
||||||
"verbatimModuleSyntax": true,
|
"verbatimModuleSyntax": true,
|
||||||
"noEmit": true,
|
"noEmit": true,
|
||||||
|
|
||||||
// Best practices
|
"types": ["bun"],
|
||||||
|
|
||||||
"strict": true,
|
"strict": true,
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"noFallthroughCasesInSwitch": true,
|
"noFallthroughCasesInSwitch": true,
|
||||||
"noUncheckedIndexedAccess": true,
|
"noUncheckedIndexedAccess": true,
|
||||||
"noImplicitOverride": true,
|
"noImplicitOverride": true,
|
||||||
|
|
||||||
// Some stricter flags (disabled by default)
|
|
||||||
"noUnusedLocals": false,
|
"noUnusedLocals": false,
|
||||||
"noUnusedParameters": false,
|
"noUnusedParameters": false,
|
||||||
"noPropertyAccessFromIndexSignature": false
|
"noPropertyAccessFromIndexSignature": false
|
||||||
|
|||||||
Reference in New Issue
Block a user