@powerhousedao/connect 6.2.0-dev.2 → 6.2.0-dev.20
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{AddDriveModal-kzL41qdg.js → AddDriveModal-BoKLJyUY.js} +4 -7
- package/dist/{AddDriveModal-kzL41qdg.js.map → AddDriveModal-BoKLJyUY.js.map} +1 -1
- package/dist/{ClearStorageModal-DIKwqiia.js → ClearStorageModal-CXoNWmRe.js} +4 -4
- package/dist/{ClearStorageModal-DIKwqiia.js.map → ClearStorageModal-CXoNWmRe.js.map} +1 -1
- package/dist/{DebugSettingsModal-Dxi_8rOa.js → DebugSettingsModal-CuTgd7TZ.js} +6 -6
- package/dist/{DebugSettingsModal-Dxi_8rOa.js.map → DebugSettingsModal-CuTgd7TZ.js.map} +1 -1
- package/dist/DriveAuthRequiredModal-D1TSn4Vh.js +36 -0
- package/dist/DriveAuthRequiredModal-D1TSn4Vh.js.map +1 -0
- package/dist/{DriveSettingsModal-L5sM-rIh.js → DriveSettingsModal-CsNFEJ9E.js} +5 -5
- package/dist/DriveSettingsModal-CsNFEJ9E.js.map +1 -0
- package/dist/{InspectorModal-C7pIf6OO.js → InspectorModal-BsSihzuc.js} +6 -6
- package/dist/{InspectorModal-C7pIf6OO.js.map → InspectorModal-BsSihzuc.js.map} +1 -1
- package/dist/{MissingPackageModal-D_N96M6a.js → MissingPackageModal-Bn036mWQ.js} +4 -4
- package/dist/{MissingPackageModal-D_N96M6a.js.map → MissingPackageModal-Bn036mWQ.js.map} +1 -1
- package/dist/{SettingsModal-B2DuAQVQ.js → SettingsModal-D0F9IxhO.js} +19 -26
- package/dist/SettingsModal-D0F9IxhO.js.map +1 -0
- package/dist/{UpgradeDriveModal-CyAUDCU7.js → UpgradeDriveModal-C7lV4rAX.js} +3 -3
- package/dist/{UpgradeDriveModal-CyAUDCU7.js.map → UpgradeDriveModal-C7lV4rAX.js.map} +1 -1
- package/dist/app-loader-CCVhETUw.js +4 -0
- package/dist/{app-loader-DvjTCKwK.js → app-loader-CWzoV72t.js} +44 -19
- package/dist/app-loader-CWzoV72t.js.map +1 -0
- package/dist/{build-info-M90cqR0U.js → build-info-BY8gRSI9.js} +4 -4
- package/dist/{build-info-M90cqR0U.js.map → build-info-BY8gRSI9.js.map} +1 -1
- package/dist/{components-BY3M6wF8.js → components-7jnuvqBG.js} +3 -3
- package/dist/{connect.config-aPx3rHZQ.js → connect.config-Bt4pvvWV.js} +9 -5
- package/dist/connect.config-Bt4pvvWV.js.map +1 -0
- package/dist/{load-BSRod9gt.js → load-C1ufF-ho.js} +8 -8
- package/dist/{load-BSRod9gt.js.map → load-C1ufF-ho.js.map} +1 -1
- package/dist/main.js +1 -1
- package/dist/{package-VvzmOTZO.js → package-DgJJtNq6.js} +6 -5
- package/dist/package-DgJJtNq6.js.map +1 -0
- package/dist/{pglite-runtime-toc1-TWa.js → pglite-runtime-Ch2YeUAA.js} +4 -4
- package/dist/{pglite-runtime-toc1-TWa.js.map → pglite-runtime-Ch2YeUAA.js.map} +1 -1
- package/dist/{pglite-seed-D8oSKkep.js → pglite-seed-BWk7yOwr.js} +4 -4
- package/dist/{pglite-seed-D8oSKkep.js.map → pglite-seed-BWk7yOwr.js.map} +1 -1
- package/dist/{reactor-BlxTyHZ5.js → reactor-BxNvHOKC.js} +32 -38
- package/dist/reactor-BxNvHOKC.js.map +1 -0
- package/dist/{registerServiceWorker-B7uKpkci.js → registerServiceWorker-D9l1kja7.js} +4 -4
- package/dist/{registerServiceWorker-B7uKpkci.js.map → registerServiceWorker-D9l1kja7.js.map} +1 -1
- package/dist/start-connect.js +1 -1
- package/dist/style.css +74 -0
- package/dist/{useRegistryPackages-Bh0luCvV.js → useRegistryPackages-B-gFECmd.js} +44 -8
- package/dist/useRegistryPackages-B-gFECmd.js.map +1 -0
- package/package.json +12 -11
- package/dist/DriveSettingsModal-L5sM-rIh.js.map +0 -1
- package/dist/SettingsModal-B2DuAQVQ.js.map +0 -1
- package/dist/app-loader-DW8p8fo5.js +0 -4
- package/dist/app-loader-DvjTCKwK.js.map +0 -1
- package/dist/connect.config-aPx3rHZQ.js.map +0 -1
- package/dist/package-VvzmOTZO.js.map +0 -1
- package/dist/reactor-BlxTyHZ5.js.map +0 -1
- package/dist/useRegistryPackages-Bh0luCvV.js.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"reactor-BxNvHOKC.js","sources":["../src/utils/reactor.ts","../src/feature-flags.ts","../src/no-registry-discovery.ts","../src/package-discovery.ts","../src/package-manager.ts","../src/pglite.db.ts","../src/store/processor-host-module.ts","../src/store/reactor.ts"],"sourcesContent":["import {\n addRemoteDrive,\n ChannelScheme,\n ReactorBuilder,\n ReactorClientBuilder,\n setDriveMetadata,\n waitForDocumentReady,\n type BrowserReactorClientModule,\n type Database,\n type IDocumentModelLoader,\n type JwtHandler,\n type SignerConfig,\n} from \"@powerhousedao/reactor-browser\";\nimport type { PHConnectDefaultDrive } from \"@powerhousedao/shared/clis\";\nimport type { RuntimePowerhouseConfig } from \"@powerhousedao/shared/connect\";\nimport type {\n DocumentModelModule,\n UpgradeManifest,\n} from \"@powerhousedao/shared/document-model\";\nimport { createSignatureVerifier, type IRenown } from \"@renown/sdk\";\nimport { ConsoleLogger } from \"document-model\";\nimport { Kysely } from \"kysely\";\nimport { PGliteDialect } from \"kysely-pglite-dialect\";\nimport {\n detectReactorPgMajor,\n loadPGliteModule,\n resolvePgMajorForRuntime,\n} from \"./pglite-runtime.js\";\nimport { REACTOR_PGLITE_NAME } from \"./storage-namespace.js\";\n\n/**\n * Creates a Reactor that plugs into legacy storage but syncs through the new\n * Reactor GQL API.\n */\nexport async function createBrowserReactor(\n documentModelModules: DocumentModelModule[],\n upgradeManifests: UpgradeManifest<readonly number[]>[],\n renown: IRenown,\n documentModelLoader?: IDocumentModelLoader,\n): Promise<BrowserReactorClientModule> {\n const signerConfig: SignerConfig = {\n signer: renown.signer,\n verifier: createSignatureVerifier(),\n };\n\n const jwtHandler: JwtHandler = async (_url: string) => {\n if (!renown.user) {\n return undefined;\n }\n // aud omitted: server verifies without an audience, so aud-bearing tokens\n // are rejected. Re-enable once both sides support audience restriction.\n return renown.getBearerToken({ expiresIn: 10 });\n };\n\n const detected = await detectReactorPgMajor();\n const major = resolvePgMajorForRuntime(detected);\n if (major !== 17) {\n console.warn(\n `[reactor] Running against legacy PGlite data dir (Postgres ${major}). Migrate to PG17 from the banner or the Inspector → Debug tab.`,\n );\n }\n const { PGlite } = await loadPGliteModule(major);\n const pg = new PGlite(`idb://${REACTOR_PGLITE_NAME}`, {\n relaxedDurability: true,\n });\n const logger = new ConsoleLogger([\"reactor-client\"]);\n const builder = new ReactorClientBuilder()\n .withLogger(logger)\n .withSigner(signerConfig)\n .withReactorBuilder(\n new ReactorBuilder()\n .withDocumentModels(documentModelModules)\n .withUpgradeManifests(upgradeManifests)\n .withChannelScheme(ChannelScheme.CONNECT)\n .withJwtHandler(jwtHandler)\n .withKysely(\n new Kysely<Database>({\n dialect: new PGliteDialect(pg),\n }),\n ),\n );\n\n if (documentModelLoader) {\n builder.withDocumentModelLoader(documentModelLoader);\n }\n\n const module = await builder.buildModule();\n return {\n ...module,\n pg,\n } as BrowserReactorClientModule;\n}\n\nexport function getDefaultDrives(\n runtimeConfig: RuntimePowerhouseConfig,\n): PHConnectDefaultDrive[] {\n return runtimeConfig.connect.drives?.defaultDrives ?? [];\n}\n\n/**\n * A drive add failed because the switchboard rejected the (anonymous or\n * unverified) caller — Forbidden/Unauthorized — rather than a transient\n * reachability error. Drives the login prompt instead of retrying.\n */\nfunction isDriveAuthError(error: unknown): boolean {\n const message = error instanceof Error ? error.message : String(error);\n return /forbidden|unauthorized|insufficient permissions|\\b401\\b|\\b403\\b/i.test(\n message,\n );\n}\n\n/**\n * Add default drives for the new reactor via sync manager.\n *\n * Drives register concurrently so a slow or unreachable drive can't delay the\n * others — in particular the one the URL slug resolves to. Retries with linear\n * backoff to handle the common race where Connect's dev server is ready before\n * the switchboard has finished binding its port.\n *\n * @param drives - Array of drive objects with url, optional name and icon\n */\nexport async function addDefaultDrivesForNewReactor(\n drives: PHConnectDefaultDrive[],\n): Promise<void> {\n const MAX_ATTEMPTS = 3;\n const BACKOFF_MS = 2000;\n\n await Promise.all(\n drives.map(async (drive) => {\n let driveId: string | undefined;\n for (let attempt = 1; attempt <= MAX_ATTEMPTS; attempt++) {\n try {\n driveId = await addRemoteDrive(drive.url);\n break;\n } catch (error) {\n if (isDriveAuthError(error)) {\n // addRemoteDrive already surfaces the login modal; auth failures\n // don't self-heal, so don't burn the remaining retries.\n break;\n }\n if (attempt === MAX_ATTEMPTS) {\n console.error(\n `Failed to add default drive ${drive.url} after ${MAX_ATTEMPTS} attempts:`,\n error,\n );\n } else {\n const delay = BACKOFF_MS * attempt;\n console.warn(\n `Default drive ${drive.url} not reachable (attempt ${attempt}/${MAX_ATTEMPTS}), retrying in ${delay}ms...`,\n );\n await new Promise((resolve) => setTimeout(resolve, delay));\n }\n }\n }\n\n if (driveId && (drive.name || drive.icon)) {\n try {\n // setDriveMetadata dispatches against the local drive document, which\n // only exists once initial backfill delivers it — wait for it first\n // so the name/icon override isn't lost to a sync race.\n const reactorClient = window.ph?.reactorClient;\n if (reactorClient) {\n await waitForDocumentReady(reactorClient, driveId, {\n timeoutMs: 15_000,\n });\n }\n await setDriveMetadata(driveId, {\n name: drive.name,\n icon: drive.icon,\n });\n } catch (error) {\n console.warn(\n `Default drive ${drive.url} was added but metadata update failed:`,\n error,\n );\n }\n }\n }),\n );\n}\n","import {\n OpenFeature,\n type ErrorCode,\n type JsonValue,\n type Provider,\n type ResolutionDetails,\n} from \"@openfeature/web-sdk\";\nimport { logger, setLogLevel, type ILogger } from \"document-model\";\n\n/**\n * QueryParamProvider reads feature flags from URL query parameters.\n *\n * Usage:\n * const params = new URLSearchParams(window.location.search);\n * const provider = new QueryParamProvider(params);\n * await OpenFeature.setProviderAndWait(provider);\n *\n * Query parameter format:\n * ?FEATURE_DUAL_ACTION_CREATE_ENABLED=true&FEATURE_FOO=false\n */\nexport class QueryParamProvider implements Provider {\n public readonly runsOn = \"client\" as const;\n\n readonly metadata = {\n name: \"QueryParamProvider\",\n } as const;\n\n private flags: Map<string, string>;\n\n constructor(searchParams: URLSearchParams) {\n this.flags = new Map();\n\n // Extract all query parameters that look like feature flags\n for (const [key, value] of searchParams.entries()) {\n this.flags.set(key, value);\n }\n }\n\n resolveBooleanEvaluation(\n flagKey: string,\n defaultValue: boolean,\n ): ResolutionDetails<boolean> {\n const value = this.flags.get(flagKey);\n\n if (value === undefined) {\n return {\n value: defaultValue,\n reason: \"DEFAULT\",\n };\n }\n\n // Parse boolean from string\n const boolValue = value.toLowerCase() === \"true\" || value === \"1\";\n\n return {\n value: boolValue,\n reason: \"STATIC\",\n variant: value,\n };\n }\n\n resolveStringEvaluation(\n flagKey: string,\n defaultValue: string,\n ): ResolutionDetails<string> {\n const value = this.flags.get(flagKey);\n\n if (value === undefined) {\n return {\n value: defaultValue,\n reason: \"DEFAULT\",\n };\n }\n\n return {\n value,\n reason: \"STATIC\",\n variant: value,\n };\n }\n\n resolveNumberEvaluation(\n flagKey: string,\n defaultValue: number,\n ): ResolutionDetails<number> {\n const value = this.flags.get(flagKey);\n\n if (value === undefined) {\n return {\n value: defaultValue,\n reason: \"DEFAULT\",\n };\n }\n\n const numValue = Number(value);\n\n if (isNaN(numValue)) {\n return {\n value: defaultValue,\n reason: \"ERROR\",\n errorCode: \"PARSE_ERROR\" as ErrorCode,\n errorMessage: `Failed to parse \"${value}\" as a number`,\n };\n }\n\n return {\n value: numValue,\n reason: \"STATIC\",\n variant: value,\n };\n }\n\n resolveObjectEvaluation<T extends JsonValue>(\n flagKey: string,\n defaultValue: T,\n ): ResolutionDetails<T> {\n const value = this.flags.get(flagKey);\n\n if (value === undefined) {\n return {\n value: defaultValue,\n reason: \"DEFAULT\",\n };\n }\n\n try {\n const objValue = JSON.parse(value) as T;\n return {\n value: objValue,\n reason: \"STATIC\",\n variant: value,\n };\n } catch (error) {\n return {\n value: defaultValue,\n reason: \"ERROR\",\n errorCode: \"PARSE_ERROR\" as ErrorCode,\n errorMessage: `Failed to parse JSON: ${error instanceof Error ? error.message : String(error)}`,\n };\n }\n }\n}\n\n/**\n * Initialize OpenFeature with the QueryParamProvider.\n * Reads feature flags from query parameters.\n */\nexport async function initFeatureFlags(\n searchParams?: URLSearchParams,\n): Promise<Map<string, boolean>> {\n const params =\n searchParams ??\n (typeof window !== \"undefined\"\n ? new URLSearchParams(window.location.search)\n : new URLSearchParams());\n const provider = new QueryParamProvider(params);\n await OpenFeature.setProviderAndWait(provider);\n\n const features = new Map<string, boolean>();\n\n // Handle LOG_LEVEL query param override\n const logLevelParam = params.get(\"LOG_LEVEL\");\n if (logLevelParam) {\n const validLogLevels = [\"verbose\", \"debug\", \"info\", \"warn\", \"error\"];\n if (validLogLevels.includes(logLevelParam.toLowerCase())) {\n setLogLevel(logLevelParam.toLowerCase() as ILogger[\"level\"]);\n logger.info(\n \"Log level set to @level via query param\",\n logLevelParam.toLowerCase(),\n );\n } else {\n logger.warn(\n \"Invalid LOG_LEVEL query param: @param. Valid values: @validLevels\",\n logLevelParam,\n validLogLevels.join(\", \"),\n );\n }\n }\n\n return features;\n}\n","import {\n type IDocumentModelLoader,\n type IPackageDiscoveryService,\n type DiscoveryEvent,\n type DiscoveryEventListener,\n type FailedInstallation,\n type PendingInstallation,\n} from \"@powerhousedao/reactor-browser\";\nimport type { DocumentModelModule } from \"@powerhousedao/shared/document-model\";\nimport type { BrowserPackageManager } from \"./package-manager.js\";\n\n/**\n * Discovery service used when no package registry is configured. Every load\n * for an unknown document type is recorded as a failure with reason\n * \"no-registry\" so the UI can surface the issue; the modal install flow is a\n * no-op because there is no registry to query.\n */\nexport class NoRegistryDiscoveryService\n implements IDocumentModelLoader, IPackageDiscoveryService\n{\n #packageManager: BrowserPackageManager;\n #failed = new Map<string, FailedInstallation>();\n #failedMemo: FailedInstallation[] = [];\n #failedSubscribers = new Set<() => void>();\n #eventSubscribers = new Set<DiscoveryEventListener>();\n\n constructor(packageManager: BrowserPackageManager) {\n this.#packageManager = packageManager;\n }\n\n load(documentType: string): Promise<DocumentModelModule<any>> {\n const existing = this.#findModuleInLoadedPackages(documentType);\n if (existing) {\n this.#clearFailed(documentType);\n return Promise.resolve(existing);\n }\n\n this.#recordFailure(documentType);\n return Promise.reject(\n new Error(\n `No package registry configured; cannot resolve \"${documentType}\"`,\n ),\n );\n }\n\n promptInstallation(): void {\n // no registry to query — nothing to prompt\n }\n\n approveInstallation(): Promise<void> {\n return Promise.resolve();\n }\n\n dismissInstallation(): void {\n // nothing to dismiss\n }\n\n getPendingInstallations(): PendingInstallation[] {\n return [];\n }\n\n subscribePending(): () => void {\n return () => {};\n }\n\n getFailedInstallations(): FailedInstallation[] {\n return this.#failedMemo;\n }\n\n subscribeFailed(listener: () => void): () => void {\n this.#failedSubscribers.add(listener);\n return () => {\n this.#failedSubscribers.delete(listener);\n };\n }\n\n subscribeEvents(listener: DiscoveryEventListener): () => void {\n this.#eventSubscribers.add(listener);\n return () => {\n this.#eventSubscribers.delete(listener);\n };\n }\n\n retryInstallation(documentType: string): Promise<void> {\n // Retry is a no-op without a registry, but re-emitting the failure lets\n // a UI that cleared the row know the underlying condition still holds.\n this.#recordFailure(documentType);\n return Promise.resolve();\n }\n\n #findModuleInLoadedPackages(\n documentType: string,\n ): DocumentModelModule<any> | undefined {\n return this.#packageManager.packages\n .flatMap((p) => p.documentModels)\n .find((m) => m.documentModel.global.id === documentType);\n }\n\n #recordFailure(documentType: string): void {\n this.#failed.set(documentType, {\n documentType,\n reason: \"no-registry\",\n packageNames: [],\n error: null,\n });\n this.#emitEvent({\n type: \"load-failed\",\n documentType,\n reason: \"no-registry\",\n });\n this.#notifyFailedChanged();\n }\n\n #clearFailed(documentType: string): void {\n if (this.#failed.delete(documentType)) {\n this.#notifyFailedChanged();\n }\n }\n\n #notifyFailedChanged(): void {\n this.#failedMemo = Array.from(this.#failed.values());\n for (const listener of this.#failedSubscribers) {\n listener();\n }\n }\n\n #emitEvent(event: DiscoveryEvent): void {\n for (const listener of this.#eventSubscribers) {\n listener(event);\n }\n }\n}\n","import {\n BrowserLocalStorage,\n type IDocumentModelLoader,\n type IPackageDiscoveryService,\n type DiscoveryEvent,\n type DiscoveryEventListener,\n type FailedInstallation,\n type FailedInstallationReason,\n type PendingInstallation,\n} from \"@powerhousedao/reactor-browser\";\nimport type { RegistryClient } from \"@powerhousedao/reactor-browser\";\nimport type { DocumentModelModule } from \"@powerhousedao/shared/document-model\";\nimport type { BrowserPackageManager } from \"./package-manager.js\";\n\nexport type DiscoveryMode = \"immediate\" | \"manual\";\n\ntype DeferredEntry = {\n packageNames: string[];\n resolve: (module: DocumentModelModule<any>) => void;\n reject: (reason: unknown) => void;\n promise: Promise<DocumentModelModule<any>>;\n};\n\nexport class PackageDiscoveryService\n implements IDocumentModelLoader, IPackageDiscoveryService\n{\n #packageManager: BrowserPackageManager;\n #registryClient: RegistryClient;\n #mode: DiscoveryMode;\n\n #deferred = new Map<string, DeferredEntry>();\n #pending = new Map<string, PendingInstallation>();\n #pendingMemo: PendingInstallation[] = [];\n #pendingSubscribers = new Set<() => void>();\n #failed = new Map<string, FailedInstallation>();\n #failedMemo: FailedInstallation[] = [];\n #failedSubscribers = new Set<() => void>();\n #eventSubscribers = new Set<DiscoveryEventListener>();\n #dismissedStorage: BrowserLocalStorage<boolean>;\n #discoveredTypes = new Map<string, string[]>();\n\n constructor(\n packageManager: BrowserPackageManager,\n registryClient: RegistryClient,\n options: { mode: DiscoveryMode; storageKey: string },\n ) {\n this.#packageManager = packageManager;\n this.#registryClient = registryClient;\n this.#mode = options.mode;\n this.#dismissedStorage = new BrowserLocalStorage<boolean>(\n options.storageKey + \":PH_DISMISSED_TYPES\",\n );\n }\n\n load(documentType: string): Promise<DocumentModelModule<any>> {\n const existing = this.#findModuleInLoadedPackages(documentType);\n if (existing) {\n this.#clearFailed(documentType);\n return Promise.resolve(existing);\n }\n\n if (this.#dismissedStorage.has(documentType)) {\n this.#recordFailure(documentType, \"dismissed\", [], null);\n return Promise.reject(\n new Error(`Document type \"${documentType}\" was dismissed`),\n );\n }\n\n const tracked = this.#deferred.get(documentType);\n if (tracked) {\n return tracked.promise;\n }\n\n return this.#discover(documentType);\n }\n\n promptInstallation(documentType: string): void {\n const packageNames = this.#discoveredTypes.get(documentType);\n if (!packageNames) return;\n if (this.#pending.has(documentType)) return;\n\n this.#discoveredTypes.delete(documentType);\n this.#addToPending(documentType, packageNames);\n }\n\n async approveInstallation(packageName: string): Promise<void> {\n const affectedTypes = this.#findTypesByPackage(packageName);\n if (affectedTypes.length === 0) return;\n\n const result = await this.#packageManager.addPackage(packageName);\n if (result.type === \"error\") {\n this.#emitEvent({\n type: \"installation-failed\",\n packageName,\n error: result.error,\n });\n for (const documentType of affectedTypes) {\n const entry = this.#deferred.get(documentType);\n if (entry) {\n entry.reject(result.error);\n this.#deferred.delete(documentType);\n this.#pending.delete(documentType);\n }\n this.#recordFailure(\n documentType,\n \"install-failed\",\n [packageName],\n result.error,\n );\n }\n this.#notifyPendingChanged();\n return;\n }\n\n this.#emitEvent({\n type: \"installation-approved\",\n packageName,\n documentTypes: affectedTypes,\n });\n\n for (const documentType of affectedTypes) {\n const entry = this.#deferred.get(documentType);\n if (!entry) continue;\n\n const module = this.#findModuleInLoadedPackages(documentType);\n if (module) {\n entry.resolve(module);\n this.#clearFailed(documentType);\n } else {\n const error = new Error(\n `Package \"${packageName}\" installed but module for \"${documentType}\" not found`,\n );\n entry.reject(error);\n this.#recordFailure(\n documentType,\n \"install-failed\",\n [packageName],\n error,\n );\n }\n this.#deferred.delete(documentType);\n this.#pending.delete(documentType);\n }\n this.#notifyPendingChanged();\n }\n\n dismissInstallation(packageName: string): void {\n const affectedTypes = this.#findTypesByPackage(packageName);\n if (affectedTypes.length === 0) return;\n\n for (const documentType of affectedTypes) {\n this.#dismissedStorage.set(documentType, true);\n const entry = this.#deferred.get(documentType);\n if (entry) {\n entry.reject(\n new Error(`Document type \"${documentType}\" was dismissed`),\n );\n }\n this.#deferred.delete(documentType);\n this.#pending.delete(documentType);\n this.#discoveredTypes.delete(documentType);\n this.#recordFailure(documentType, \"dismissed\", [packageName], null);\n }\n\n this.#emitEvent({\n type: \"installation-dismissed\",\n packageName,\n documentTypes: affectedTypes,\n });\n this.#notifyPendingChanged();\n }\n\n getPendingInstallations(): PendingInstallation[] {\n return this.#pendingMemo;\n }\n\n subscribePending(listener: () => void): () => void {\n this.#pendingSubscribers.add(listener);\n return () => {\n this.#pendingSubscribers.delete(listener);\n };\n }\n\n getFailedInstallations(): FailedInstallation[] {\n return this.#failedMemo;\n }\n\n subscribeFailed(listener: () => void): () => void {\n this.#failedSubscribers.add(listener);\n return () => {\n this.#failedSubscribers.delete(listener);\n };\n }\n\n subscribeEvents(listener: DiscoveryEventListener): () => void {\n this.#eventSubscribers.add(listener);\n return () => {\n this.#eventSubscribers.delete(listener);\n };\n }\n\n async retryInstallation(documentType: string): Promise<void> {\n this.#dismissedStorage.delete(documentType);\n this.#clearFailed(documentType);\n try {\n await this.#discover(documentType);\n } catch {\n // failure has been recorded via #recordFailure; swallow so the\n // caller (UI button) doesn't have to handle the rejection\n }\n }\n\n async #discover(documentType: string): Promise<DocumentModelModule<any>> {\n let packageNames: string[];\n try {\n packageNames =\n await this.#registryClient.getPackagesByDocumentType(documentType);\n } catch (error) {\n const normalized =\n error instanceof Error ? error : new Error(String(error));\n this.#emitEvent({\n type: \"registry-query-failed\",\n documentType,\n error: normalized,\n });\n this.#recordFailure(documentType, \"registry-error\", [], normalized);\n return Promise.reject(normalized);\n }\n\n if (packageNames.length === 0) {\n this.#recordFailure(documentType, \"not-in-registry\", [], null);\n return Promise.reject(\n new Error(`No packages found for document type \"${documentType}\"`),\n );\n }\n\n const entry = this.#createDeferredEntry(documentType, packageNames);\n\n this.#emitEvent({\n type: \"type-discovered\",\n documentType,\n packageNames,\n });\n\n if (this.#mode === \"immediate\") {\n this.#addToPending(documentType, packageNames);\n } else {\n this.#discoveredTypes.set(documentType, packageNames);\n }\n\n return entry.promise;\n }\n\n #createDeferredEntry(\n documentType: string,\n packageNames: string[],\n ): DeferredEntry {\n let resolve!: (module: DocumentModelModule<any>) => void;\n let reject!: (reason: unknown) => void;\n const promise = new Promise<DocumentModelModule<any>>((res, rej) => {\n resolve = res;\n reject = rej;\n });\n const entry: DeferredEntry = { packageNames, resolve, reject, promise };\n this.#deferred.set(documentType, entry);\n return entry;\n }\n\n #addToPending(documentType: string, packageNames: string[]): void {\n const installation: PendingInstallation = { documentType, packageNames };\n this.#pending.set(documentType, installation);\n this.#emitEvent({\n type: \"installation-prompted\",\n documentType,\n packageNames,\n });\n this.#notifyPendingChanged();\n }\n\n #findModuleInLoadedPackages(\n documentType: string,\n ): DocumentModelModule<any> | undefined {\n return this.#packageManager.packages\n .flatMap((p) => p.documentModels)\n .find((m) => m.documentModel.global.id === documentType);\n }\n\n #findTypesByPackage(packageName: string): string[] {\n const types: string[] = [];\n for (const [documentType, entry] of this.#deferred) {\n if (entry.packageNames.includes(packageName)) {\n types.push(documentType);\n }\n }\n return types;\n }\n\n #notifyPendingChanged(): void {\n this.#pendingMemo = Array.from(this.#pending.values());\n for (const listener of this.#pendingSubscribers) {\n listener();\n }\n }\n\n #notifyFailedChanged(): void {\n this.#failedMemo = Array.from(this.#failed.values());\n for (const listener of this.#failedSubscribers) {\n listener();\n }\n }\n\n #recordFailure(\n documentType: string,\n reason: FailedInstallationReason,\n packageNames: string[],\n error: Error | null,\n ): void {\n this.#failed.set(documentType, {\n documentType,\n reason,\n packageNames,\n error,\n });\n this.#emitEvent({ type: \"load-failed\", documentType, reason });\n this.#notifyFailedChanged();\n }\n\n #clearFailed(documentType: string): void {\n if (this.#failed.delete(documentType)) {\n this.#notifyFailedChanged();\n }\n }\n\n #emitEvent(event: DiscoveryEvent): void {\n for (const listener of this.#eventSubscribers) {\n listener(event);\n }\n }\n}\n","import * as common from \"@powerhousedao/powerhouse-vetra-packages\";\nimport commonPkg from \"@powerhousedao/powerhouse-vetra-packages/package.json\" with { type: \"json\" };\nimport type {\n IPackagesListener,\n PackageManagerInstallResult,\n} from \"@powerhousedao/reactor-browser\";\nimport {\n BrowserLocalStorage,\n type IPackageListerUnsubscribe,\n type IPackageManager,\n} from \"@powerhousedao/reactor-browser\";\nimport {\n type DocumentModelLib,\n type DocumentModelModule,\n} from \"@powerhousedao/shared/document-model\";\nimport * as vetra from \"@powerhousedao/vetra\";\nimport vetraPkg from \"@powerhousedao/vetra/package.json\" with { type: \"json\" };\n\ntype PackageMeta = {\n name: string;\n importUrl: string | null;\n stylesheetUrl: string | null;\n version?: string;\n /**\n * Full spec the user asked for (e.g. `@scope/pkg@2.0.0-beta.2`). Kept so a\n * reload re-installs exactly the tag/version originally picked rather than\n * sliding to `latest`. Absent for legacy entries and local packages.\n */\n spec?: string;\n};\n\n/**\n * Strip any `@tag` / `@version` suffix from a package spec, returning the bare\n * package name. Mirrors `parsePackageSpec` on the UI side.\n */\nfunction parseBareName(spec: string): string {\n const trimmed = spec.trim();\n const at = trimmed.startsWith(\"@\")\n ? trimmed.lastIndexOf(\"@\")\n : trimmed.indexOf(\"@\");\n return at > 0 ? trimmed.slice(0, at) : trimmed;\n}\n\ntype PackageWithMeta = PackageMeta & {\n loadedPackage: DocumentModelLib<any>;\n spec?: string;\n};\n\nasync function fetchPackageJsonVersion(\n baseUrl: string,\n): Promise<string | undefined> {\n try {\n const res = await fetch(baseUrl);\n if (!res.ok) return undefined;\n const pkg = (await res.json()) as { version?: unknown };\n return typeof pkg.version === \"string\" ? pkg.version : undefined;\n } catch {\n return undefined;\n }\n}\n\nconst LOCAL_PACKAGE_NAME = \"Local\" as const;\n\nexport class BrowserPackageManager implements IPackageManager {\n registryUrl: string | null;\n #storage: BrowserLocalStorage<PackageMeta>;\n #packages: Map<string, DocumentModelLib<any>> = new Map();\n #subscribers = new Set<IPackagesListener>();\n #packagesMemo: DocumentModelLib<any>[] = [];\n #stylesheets: Map<string, HTMLStyleElement> = new Map();\n #localPackage: DocumentModelLib<any> | undefined;\n\n #cdnUrl: string | null;\n #localPackageVersion: string | undefined;\n #localPackageNames: Set<string> = new Set([LOCAL_PACKAGE_NAME]);\n\n constructor(namespace: string, registryUrl: string | null) {\n this.#storage = new BrowserLocalStorage<PackageMeta>(\n namespace + \":PH_PACKAGES\",\n );\n this.registryUrl = registryUrl;\n this.#cdnUrl = registryUrl !== null ? this.#toCdnUrl(registryUrl) : null;\n }\n\n #toCdnUrl(baseUrl: string): string {\n if (baseUrl.includes(\"/-/cdn\")) return baseUrl;\n const base = baseUrl.endsWith(\"/\") ? baseUrl.slice(0, -1) : baseUrl;\n return `${base}/-/cdn`;\n }\n\n async init(\n localPackage?: DocumentModelLib<any>,\n localPackageVersion?: string,\n ) {\n this.addLocalPackage(common.manifest.name, common, commonPkg.version);\n this.addLocalPackage(vetra.manifest.name, vetra, vetraPkg.version);\n if (localPackage) {\n this.updateLocalPackage(localPackage, localPackageVersion);\n }\n for (const packageName of this.#storage.keys()) {\n // Re-hydrate with the originally-requested spec so a tag/version pick\n // sticks across reloads. Fall back to the bare key for legacy entries\n // (pre-spec) and for entries that never had a tag.\n const existingMeta = this.#storage.get(packageName);\n const specForReload = existingMeta?.spec ?? packageName;\n console.debug(\n `[Connect][PackageManager] Rehydrating \"${packageName}\" via spec \"${specForReload}\"`,\n );\n const result = await this.addPackage(specForReload);\n // Previously-installed package that no longer resolves (version\n // withdrawn from npm, registry moved, etc.) would otherwise\n // 404-toast on every boot forever. Drop it from persistent storage\n // so the failure is one-shot instead of sticky.\n if (result.type === \"error\") {\n this.#storage.delete(packageName);\n }\n }\n }\n\n addLocalPackage(\n name: string,\n loadedPackage: DocumentModelLib<any>,\n version?: string,\n ) {\n this.#localPackageNames.add(name);\n this.#registerPackage({\n name,\n importUrl: null,\n stylesheetUrl: null,\n loadedPackage,\n version,\n });\n }\n\n updateLocalPackage(pkg: DocumentModelLib<any>, version?: string) {\n console.debug(\"Updating local package:\", pkg);\n this.#localPackage = pkg;\n this.#registerPackage({\n name: LOCAL_PACKAGE_NAME,\n stylesheetUrl: null,\n importUrl: null,\n loadedPackage: pkg,\n });\n if (version) {\n this.#localPackageVersion = version;\n this.#notifyPackagesChanged();\n return;\n }\n fetchPackageJsonVersion(\"/package.json\")\n .then((fetchedVersion) => {\n this.#localPackageVersion = fetchedVersion;\n if (fetchedVersion) this.#notifyPackagesChanged();\n })\n .catch(() => {});\n }\n\n get packages() {\n return this.#packagesMemo;\n }\n\n get cdnUrl(): string | null {\n return this.#cdnUrl;\n }\n\n getPackageSource(packageName: string) {\n // check vs packages registered as local (Common, Vetra, bundled packages...)\n if (this.#localPackageNames.has(packageName)) {\n return \"common\";\n }\n // check if the package has the same name as the local project\n if (packageName === this.#localPackage?.manifest.name) return \"project\";\n const packageMeta = this.#storage.get(packageName);\n // if meta does not exist the package is not installed\n if (!packageMeta) return null;\n // if imported from node_modules then the package is installed locally\n if (packageMeta.importUrl === `/node_modules/${packageName}`)\n return \"local-install\";\n // all other import urls point to a registry\n return \"registry-install\";\n }\n\n /**\n * Return the registry-installed packages keyed by their storage name (which\n * is the registry spec, e.g. \"@powerhousedao/clint-common\"), with the\n * installed version when known. Excludes bundled (common) packages, the\n * local project package, and packages resolved out of `/node_modules/`.\n */\n getRegistryPackages(): { name: string; version: string | undefined }[] {\n const out: { name: string; version: string | undefined }[] = [];\n for (const [name, meta] of this.#storage) {\n if (this.#localPackageNames.has(name)) continue;\n if (name === LOCAL_PACKAGE_NAME) continue;\n if (meta.importUrl === `/node_modules/${name}`) continue;\n out.push({ name, version: meta.version });\n }\n return out;\n }\n\n getPackageVersion(packageName: string): string | undefined {\n if (packageName === this.#localPackage?.manifest.name) {\n return this.#localPackageVersion;\n }\n return this.#storage.get(packageName)?.version;\n }\n\n async addPackage(packageSpec: string): Promise<PackageManagerInstallResult> {\n // `packageSpec` may include a `@tag` / `@version` suffix (e.g. user picked\n // a specific version in the package manager UI, or re-hydrated on reload\n // from the persisted spec). We always register/look up by the bare name so\n // status tracking, version lookups, and uninstall all go through a single\n // canonical key. The spec itself is kept on the meta so reloads re-fetch\n // the same tag.\n const bareName = parseBareName(packageSpec);\n const hasTagOrVersion = bareName !== packageSpec;\n console.debug(\n `[Connect][PackageManager] addPackage spec=\"${packageSpec}\" bareName=\"${bareName}\"`,\n );\n\n const existingPackage = this.#packages.get(bareName);\n if (existingPackage) {\n const requestedVersion = hasTagOrVersion\n ? packageSpec.slice(bareName.length + 1)\n : undefined;\n const currentVersion = this.#storage.get(bareName)?.version;\n const isLocal = this.#localPackageNames.has(bareName);\n const sameVersion =\n !requestedVersion ||\n !currentVersion ||\n requestedVersion === currentVersion;\n if (isLocal || sameVersion) {\n console.debug(\n `[Connect][PackageManager] \"${bareName}\" already loaded; skipping re-fetch`,\n );\n return {\n type: \"success\",\n package: existingPackage,\n };\n }\n console.debug(\n `[Connect][PackageManager] \"${bareName}\" version bump ${currentVersion} → ${requestedVersion}; refetching`,\n );\n }\n try {\n const packageWithMeta = await this.#loadPackage(packageSpec);\n // `#loadPackage*` set `name` to whatever spec it built the URL from.\n // Canonicalize to the bare name for the map/storage keys, and stash the\n // original spec separately so reloads re-use it.\n packageWithMeta.name = bareName;\n if (hasTagOrVersion) {\n packageWithMeta.spec = packageSpec;\n }\n // Swap stylesheets only once the new package is in hand. If `#loadPackage`\n // throws above, the old stylesheet stays mounted and the UI keeps\n // rendering the previous version instead of unstyled markup.\n if (existingPackage) {\n this.#unmountStylesheet(bareName);\n }\n this.#registerPackage(packageWithMeta);\n\n return {\n type: \"success\",\n package: packageWithMeta.loadedPackage,\n };\n } catch (error) {\n const normalized =\n error instanceof Error ? error : new Error(String(error));\n console.error(\n `[Connect][PackageManager] Failed to install package \"${packageSpec}\": ${normalized.message}`,\n normalized,\n );\n return {\n type: \"error\",\n error: normalized,\n };\n }\n }\n\n async addPackages(packageNames: string[]) {\n const results: PackageManagerInstallResult[] = [];\n for (const packageName of packageNames) {\n const result = await this.addPackage(packageName);\n results.push(result);\n }\n return results;\n }\n\n removePackage(name: string) {\n this.#packages.delete(name);\n this.#storage.delete(name);\n this.#unmountStylesheet(name);\n this.#notifyPackagesChanged();\n }\n\n subscribe(handler: IPackagesListener): IPackageListerUnsubscribe {\n this.#subscribers.add(handler);\n return () => {\n this.#subscribers.delete(handler);\n };\n }\n\n load(documentType: string): Promise<DocumentModelModule<any>> {\n const documentModelModule = Array.from(\n this.#packages.values().flatMap((p) => p.documentModels),\n ).find((m) => m.documentModel.global.id === documentType);\n\n if (documentModelModule) return Promise.resolve(documentModelModule);\n return Promise.reject(new Error(\"Model not available\"));\n }\n\n async #loadPackageFromNodeModules(name: string): Promise<PackageWithMeta> {\n const importUrl = `/node_modules/${name}/browser/index.js`;\n const stylesheetUrl = `/node_modules/${name}/style.css`;\n\n const packageWithMeta = await this.#importPackage({\n name,\n importUrl,\n stylesheetUrl,\n });\n packageWithMeta.version = await fetchPackageJsonVersion(\n `/node_modules/${name}/package.json`,\n );\n\n return packageWithMeta;\n }\n\n async #loadPackageFromRegistry(name: string): Promise<PackageWithMeta> {\n const importUrl = `${this.#cdnUrl}/${name}/browser/index.js`;\n const stylesheetUrl = `${this.#cdnUrl}/${name}/style.css`;\n const packageWithMeta = await this.#importPackage({\n name,\n importUrl,\n stylesheetUrl,\n });\n packageWithMeta.version = await fetchPackageJsonVersion(\n `${this.#cdnUrl}/${name}/package.json`,\n );\n\n return packageWithMeta;\n }\n\n async #importPackage(packageMeta: PackageMeta): Promise<PackageWithMeta> {\n const { name, importUrl, stylesheetUrl } = packageMeta;\n if (!importUrl) {\n throw new Error(`Import url not defined for package \"${name}\".`);\n }\n\n const loadedPackage = (await import(\n /* @vite-ignore */ importUrl\n )) as DocumentModelLib<any>;\n\n return {\n name,\n loadedPackage,\n importUrl,\n stylesheetUrl,\n };\n }\n\n async #loadPackage(packageName: string): Promise<PackageWithMeta> {\n if (this.#localPackageNames.has(packageName)) {\n throw new Error(\n `Package \"${packageName}\" is a local package and cannot be loaded dynamically.`,\n );\n }\n\n // only attemp to load from node_modules in dev mode\n if (!import.meta.env.PROD) {\n try {\n const packageWithMeta =\n await this.#loadPackageFromNodeModules(packageName);\n return packageWithMeta;\n } catch (error) {\n console.warn(\n `Failed to load package \"${packageName}\" from node_modules:`,\n error,\n );\n }\n }\n\n if (!this.registryUrl) {\n throw new Error(\"Registry url not defined.\");\n }\n\n return await this.#loadPackageFromRegistry(packageName);\n }\n\n #registerPackage(packageWithMeta: PackageWithMeta) {\n const { name, loadedPackage, importUrl, stylesheetUrl, version, spec } =\n packageWithMeta;\n\n if (stylesheetUrl !== null) {\n this.#mountStylesheet(name, stylesheetUrl);\n }\n this.#packages.set(name, loadedPackage);\n this.#storage.set(name, {\n name,\n importUrl,\n stylesheetUrl,\n version,\n ...(spec ? { spec } : {}),\n });\n console.debug(\n `[Connect][PackageManager] Registered \"${name}\" (version=${version ?? \"?\"}, spec=${spec ?? \"—\"})`,\n );\n\n this.#notifyPackagesChanged();\n }\n\n #mountStylesheet(name: string, href: string) {\n const existing = this.#stylesheets.get(name);\n if (existing) return existing;\n\n const style = document.createElement(\"style\");\n style.textContent = `@import url(\"${href}\") layer(external-packages);`;\n document.head.appendChild(style);\n\n this.#stylesheets.set(name, style);\n }\n\n #unmountStylesheet(name: string): void {\n const style = this.#stylesheets.get(name);\n if (!style) return;\n\n style.remove();\n this.#stylesheets.delete(name);\n }\n\n #notifyPackagesChanged() {\n this.#packagesMemo = Array.from(this.#packages.values());\n const packages = this.packages;\n this.#subscribers.forEach((handler) => {\n handler({ packages });\n });\n }\n}\n","import type { PGliteWorker } from \"@electric-sql/pglite/worker\";\nimport { createRelationalDb } from \"@powerhousedao/shared/processors\";\nimport { Kysely } from \"kysely\";\nimport { PGliteDialect } from \"kysely-pglite-dialect\";\nimport {\n detectRelationalPgMajor,\n resolvePgMajorForRuntime,\n type SupportedPgMajor,\n} from \"./utils/pglite-runtime.js\";\nimport { RELATIONAL_PGLITE_NAME } from \"./utils/storage-namespace.js\";\n\nasync function createPGliteWorkerForMajor(\n major: SupportedPgMajor,\n): Promise<PGliteWorker> {\n // The worker reads the data-dir name from meta so the namespace is owned on\n // the main thread alongside the other origin-scoped stores.\n const meta = { dbName: RELATIONAL_PGLITE_NAME };\n if (major === 16) {\n const [legacyWorker, legacyLive] = await Promise.all([\n import(\"pglite-legacy-02/worker\"),\n import(\"pglite-legacy-02/live\"),\n ]);\n const worker = new Worker(\n new URL(\"./pglite.worker.legacy.js\", import.meta.url),\n { type: \"module\" },\n );\n return legacyWorker.PGliteWorker.create(worker, {\n meta,\n extensions: { live: legacyLive.live },\n }) as unknown as PGliteWorker;\n }\n const [{ PGliteWorker }, { live }] = await Promise.all([\n import(\"@electric-sql/pglite/worker\"),\n import(\"@electric-sql/pglite/live\"),\n ]);\n const worker = new Worker(new URL(\"./pglite.worker.js\", import.meta.url), {\n type: \"module\",\n });\n return PGliteWorker.create(worker, { meta, extensions: { live } });\n}\n\nexport async function getDb() {\n const detected = await detectRelationalPgMajor();\n const major = resolvePgMajorForRuntime(detected);\n if (major !== 17) {\n console.warn(\n `[reactor] Relational worker is opening legacy Postgres ${major} data dir. Migrate to PG17 from the banner or the Inspector → Debug tab.`,\n );\n }\n\n const pgLite = await createPGliteWorkerForMajor(major);\n\n const kysely = new Kysely({\n dialect: new PGliteDialect(pgLite),\n });\n\n const relationalDb = createRelationalDb(kysely);\n\n return { pgLite, relationalDb };\n}\n","import {\n createAttachmentClient,\n type AttachmentHeader,\n type AttachmentResponse,\n type IAttachmentService,\n} from \"@powerhousedao/reactor-attachments/client\";\nimport {\n createAnalyticsStore,\n type IReactorClient,\n type IReactorProcessorHostModule,\n} from \"@powerhousedao/reactor-browser\";\nimport { getDb } from \"../pglite.db.js\";\n\n/** @deprecated Use IReactorClient from @powerhousedao/reactor-browser */\nexport type IReactorDispatch = IReactorClient;\n\nclass NullAttachmentService implements IAttachmentService {\n reserve(): Promise<never> {\n return Promise.reject(\n new Error(\"NullAttachmentService: no attachment service configured\"),\n );\n }\n stat(): Promise<AttachmentHeader> {\n return Promise.reject(\n new Error(\"NullAttachmentService: no attachment service configured\"),\n );\n }\n get(): Promise<AttachmentResponse> {\n return Promise.reject(\n new Error(\"NullAttachmentService: no attachment service configured\"),\n );\n }\n}\n\ninterface INamedReadModel {\n readonly name: string;\n}\n\nexport async function createProcessorHostModule(\n reactorClient: IReactorClient,\n readModels: INamedReadModel[],\n attachmentService?: IAttachmentService,\n): Promise<IReactorProcessorHostModule | undefined> {\n try {\n const { pgLite, relationalDb } = await getDb();\n const { store: analyticsStore } = await createAnalyticsStore({\n pgLite,\n });\n const processorApp = \"connect\" as const;\n return {\n relationalDb,\n analyticsStore,\n processorApp,\n client: reactorClient,\n attachments: createAttachmentClient(\n attachmentService ?? new NullAttachmentService(),\n ),\n dispatch: {\n async execute(docId, branch, actions, signal) {\n const jobInfo = await reactorClient.executeAsync(\n docId,\n branch,\n actions,\n signal,\n );\n return { id: jobInfo.id, status: jobInfo.status };\n },\n },\n getReadModel<T>(name: string): T {\n const model = readModels.find((m) => m.name === name);\n if (!model) {\n throw new Error(`Read model \"${name}\" not found`);\n }\n return model as unknown as T;\n },\n };\n } catch (error) {\n console.error(`Failed to initialize processor host module:`);\n console.error(error);\n }\n}\n","import {\n buildPHGlobalConfig,\n phGlobalConfig,\n} from \"@powerhousedao/connect/config\";\nimport { toast } from \"@powerhousedao/connect/services\";\nimport {\n addDefaultDrivesForNewReactor,\n createBrowserReactor,\n getDefaultDrives,\n} from \"@powerhousedao/connect/utils\";\nimport { createRemoteAttachmentService } from \"@powerhousedao/reactor-attachments/client\";\nimport {\n addPHEventHandlers,\n addRemoteDrive,\n DocumentCache,\n DocumentChangeType,\n extractDriveSlugFromPath,\n extractNodeSlugFromPath,\n getDrives,\n login,\n refreshReactorDataClient,\n RegistryClient,\n setAttachmentService,\n setDefaultPHGlobalConfig,\n setDocumentCache,\n setDrives,\n setFeatures,\n setPackageDiscoveryService,\n setPHToast,\n setReactorClient,\n setReactorClientModule,\n setRenown,\n setSelectedDrive,\n setSelectedNode,\n setVetraPackageManager,\n type IPackageManager,\n type PHToastFn,\n} from \"@powerhousedao/reactor-browser\";\nimport {\n BrowserKeyStorage,\n RenownBuilder,\n RenownCryptoBuilder,\n} from \"@renown/sdk\";\nimport {\n logger,\n type DocumentModelLib,\n type UpgradeManifest,\n} from \"document-model\";\nimport { initFeatureFlags } from \"../feature-flags.js\";\nimport { NoRegistryDiscoveryService } from \"../no-registry-discovery.js\";\nimport { PackageDiscoveryService } from \"../package-discovery.js\";\nimport { BrowserPackageManager } from \"../package-manager.js\";\nimport { getRuntimeConfig } from \"../runtime-config.js\";\nimport { createProcessorHostModule } from \"./processor-host-module.js\";\n\n/**\n * Subscribe to the `/__packages` SSE channel exposed by ph-clint's\n * static-mode `connect-server.js`. On each `packages-changed` event the\n * server sends the full live list; we diff against what the packageManager\n * already has loaded and call `addPackage`/`removePackage` to converge.\n *\n * Best-effort — silently no-ops when the SSE endpoint doesn't exist\n * (e.g. running under vite dev, or hosted somewhere without this protocol).\n */\nfunction subscribeToPackagesChannel(packageManager: IPackageManager): void {\n if (typeof window === \"undefined\" || typeof EventSource === \"undefined\") {\n return;\n }\n let source: EventSource;\n try {\n source = new EventSource(\"/__packages\");\n } catch (err) {\n console.debug(\"[Connect] /__packages subscribe failed:\", err);\n return;\n }\n let firstEvent = true;\n source.addEventListener(\"packages-changed\", (event) => {\n try {\n const payload = JSON.parse((event as MessageEvent<string>).data) as {\n packages?: unknown;\n };\n if (!Array.isArray(payload.packages)) return;\n const next = payload.packages.filter(\n (p): p is string => typeof p === \"string\",\n );\n // Split each incoming spec into (bareName, version). The server may\n // send either bare names (\"@scope/pkg\") or version-qualified specs\n // (\"@scope/pkg@1.2.3\"). parseBareName-style logic: for scoped specs\n // the first `@` belongs to the scope, so version starts after the\n // last `@` only when that `@` is past index 0.\n const parseBare = (\n spec: string,\n ): { bareName: string; version?: string } => {\n const at = spec.startsWith(\"@\")\n ? spec.lastIndexOf(\"@\")\n : spec.indexOf(\"@\");\n if (at > 0) {\n return { bareName: spec.slice(0, at), version: spec.slice(at + 1) };\n }\n return { bareName: spec };\n };\n\n // Diff against the registry-tracked subset only. Bundled \"common\"\n // packages and the project's local package never appear in the\n // server's list and removing them on every event would wipe the\n // drive editors and break the AddDrive modal.\n const currentByName = new Map(\n packageManager\n .getRegistryPackages()\n .map(({ name, version }) => [name, version]),\n );\n const nextByName = new Map<string, string | undefined>();\n for (const spec of next) {\n const { bareName, version } = parseBare(spec);\n nextByName.set(bareName, version);\n }\n\n const isFirst = firstEvent;\n firstEvent = false;\n\n for (const name of currentByName.keys()) {\n if (!nextByName.has(name)) {\n packageManager.removePackage(name);\n if (!isFirst) {\n toast(`Removed package ${name}`, { type: \"connect-deleted\" });\n }\n }\n }\n for (const spec of next) {\n const { bareName, version } = parseBare(spec);\n const currentVersion = currentByName.get(bareName);\n const isKnown = currentByName.has(bareName);\n // Skip when the package is already present at the same version (or\n // when no version info is available on either side to compare).\n if (\n isKnown &&\n (!version || !currentVersion || version === currentVersion)\n ) {\n continue;\n }\n const isUpdate = isKnown;\n Promise.resolve(packageManager.addPackage(spec)).then(\n (result) => {\n if (result.type === \"error\") {\n console.error(\n `[Connect] /__packages addPackage(${spec}) failed:`,\n result.error,\n );\n return;\n }\n if (isFirst) return;\n const name = result.package.manifest.name;\n toast(\n isUpdate\n ? `Updated package ${name}`\n : `Installed package ${name}`,\n { type: \"connect-success\" },\n );\n },\n (err: unknown) => {\n console.error(\n `[Connect] /__packages addPackage(${spec}) threw:`,\n err,\n );\n },\n );\n }\n } catch (err) {\n console.error(\"[Connect] /__packages event parse failed:\", err);\n }\n });\n source.addEventListener(\"error\", () => {\n // EventSource auto-reconnects; nothing to do.\n });\n}\n\nexport async function clearReactorStorage() {\n await window.ph?.reactorClientModule?.pg?.close();\n\n // Dropping tables inside an existing PGlite instance is unreliable with\n // `relaxedDurability: true` followed by an immediate page reload — pending\n // IDB writes can be lost. Deleting the underlying database outright sidesteps\n // flush-timing; the next startup re-creates and re-migrates from scratch.\n const dbs = await indexedDB.databases();\n const targets = dbs\n .map((d) => d.name)\n .filter(\n (n): n is string =>\n !!n && !n.startsWith(\"ph-pglite-backup::\") && /pglite|reactor/i.test(n),\n );\n\n await Promise.all(\n targets.map(\n (name) =>\n new Promise<void>((resolve) => {\n const req = indexedDB.deleteDatabase(name);\n req.onsuccess = req.onerror = req.onblocked = () => resolve();\n }),\n ),\n );\n}\n\nexport async function createReactor(localPackage?: DocumentModelLib) {\n if (!window.ph) {\n window.ph = {};\n }\n if (window.ph.loading) return;\n\n window.ph.loading = true;\n\n // add window event handlers for updates\n addPHEventHandlers();\n\n // register toast function for use in editor components\n setPHToast(toast as PHToastFn);\n\n // initialize feature flags\n const features = await initFeatureFlags();\n\n logger.info(\n \"Features: @features\",\n JSON.stringify(Object.fromEntries(features), null, 2),\n );\n\n // initialize renown crypto\n const keyPairStorage = await BrowserKeyStorage.create();\n const renownCrypto = await new RenownCryptoBuilder()\n .withKeyPairStorage(keyPairStorage)\n .build();\n\n // initialize Renown\n const renown = await new RenownBuilder(\"connect\", {\n basename: phGlobalConfig.routerBasename,\n baseUrl: phGlobalConfig.renownUrl,\n })\n .withCrypto(renownCrypto)\n .build();\n\n // Read the runtime config from cache. loadComponent() awaited\n // loadRuntimeConfig() before calling createReactor, so the cache is warm;\n // using the sync getter matches the convention used by connect.config.ts,\n // useRegistryPackages, and pages/content.tsx — and throws loudly if a\n // future caller violates the boot ordering.\n const runtimeConfig = getRuntimeConfig();\n\n // initialize package manager\n const packageManager = new BrowserPackageManager(\n phGlobalConfig.routerBasename ?? \"\",\n runtimeConfig.packageRegistryUrl ?? null,\n );\n setVetraPackageManager(packageManager);\n await packageManager.init(localPackage, runtimeConfig.localPackage?.version);\n // Register any packages marked as provider: \"local\" in powerhouse.config.json\n // that the vite plugin bundled into this build. The virtual module is only\n // emitted when `phBundledPackagesPlugin` is registered (ph-cli's Connect\n // flow); running `vite dev` against apps/connect's own config has no\n // bundled packages, so a resolution failure here is expected. The\n // indirection + @vite-ignore also keeps Vite's dep scanner from treating\n // the specifier as a real npm package to pre-bundle.\n try {\n const bundledPackagesModule = \"ph-bundled-packages-virtual\";\n const { default: registerBundledPackages } = (await import(\n /* @vite-ignore */ bundledPackagesModule\n )) as { default: (pm: IPackageManager) => void };\n registerBundledPackages(packageManager);\n } catch {\n // no bundled packages in this build\n }\n const remotePackages = runtimeConfig.packages\n .filter((p) => p.provider !== \"local\")\n .map((p) => (p.version ? `${p.packageName}@${p.version}` : p.packageName));\n const packagesResult = await packageManager.addPackages(remotePackages);\n packagesResult.map((r) => {\n if (r.type === \"error\") console.error(r.error);\n });\n\n // Opt-in: subscribe to the static-mode `/__packages` SSE channel so live\n // publishes (e.g. ph-clint's publish-reload trigger pushing a new list) flow\n // into the running tab without a page reload. Enabled via\n // `connect.packages.liveReload`; the channel only exists in static hosting\n // that speaks this protocol.\n if (runtimeConfig.connect?.packages?.liveReload) {\n subscribeToPackagesChannel(packageManager);\n }\n\n // get document models to set in the reactor (all versions)\n const documentModelModules = packageManager.packages\n .flatMap((pkg) => pkg.documentModels)\n .filter(\n (module, index, modules) =>\n // deduplicate by documentType and version\n modules.findIndex(\n (m) =>\n m.documentModel.global.id === module.documentModel.global.id &&\n m.version === module.version,\n ) === index,\n );\n\n // get upgrade manifests from packages\n const upgradeManifests = packageManager.packages\n .flatMap((pkg) => pkg.upgradeManifests)\n .filter(\n (manifest, index, manifests) =>\n // deduplicate by documentType and version\n manifest !== undefined &&\n manifests.findIndex(\n (m) => m && m.documentType === manifest.documentType,\n ) === index,\n ) as UpgradeManifest<readonly number[]>[];\n\n // initialize package discovery service for auto-installing unknown document types\n const discoveryService =\n packageManager.cdnUrl !== null\n ? new PackageDiscoveryService(\n packageManager,\n new RegistryClient(packageManager.cdnUrl),\n {\n mode: \"immediate\",\n storageKey: phGlobalConfig.routerBasename ?? \"\",\n },\n )\n : new NoRegistryDiscoveryService(packageManager);\n\n setPackageDiscoveryService(discoveryService);\n\n // create reactor v2 with all versions and upgrade manifests\n const reactorClientModule = await createBrowserReactor(\n documentModelModules,\n upgradeManifests,\n renown,\n discoveryService,\n );\n\n // get the drives from the reactor\n const drives = await getDrives(reactorClientModule.client);\n\n // initialize user from URL parameter\n const didFromUrl = getDidFromUrl();\n await login(didFromUrl, renown);\n\n const documentCache = new DocumentCache(reactorClientModule.client);\n\n // dispatch the events to set the values in the window object\n const basePath = phGlobalConfig.basePath ?? \"/\";\n const routerBasename = phGlobalConfig.routerBasename ?? \"/\";\n const mergedGlobalConfig = buildPHGlobalConfig(\n basePath,\n routerBasename,\n runtimeConfig.connect ?? {},\n );\n setDefaultPHGlobalConfig(mergedGlobalConfig);\n\n // Slug extraction needs window.ph.basePath, published by\n // setDefaultPHGlobalConfig above.\n const path = window.location.pathname;\n const driveSlug = extractDriveSlugFromPath(path);\n const nodeSlug = extractNodeSlugFromPath(path);\n setReactorClientModule(reactorClientModule);\n setReactorClient(reactorClientModule.client);\n\n const _defaultDrivesUrl = phGlobalConfig.defaultDrivesUrl;\n if (_defaultDrivesUrl) {\n let switchboardOrigin: string | undefined;\n try {\n switchboardOrigin = new URL(_defaultDrivesUrl).origin;\n } catch {\n // malformed URL — skip attachment service construction\n }\n if (switchboardOrigin) {\n const attachmentJwtHandler = async (_url: string) => {\n if (!renown.user) return undefined;\n // aud omitted: server verifies without an audience, so aud-bearing\n // tokens are rejected. Re-enable once both sides support it.\n return renown.getBearerToken({ expiresIn: 10 });\n };\n const attachmentService = createRemoteAttachmentService({\n remoteUrl: switchboardOrigin,\n jwtHandler: attachmentJwtHandler,\n });\n setAttachmentService(attachmentService);\n }\n }\n\n setDocumentCache(documentCache);\n setRenown(renown);\n setDrives(drives);\n setFeatures(features);\n\n // Add default drives and any URL-supplied remote drive in the background so\n // the app renders immediately. setSelectedDrive defers selection until the\n // drive matching the URL slug syncs in (see deferDriveSelection), so we only\n // ever wait for the selected drive, never for unrelated default drives.\n const defaultDrivesConfig = getDefaultDrives(runtimeConfig);\n if (defaultDrivesConfig.length > 0) {\n void addDefaultDrivesForNewReactor(defaultDrivesConfig).catch((error) =>\n console.error(\"Failed to add default drives:\", error),\n );\n }\n\n const remoteUrl = getDriveUrl();\n if (remoteUrl) {\n void addRemoteDrive(remoteUrl, undefined).catch((error) =>\n console.error(`Failed to add remote drive from ${remoteUrl}:`, error),\n );\n }\n\n setSelectedDrive(driveSlug);\n setSelectedNode(nodeSlug);\n\n // Subscribe via ReactorClient interface\n const reactorClient = reactorClientModule.client;\n reactorClient.subscribe({ type: \"powerhouse/document-drive\" }, (event) => {\n logger.verbose(\"ReactorClient subscription event: @event\", event);\n refreshReactorDataClient(reactorClientModule.client).catch((e) =>\n logger.error(\"@error\", e),\n );\n });\n\n // Redirect when a currently-viewed document or drive is deleted remotely\n reactorClient.subscribe({}, (event) => {\n if (event.type !== DocumentChangeType.Deleted) return;\n const deletedId = event.context?.childId;\n if (!deletedId) return;\n\n const selectedDriveId = window.ph?.selectedDriveId;\n const selectedNodeId = window.ph?.selectedNodeId;\n\n if (selectedDriveId && deletedId === selectedDriveId) {\n setSelectedDrive(undefined);\n toast(\"The drive you were viewing has been deleted\");\n return;\n }\n\n if (selectedNodeId && deletedId === selectedNodeId) {\n setSelectedNode(undefined);\n toast(\"The document you were editing has been deleted\");\n }\n });\n\n // Refresh from ReactorClient to pick up any synced drives\n await refreshReactorDataClient(reactorClientModule.client);\n\n // Setup processor factories for packages that have them\n const packagesWithProcessorFactories = packageManager.packages.filter(\n (pkg) => pkg.processorFactory !== undefined,\n );\n\n if (packagesWithProcessorFactories.length > 0) {\n const readModels =\n reactorClientModule.reactorModule?.readModelCoordinator?.readModels ?? [];\n const processorHostModule = await createProcessorHostModule(\n reactorClientModule.client,\n readModels,\n window.ph?.attachmentService,\n );\n if (processorHostModule !== undefined) {\n await Promise.all(\n packagesWithProcessorFactories.map(async (pkg) => {\n const { manifest, processorFactory } = pkg;\n const name = manifest.name;\n const id = manifest.name;\n const version = packageManager.getPackageVersion(name);\n const label = version ? `${name}@${version}` : name;\n logger.info(\"Loading processor factory: @label\", label);\n try {\n const factory = await processorFactory?.(processorHostModule);\n if (!factory) return;\n await reactorClientModule.reactorModule?.processorManager.registerFactory(\n id,\n factory,\n );\n } catch (error) {\n logger.error(`Error registering processor: @label`, label);\n logger.error(\"@error\", error);\n }\n }),\n );\n }\n }\n\n window.ph.loading = false;\n}\n\nfunction getDidFromUrl() {\n const searchParams = new URLSearchParams(window.location.search);\n const didComponent = searchParams.get(\"user\");\n const did = didComponent ? decodeURIComponent(didComponent) : undefined;\n return did;\n}\n\nfunction getDriveUrl() {\n const searchParams = new URLSearchParams(window.location.search);\n const driveUrl = searchParams.get(\"driveUrl\");\n const url = driveUrl ? decodeURIComponent(driveUrl) : undefined;\n return url;\n}\n"],"names":["#packageManager","#findModuleInLoadedPackages","#clearFailed","#recordFailure","#failedMemo","#failedSubscribers","#eventSubscribers","#failed","#emitEvent","#notifyFailedChanged","#packageManager","#registryClient","#mode","#dismissedStorage","#findModuleInLoadedPackages","#clearFailed","#recordFailure","#deferred","#discover","#discoveredTypes","#pending","#addToPending","#findTypesByPackage","#emitEvent","#notifyPendingChanged","#pendingMemo","#pendingSubscribers","#failedMemo","#failedSubscribers","#eventSubscribers","#createDeferredEntry","#failed","#notifyFailedChanged","#storage","#cdnUrl","#toCdnUrl","#localPackageNames","#registerPackage","#localPackage","#localPackageVersion","#notifyPackagesChanged","#packagesMemo","#packages","#loadPackage","#unmountStylesheet","#subscribers","#loadPackageFromNodeModules","#importPackage","#loadPackageFromRegistry","#mountStylesheet","#stylesheets"],"mappings":"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","debug_id":"d17abc36-680f-59c7-9a6f-834f46f065ea"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
|
|
2
|
-
!function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="
|
|
3
|
-
import { r as connectConfig } from "./connect.config-
|
|
2
|
+
!function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="a37b97c0-56d1-535a-84fc-40aac3c069a4")}catch(e){}}();
|
|
3
|
+
import { r as connectConfig } from "./connect.config-Bt4pvvWV.js";
|
|
4
4
|
//#region src/utils/registerServiceWorker.ts
|
|
5
5
|
const serviceWorkerScriptPath = [connectConfig.routerBasename, "service-worker.js"].join("/").replace(/\/{2,}/gm, "/");
|
|
6
6
|
var ServiceWorkerManager = class {
|
|
@@ -77,5 +77,5 @@ const serviceWorkerManager = new ServiceWorkerManager();
|
|
|
77
77
|
//#endregion
|
|
78
78
|
export { serviceWorkerManager as t };
|
|
79
79
|
|
|
80
|
-
//# sourceMappingURL=registerServiceWorker-
|
|
81
|
-
//# debugId=
|
|
80
|
+
//# sourceMappingURL=registerServiceWorker-D9l1kja7.js.map
|
|
81
|
+
//# debugId=a37b97c0-56d1-535a-84fc-40aac3c069a4
|
package/dist/{registerServiceWorker-B7uKpkci.js.map → registerServiceWorker-D9l1kja7.js.map}
RENAMED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"registerServiceWorker-
|
|
1
|
+
{"version":3,"file":"registerServiceWorker-D9l1kja7.js","sources":["../src/utils/registerServiceWorker.ts"],"sourcesContent":["import { connectConfig } from \"@powerhousedao/connect/config\";\n\nconst basePath = connectConfig.routerBasename;\n\nconst serviceWorkerScriptPath = [basePath, \"service-worker.js\"]\n .join(\"/\")\n .replace(/\\/{2,}/gm, \"/\");\n\nclass ServiceWorkerManager {\n ready = false;\n debug = false;\n registration: ServiceWorkerRegistration | null = null;\n\n constructor(debug = false) {\n this.debug = debug;\n }\n\n setDebug(debug: boolean) {\n this.debug = debug;\n }\n\n #handleServiceWorkerMessage(\n event: MessageEvent<{\n type: \"NEW_VERSION_AVAILABLE\";\n version: string;\n requiresHardRefresh: boolean;\n }>,\n ) {\n if (this.debug) {\n console.log(\"ServiceWorker message: \", event);\n }\n const message = \"type\" in event.data ? event : null;\n switch (message?.data.type) {\n case \"NEW_VERSION_AVAILABLE\": {\n if (message.data.version === connectConfig.appVersion) {\n return;\n }\n if (message.data.requiresHardRefresh) {\n if (this.debug) {\n console.log(\"New version available\");\n }\n\n window.location.reload(); // Reload the page to load the new version\n }\n break;\n }\n default: {\n console.warn(\"Unhandled message:\", message);\n break;\n }\n }\n }\n\n #handleServiceWorker(registration: ServiceWorkerRegistration) {\n {\n // Listen for messages from the service worker\n if (this.debug) {\n console.log(\"ServiceWorker registered: \", registration);\n }\n\n navigator.serviceWorker.addEventListener(\n \"message\",\n this.#handleServiceWorkerMessage.bind(this),\n );\n\n this.registration = registration;\n this.ready = true;\n }\n }\n\n async #register() {\n try {\n // checks if there is a service worker installed already and calls\n // its the update method to check if there is a new version available\n const existingRegistration =\n await navigator.serviceWorker.getRegistration();\n if (existingRegistration) {\n await existingRegistration.update();\n this.#handleServiceWorker(existingRegistration);\n }\n\n // if no service worker is installed then registers the service worker\n else {\n const registration = await navigator.serviceWorker.register(\n serviceWorkerScriptPath,\n );\n this.#handleServiceWorker(registration);\n\n registration.addEventListener(\"updatefound\", () => {\n this.#handleServiceWorker(registration);\n });\n }\n\n // calls the update on an interval to force\n // the browser to check for a new version\n const intervalId = setInterval(() => {\n void (async () => {\n const existingRegistration =\n await navigator.serviceWorker.getRegistration();\n if (existingRegistration) {\n await existingRegistration.update();\n } else {\n clearInterval(intervalId);\n this.registerServiceWorker();\n }\n })();\n }, connectConfig.appVersionCheckInterval);\n } catch (error) {\n console.error(\"ServiceWorker registration failed: \", error);\n }\n }\n\n registerServiceWorker(debug = false) {\n this.debug = debug;\n\n if (!(\"serviceWorker\" in navigator)) {\n console.warn(\"Service Worker not available\");\n return;\n }\n window.addEventListener(\"load\", () => {\n this.#register().catch(console.error);\n });\n }\n}\n\nexport const serviceWorkerManager = new ServiceWorkerManager();\n"],"names":["#handleServiceWorkerMessage","#register","#handleServiceWorker"],"mappings":";;;;AAIA,MAAM,0BAA0B,CAFf,cAAc,gBAEY,oBAAoB,CAC5D,KAAK,IAAI,CACT,QAAQ,YAAY,IAAI;AAE3B,IAAM,uBAAN,MAA2B;CACzB,QAAQ;CACR,QAAQ;CACR,eAAiD;CAEjD,YAAY,QAAQ,OAAO;AACzB,OAAK,QAAQ;;CAGf,SAAS,OAAgB;AACvB,OAAK,QAAQ;;CAGf,4BACE,OAKA;AACA,MAAI,KAAK,MACP,SAAQ,IAAI,2BAA2B,MAAM;EAE/C,MAAM,UAAU,UAAU,MAAM,OAAO,QAAQ;AAC/C,UAAQ,SAAS,KAAK,MAAtB;GACE,KAAK;AACH,QAAI,QAAQ,KAAK,YAAY,cAAc,WACzC;AAEF,QAAI,QAAQ,KAAK,qBAAqB;AACpC,SAAI,KAAK,MACP,SAAQ,IAAI,wBAAwB;AAGtC,YAAO,SAAS,QAAQ;;AAE1B;GAEF;AACE,YAAQ,KAAK,sBAAsB,QAAQ;AAC3C;;;CAKN,qBAAqB,cAAyC;AAG1D,MAAI,KAAK,MACP,SAAQ,IAAI,8BAA8B,aAAa;AAGzD,YAAU,cAAc,iBACtB,WACA,MAAA,2BAAiC,KAAK,KAAK,CAC5C;AAED,OAAK,eAAe;AACpB,OAAK,QAAQ;;CAIjB,OAAA,WAAkB;AAChB,MAAI;GAGF,MAAM,uBACJ,MAAM,UAAU,cAAc,iBAAiB;AACjD,OAAI,sBAAsB;AACxB,UAAM,qBAAqB,QAAQ;AACnC,UAAA,oBAA0B,qBAAqB;UAI5C;IACH,MAAM,eAAe,MAAM,UAAU,cAAc,SACjD,wBACD;AACD,UAAA,oBAA0B,aAAa;AAEvC,iBAAa,iBAAiB,qBAAqB;AACjD,WAAA,oBAA0B,aAAa;MACvC;;GAKJ,MAAM,aAAa,kBAAkB;AACnC,KAAM,YAAY;KAChB,MAAM,uBACJ,MAAM,UAAU,cAAc,iBAAiB;AACjD,SAAI,qBACF,OAAM,qBAAqB,QAAQ;UAC9B;AACL,oBAAc,WAAW;AACzB,WAAK,uBAAuB;;QAE5B;MACH,cAAc,wBAAwB;WAClC,OAAO;AACd,WAAQ,MAAM,uCAAuC,MAAM;;;CAI/D,sBAAsB,QAAQ,OAAO;AACnC,OAAK,QAAQ;AAEb,MAAI,EAAE,mBAAmB,YAAY;AACnC,WAAQ,KAAK,+BAA+B;AAC5C;;AAEF,SAAO,iBAAiB,cAAc;AACpC,SAAA,UAAgB,CAAC,MAAM,QAAQ,MAAM;IACrC;;;AAIN,MAAa,uBAAuB,IAAI,sBAAsB","debug_id":"a37b97c0-56d1-535a-84fc-40aac3c069a4"}
|
package/dist/start-connect.js
CHANGED
|
@@ -5,7 +5,7 @@ import { createRoot } from "react-dom/client";
|
|
|
5
5
|
import { jsx } from "react/jsx-runtime";
|
|
6
6
|
//#region start-connect.tsx
|
|
7
7
|
await loadRuntimeConfig();
|
|
8
|
-
const { AppLoader } = await import("./components-
|
|
8
|
+
const { AppLoader } = await import("./components-7jnuvqBG.js");
|
|
9
9
|
function updateLocalPackage(pkg) {
|
|
10
10
|
window.ph?.vetraPackageManager?.updateLocalPackage(pkg);
|
|
11
11
|
}
|
package/dist/style.css
CHANGED
|
@@ -292,6 +292,9 @@
|
|
|
292
292
|
}
|
|
293
293
|
}
|
|
294
294
|
@layer utilities {
|
|
295
|
+
.pointer-events-auto {
|
|
296
|
+
pointer-events: auto;
|
|
297
|
+
}
|
|
295
298
|
.pointer-events-auto\! {
|
|
296
299
|
pointer-events: auto !important;
|
|
297
300
|
}
|
|
@@ -567,6 +570,9 @@
|
|
|
567
570
|
.mt-\[11px\] {
|
|
568
571
|
margin-top: 11px;
|
|
569
572
|
}
|
|
573
|
+
.mt-auto {
|
|
574
|
+
margin-top: auto;
|
|
575
|
+
}
|
|
570
576
|
.-mr-1 {
|
|
571
577
|
margin-right: calc(var(--spacing) * -1);
|
|
572
578
|
}
|
|
@@ -1038,6 +1044,9 @@
|
|
|
1038
1044
|
.w-\[26px\] {
|
|
1039
1045
|
width: 26px;
|
|
1040
1046
|
}
|
|
1047
|
+
.w-\[28rem\] {
|
|
1048
|
+
width: 28rem;
|
|
1049
|
+
}
|
|
1041
1050
|
.w-\[50px\] {
|
|
1042
1051
|
width: 50px;
|
|
1043
1052
|
}
|
|
@@ -1116,6 +1125,9 @@
|
|
|
1116
1125
|
.max-w-\[1024px\] {
|
|
1117
1126
|
max-width: 1024px;
|
|
1118
1127
|
}
|
|
1128
|
+
.max-w-\[calc\(100\%-2rem\)\] {
|
|
1129
|
+
max-width: calc(100% - 2rem);
|
|
1130
|
+
}
|
|
1119
1131
|
.max-w-fit {
|
|
1120
1132
|
max-width: fit-content;
|
|
1121
1133
|
}
|
|
@@ -1206,6 +1218,10 @@
|
|
|
1206
1218
|
--tw-translate-y: calc(calc(1 / 2 * 100%) * -1);
|
|
1207
1219
|
translate: var(--tw-translate-x) var(--tw-translate-y);
|
|
1208
1220
|
}
|
|
1221
|
+
.-translate-y-\[3px\] {
|
|
1222
|
+
--tw-translate-y: calc(3px * -1);
|
|
1223
|
+
translate: var(--tw-translate-x) var(--tw-translate-y);
|
|
1224
|
+
}
|
|
1209
1225
|
.-scale-x-100 {
|
|
1210
1226
|
--tw-scale-x: calc(100% * -1);
|
|
1211
1227
|
scale: var(--tw-scale-x) var(--tw-scale-y);
|
|
@@ -1378,6 +1394,9 @@
|
|
|
1378
1394
|
margin-block-end: calc(calc(var(--spacing) * 6) * calc(1 - var(--tw-space-y-reverse)));
|
|
1379
1395
|
}
|
|
1380
1396
|
}
|
|
1397
|
+
.gap-x-0\.5 {
|
|
1398
|
+
column-gap: calc(var(--spacing) * 0.5);
|
|
1399
|
+
}
|
|
1381
1400
|
.gap-x-1 {
|
|
1382
1401
|
column-gap: calc(var(--spacing) * 1);
|
|
1383
1402
|
}
|
|
@@ -1706,6 +1725,12 @@
|
|
|
1706
1725
|
.bg-gray-900 {
|
|
1707
1726
|
background-color: var(--color-gray-900);
|
|
1708
1727
|
}
|
|
1728
|
+
.bg-gray-900\/30 {
|
|
1729
|
+
background-color: color-mix(in srgb, hsl(192 5% 21%) 30%, transparent);
|
|
1730
|
+
@supports (color: color-mix(in lab, red, red)) {
|
|
1731
|
+
background-color: color-mix(in oklab, var(--color-gray-900) 30%, transparent);
|
|
1732
|
+
}
|
|
1733
|
+
}
|
|
1709
1734
|
.bg-gray-900\/50 {
|
|
1710
1735
|
background-color: color-mix(in srgb, hsl(192 5% 21%) 50%, transparent);
|
|
1711
1736
|
@supports (color: color-mix(in lab, red, red)) {
|
|
@@ -1778,6 +1803,9 @@
|
|
|
1778
1803
|
.bg-violet-400 {
|
|
1779
1804
|
background-color: var(--color-violet-400);
|
|
1780
1805
|
}
|
|
1806
|
+
.bg-white {
|
|
1807
|
+
background-color: var(--color-white);
|
|
1808
|
+
}
|
|
1781
1809
|
.bg-white\/90 {
|
|
1782
1810
|
background-color: color-mix(in srgb, hsl(0 0% 100%) 90%, transparent);
|
|
1783
1811
|
@supports (color: color-mix(in lab, red, red)) {
|
|
@@ -1883,6 +1911,9 @@
|
|
|
1883
1911
|
.py-2 {
|
|
1884
1912
|
padding-block: calc(var(--spacing) * 2);
|
|
1885
1913
|
}
|
|
1914
|
+
.py-2\.5 {
|
|
1915
|
+
padding-block: calc(var(--spacing) * 2.5);
|
|
1916
|
+
}
|
|
1886
1917
|
.py-3 {
|
|
1887
1918
|
padding-block: calc(var(--spacing) * 3);
|
|
1888
1919
|
}
|
|
@@ -2213,6 +2244,9 @@
|
|
|
2213
2244
|
.text-gray-500 {
|
|
2214
2245
|
color: var(--color-gray-500);
|
|
2215
2246
|
}
|
|
2247
|
+
.text-gray-600 {
|
|
2248
|
+
color: var(--color-gray-600);
|
|
2249
|
+
}
|
|
2216
2250
|
.text-gray-700 {
|
|
2217
2251
|
color: var(--color-gray-700);
|
|
2218
2252
|
}
|
|
@@ -2341,6 +2375,10 @@
|
|
|
2341
2375
|
--tw-shadow: 0 1px 3px 0 var(--tw-shadow-color, rgb(0 0 0 / 0.1)), 0 1px 2px -1px var(--tw-shadow-color, rgb(0 0 0 / 0.1));
|
|
2342
2376
|
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
2343
2377
|
}
|
|
2378
|
+
.shadow-2xl {
|
|
2379
|
+
--tw-shadow: 0 25px 50px -12px var(--tw-shadow-color, rgb(0 0 0 / 0.25));
|
|
2380
|
+
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
2381
|
+
}
|
|
2344
2382
|
.shadow-button {
|
|
2345
2383
|
--tw-shadow: 0px -1px 1px 0px var(--tw-shadow-color, rgba(0, 0, 0, 0.04)) inset, 0px 2px 0px 0px var(--tw-shadow-color, rgba(255, 255, 255, 0.25)) inset, 0px 4px 8px -4px var(--tw-shadow-color, rgba(0, 0, 0, 0.1));
|
|
2346
2384
|
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
@@ -2373,12 +2411,22 @@
|
|
|
2373
2411
|
--tw-shadow: 0px 2px 8px 0px var(--tw-shadow-color, rgba(0, 0, 0, 0.05)), 0px 44px 48px -12px var(--tw-shadow-color, rgba(0, 0, 0, 0.15)), 0px 0px 24px 4px var(--tw-shadow-color, rgba(0, 0, 0, 0.05));
|
|
2374
2412
|
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
2375
2413
|
}
|
|
2414
|
+
.ring-1 {
|
|
2415
|
+
--tw-ring-shadow: var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color, currentcolor);
|
|
2416
|
+
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
2417
|
+
}
|
|
2376
2418
|
.shadow-gray-900\/4 {
|
|
2377
2419
|
--tw-shadow-color: color-mix(in srgb, hsl(192 5% 21%) 4%, transparent);
|
|
2378
2420
|
@supports (color: color-mix(in lab, red, red)) {
|
|
2379
2421
|
--tw-shadow-color: color-mix(in oklab, color-mix(in oklab, var(--color-gray-900) 4%, transparent) var(--tw-shadow-alpha), transparent);
|
|
2380
2422
|
}
|
|
2381
2423
|
}
|
|
2424
|
+
.ring-black\/5 {
|
|
2425
|
+
--tw-ring-color: color-mix(in srgb, hsl(0 0% 0%) 5%, transparent);
|
|
2426
|
+
@supports (color: color-mix(in lab, red, red)) {
|
|
2427
|
+
--tw-ring-color: color-mix(in oklab, var(--color-black) 5%, transparent);
|
|
2428
|
+
}
|
|
2429
|
+
}
|
|
2382
2430
|
.outline {
|
|
2383
2431
|
outline-style: var(--tw-outline-style);
|
|
2384
2432
|
outline-width: 1px;
|
|
@@ -3371,6 +3419,11 @@
|
|
|
3371
3419
|
box-shadow: var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow);
|
|
3372
3420
|
}
|
|
3373
3421
|
}
|
|
3422
|
+
.focus-visible\:ring-gray-400 {
|
|
3423
|
+
&:focus-visible {
|
|
3424
|
+
--tw-ring-color: var(--color-gray-400);
|
|
3425
|
+
}
|
|
3426
|
+
}
|
|
3374
3427
|
.focus-visible\:ring-gray-900 {
|
|
3375
3428
|
&:focus-visible {
|
|
3376
3429
|
--tw-ring-color: var(--color-gray-900);
|
|
@@ -3830,6 +3883,11 @@
|
|
|
3830
3883
|
border-color: var(--color-slate-600);
|
|
3831
3884
|
}
|
|
3832
3885
|
}
|
|
3886
|
+
.dark\:border-slate-700 {
|
|
3887
|
+
&:where(.dark, .dark *) {
|
|
3888
|
+
border-color: var(--color-slate-700);
|
|
3889
|
+
}
|
|
3890
|
+
}
|
|
3833
3891
|
.dark\:border-slate-800 {
|
|
3834
3892
|
&:where(.dark, .dark *) {
|
|
3835
3893
|
border-color: var(--color-slate-800);
|
|
@@ -4029,6 +4087,14 @@
|
|
|
4029
4087
|
}
|
|
4030
4088
|
}
|
|
4031
4089
|
}
|
|
4090
|
+
.dark\:bg-slate-900\/70 {
|
|
4091
|
+
&:where(.dark, .dark *) {
|
|
4092
|
+
background-color: color-mix(in srgb, hsl(195 6% 6%) 70%, transparent);
|
|
4093
|
+
@supports (color: color-mix(in lab, red, red)) {
|
|
4094
|
+
background-color: color-mix(in oklab, var(--color-slate-900) 70%, transparent);
|
|
4095
|
+
}
|
|
4096
|
+
}
|
|
4097
|
+
}
|
|
4032
4098
|
.dark\:bg-slate-900\/90 {
|
|
4033
4099
|
&:where(.dark, .dark *) {
|
|
4034
4100
|
background-color: color-mix(in srgb, hsl(195 6% 6%) 90%, transparent);
|
|
@@ -4162,6 +4228,14 @@
|
|
|
4162
4228
|
color: var(--color-yellow-100);
|
|
4163
4229
|
}
|
|
4164
4230
|
}
|
|
4231
|
+
.dark\:ring-white\/10 {
|
|
4232
|
+
&:where(.dark, .dark *) {
|
|
4233
|
+
--tw-ring-color: color-mix(in srgb, hsl(0 0% 100%) 10%, transparent);
|
|
4234
|
+
@supports (color: color-mix(in lab, red, red)) {
|
|
4235
|
+
--tw-ring-color: color-mix(in oklab, var(--color-white) 10%, transparent);
|
|
4236
|
+
}
|
|
4237
|
+
}
|
|
4238
|
+
}
|
|
4165
4239
|
.dark\:group-focus-within\:text-slate-50 {
|
|
4166
4240
|
&:where(.dark, .dark *) {
|
|
4167
4241
|
&:is(:where(.group):focus-within *) {
|
|
@@ -1,15 +1,15 @@
|
|
|
1
1
|
|
|
2
|
-
!function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="
|
|
2
|
+
!function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="ed326e42-f833-53ca-9b66-6d0b7e627b32")}catch(e){}}();
|
|
3
3
|
import { n as getRuntimeConfig } from "./runtime-config-DL3QjcfM.js";
|
|
4
|
-
import { useEffect, useMemo } from "react";
|
|
4
|
+
import { useEffect, useMemo, useState } from "react";
|
|
5
5
|
import { getPackages, trimTrailingSlash, useVetraPackageManager } from "@powerhousedao/reactor-browser";
|
|
6
|
-
import {
|
|
6
|
+
import { slimManifest } from "@powerhousedao/shared/registry/manifest-slim";
|
|
7
7
|
//#region src/hooks/useRegistryPackages.ts
|
|
8
8
|
function useRegistryPackages() {
|
|
9
9
|
const packageManager = useVetraPackageManager();
|
|
10
10
|
const packageManagerPackages = packageManager?.packages;
|
|
11
11
|
const registryUrl = getRuntimeConfig().packageRegistryUrl ?? null;
|
|
12
|
-
const [registryPackagesMap, setRegistryPackagesMap] =
|
|
12
|
+
const [registryPackagesMap, setRegistryPackagesMap] = useBestEffortLocalStorage(`REGISTRY_PACKAGES:${registryUrl === null ? null : trimTrailingSlash(registryUrl)}`, {});
|
|
13
13
|
const registryPackageList = useMemo(() => {
|
|
14
14
|
return Array.from(Object.values(registryPackagesMap)).filter((p) => p !== void 0);
|
|
15
15
|
}, [registryPackagesMap]);
|
|
@@ -28,7 +28,7 @@ function useRegistryPackages() {
|
|
|
28
28
|
const installedVersion = packageManager.getPackageVersion(packageInfo.name);
|
|
29
29
|
newRegistryPackages[packageInfo.name] = {
|
|
30
30
|
...existingPackage,
|
|
31
|
-
manifest: packageInfo.manifest ?? existingPackage.manifest,
|
|
31
|
+
manifest: slimManifest(packageInfo.manifest) ?? existingPackage.manifest,
|
|
32
32
|
version: installedVersion ?? packageInfo.version ?? existingPackage.version,
|
|
33
33
|
distTags: packageInfo.distTags ?? existingPackage.distTags,
|
|
34
34
|
versions: packageInfo.versions ?? existingPackage.versions,
|
|
@@ -107,18 +107,54 @@ function makeRegistryPackageFromDocumentModelLib(documentModelLib, status, versi
|
|
|
107
107
|
path: "stub-path",
|
|
108
108
|
documentTypes: documentModelLib.documentModels.map((d) => d.documentModel.global.id),
|
|
109
109
|
status,
|
|
110
|
-
manifest: documentModelLib.manifest,
|
|
110
|
+
manifest: slimManifest(documentModelLib.manifest),
|
|
111
111
|
version
|
|
112
112
|
};
|
|
113
113
|
}
|
|
114
114
|
function makeRegistryPackageFromPackageInfo(packageInfo, status) {
|
|
115
115
|
return {
|
|
116
116
|
...packageInfo,
|
|
117
|
+
manifest: slimManifest(packageInfo.manifest),
|
|
117
118
|
documentTypes: packageInfo.manifest?.documentModels?.map((d) => d.id) ?? [],
|
|
118
119
|
status
|
|
119
120
|
};
|
|
120
121
|
}
|
|
121
122
|
/**
|
|
123
|
+
* Like usehooks-ts' useLocalStorage, but persistence is best-effort: a failed
|
|
124
|
+
* `setItem` (e.g. QuotaExceededError on an oversized payload) keeps the
|
|
125
|
+
* in-memory state so the package manager still works for the session.
|
|
126
|
+
* useLocalStorage couples the state update to the write — when the write
|
|
127
|
+
* threw, the freshly fetched package list was discarded and the registry UI
|
|
128
|
+
* rendered (and searched) an empty list.
|
|
129
|
+
*
|
|
130
|
+
* The key is derived from the boot-time runtime config and never changes
|
|
131
|
+
* within a session, so no key-change rehydration is needed.
|
|
132
|
+
*/
|
|
133
|
+
function useBestEffortLocalStorage(key, initialValue) {
|
|
134
|
+
const [value, setValue] = useState(() => readJsonFromStorage(key) ?? initialValue);
|
|
135
|
+
useEffect(() => {
|
|
136
|
+
try {
|
|
137
|
+
window.localStorage.setItem(key, JSON.stringify(value));
|
|
138
|
+
} catch (error) {
|
|
139
|
+
try {
|
|
140
|
+
window.localStorage.removeItem(key);
|
|
141
|
+
window.localStorage.setItem(key, JSON.stringify(value));
|
|
142
|
+
} catch {
|
|
143
|
+
console.warn(`Failed to persist "${key}" to localStorage; continuing with in-memory data.`, error);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}, [key, value]);
|
|
147
|
+
return [value, setValue];
|
|
148
|
+
}
|
|
149
|
+
function readJsonFromStorage(key) {
|
|
150
|
+
try {
|
|
151
|
+
const raw = window.localStorage.getItem(key);
|
|
152
|
+
return raw === null ? null : JSON.parse(raw);
|
|
153
|
+
} catch {
|
|
154
|
+
return null;
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
122
158
|
* When refreshing a cached entry, promote `"available"` to whatever the
|
|
123
159
|
* packageManager now reports — the cached `"available"` was almost certainly
|
|
124
160
|
* recorded before `packageManager.addPackages()` finished on a prior session.
|
|
@@ -139,5 +175,5 @@ function getPackageStatusFromPackageSource(packageSource) {
|
|
|
139
175
|
//#endregion
|
|
140
176
|
export { useRegistryPackages as t };
|
|
141
177
|
|
|
142
|
-
//# sourceMappingURL=useRegistryPackages-
|
|
143
|
-
//# debugId=
|
|
178
|
+
//# sourceMappingURL=useRegistryPackages-B-gFECmd.js.map
|
|
179
|
+
//# debugId=ed326e42-f833-53ca-9b66-6d0b7e627b32
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"useRegistryPackages-B-gFECmd.js","sources":["../src/hooks/useRegistryPackages.ts"],"sourcesContent":["import {\n getPackages,\n trimTrailingSlash,\n useVetraPackageManager,\n} from \"@powerhousedao/reactor-browser\";\nimport type {\n PackageInfo,\n RegistryPackage,\n RegistryPackageList,\n RegistryPackageMap,\n RegistryPackageSource,\n RegistryPackageStatus,\n} from \"@powerhousedao/shared/registry\";\nimport { slimManifest } from \"@powerhousedao/shared/registry/manifest-slim\";\nimport type { DocumentModelLib } from \"document-model\";\nimport { useEffect, useMemo, useState } from \"react\";\nimport { getRuntimeConfig } from \"../runtime-config.js\";\n\nexport function useRegistryPackages() {\n const packageManager = useVetraPackageManager();\n const packageManagerPackages = packageManager?.packages;\n const registryUrl = getRuntimeConfig().packageRegistryUrl ?? null;\n // Normalize so `http://host` and `http://host/` don't produce two separate\n // localStorage maps where the install/status flow reads from one while the\n // registry fetch writes to the other.\n const registryPackagesKey = `REGISTRY_PACKAGES:${registryUrl === null ? null : trimTrailingSlash(registryUrl)}`;\n const [registryPackagesMap, setRegistryPackagesMap] =\n useBestEffortLocalStorage<RegistryPackageMap>(registryPackagesKey, {});\n const registryPackageList: RegistryPackageList = useMemo(() => {\n return Array.from(Object.values(registryPackagesMap)).filter(\n (p) => p !== undefined,\n );\n }, [registryPackagesMap]);\n\n useEffect(() => {\n async function refreshPackages() {\n if (registryUrl === null || !packageManager) return;\n\n const packageInfos = await getPackages(registryUrl);\n\n setRegistryPackagesMap((oldPackages) => {\n const newRegistryPackages: RegistryPackageMap = {\n ...oldPackages,\n };\n\n for (const packageInfo of packageInfos) {\n const existingPackage = newRegistryPackages[packageInfo.name];\n\n if (!existingPackage) {\n const packageSource = packageManager.getPackageSource(\n packageInfo.name,\n );\n const status = getPackageStatusFromPackageSource(packageSource);\n newRegistryPackages[packageInfo.name] =\n makeRegistryPackageFromPackageInfo(packageInfo, status);\n } else {\n // Keep the cached entry's status, but refresh anything the\n // registry sent. This includes distTags and the versions list\n // the Package Manager UI filters on.\n //\n // For `version`, prefer the actually-installed version from the\n // package manager (set by the rehydration effect). `packageInfo.version`\n // is the registry's newest-published version, which for installed\n // packages would incorrectly overwrite the user's picked version\n // on the next /packages refresh.\n const installedVersion = packageManager.getPackageVersion(\n packageInfo.name,\n );\n newRegistryPackages[packageInfo.name] = {\n ...existingPackage,\n manifest:\n slimManifest(packageInfo.manifest) ?? existingPackage.manifest,\n version:\n installedVersion ??\n packageInfo.version ??\n existingPackage.version,\n distTags: packageInfo.distTags ?? existingPackage.distTags,\n versions: packageInfo.versions ?? existingPackage.versions,\n documentTypes: packageInfo.documentTypes.length\n ? packageInfo.documentTypes\n : existingPackage.documentTypes,\n status: promoteStatus(\n existingPackage.status,\n packageManager.getPackageSource(packageInfo.name),\n ),\n };\n }\n }\n\n return newRegistryPackages;\n });\n }\n\n refreshPackages().catch(console.error);\n }, []);\n\n useEffect(() => {\n if (!packageManager) return;\n\n if (packageManagerPackages?.length) {\n for (const packageManagerPackage of packageManagerPackages) {\n setRegistryPackagesMap((existingRegistryPackages) => {\n const packageName = packageManagerPackage.manifest.name;\n const existingPackage = existingRegistryPackages[packageName];\n const newRegistryPackages = { ...existingRegistryPackages };\n const version = packageManager.getPackageVersion(packageName);\n if (existingPackage) {\n newRegistryPackages[packageName] = {\n ...existingPackage,\n version: version ?? existingPackage.version,\n status: promoteStatus(\n existingPackage.status,\n packageManager.getPackageSource(packageName),\n ),\n };\n } else {\n const packageSource = packageManager.getPackageSource(packageName);\n const status = getPackageStatusFromPackageSource(packageSource);\n const newRegistryPackage = makeRegistryPackageFromDocumentModelLib(\n packageManagerPackage,\n status,\n version,\n );\n newRegistryPackages[packageName] = newRegistryPackage;\n }\n return newRegistryPackages;\n });\n }\n }\n }, [packageManagerPackages]);\n\n function updateRegistryPackageStatus(\n packageName: string,\n newStatus: RegistryPackageStatus,\n ) {\n setRegistryPackagesMap((oldRegistryPackages) => {\n const newRegistryPackages = { ...oldRegistryPackages };\n const newRegistryPackage = newRegistryPackages[packageName];\n if (!newRegistryPackage) {\n console.error(\n \"Attempting to update status for package that does not exist.\",\n );\n return newRegistryPackages;\n }\n newRegistryPackages[packageName] = {\n ...newRegistryPackage,\n status: newStatus,\n };\n\n return newRegistryPackages;\n });\n }\n\n /**\n * Register a freshly-installed package that came in via the npm-uplink\n * fallback — the user typed a bare name, our local `/packages` didn't know\n * it, but the install succeeded because verdaccio proxy-fetched the tarball.\n *\n * This is the one legitimate case where the status update runs against a\n * name that wasn't in the registry map. We treat it as an insert rather\n * than logging the \"does not exist\" error. Data is pulled from the loaded\n * module so the UI card shows the real manifest immediately; the next\n * `/packages` refresh will add `versions`/`distTags`.\n */\n function registerFallbackRegistryPackage(\n packageName: string,\n loadedPackage: DocumentModelLib,\n version: string | undefined,\n status: RegistryPackageStatus,\n ) {\n setRegistryPackagesMap((oldRegistryPackages) => {\n const newRegistryPackages = { ...oldRegistryPackages };\n newRegistryPackages[packageName] = {\n ...makeRegistryPackageFromDocumentModelLib(\n loadedPackage,\n status,\n version,\n ),\n name: packageName,\n };\n return newRegistryPackages;\n });\n }\n\n return {\n registryPackagesMap,\n registryPackageList,\n updateRegistryPackageStatus,\n registerFallbackRegistryPackage,\n };\n}\n\nfunction makeRegistryPackageFromDocumentModelLib(\n documentModelLib: DocumentModelLib,\n status: RegistryPackageStatus,\n version?: string,\n): RegistryPackage {\n return {\n name: documentModelLib.manifest.name,\n path: \"stub-path\",\n documentTypes: documentModelLib.documentModels.map(\n (d) => d.documentModel.global.id,\n ),\n status,\n manifest: slimManifest(documentModelLib.manifest),\n version,\n };\n}\n\nfunction makeRegistryPackageFromPackageInfo(\n packageInfo: PackageInfo,\n status: RegistryPackageStatus,\n): RegistryPackage {\n return {\n ...packageInfo,\n // Slim before caching: registry manifests are unvalidated JSON and have\n // carried multi-megabyte junk fields that blew the localStorage quota\n // (and the UI only reads the summary fields anyway).\n manifest: slimManifest(packageInfo.manifest),\n documentTypes: packageInfo.manifest?.documentModels?.map((d) => d.id) ?? [],\n status,\n };\n}\n\n/**\n * Like usehooks-ts' useLocalStorage, but persistence is best-effort: a failed\n * `setItem` (e.g. QuotaExceededError on an oversized payload) keeps the\n * in-memory state so the package manager still works for the session.\n * useLocalStorage couples the state update to the write — when the write\n * threw, the freshly fetched package list was discarded and the registry UI\n * rendered (and searched) an empty list.\n *\n * The key is derived from the boot-time runtime config and never changes\n * within a session, so no key-change rehydration is needed.\n */\nfunction useBestEffortLocalStorage<T>(key: string, initialValue: T) {\n const [value, setValue] = useState<T>(\n () => readJsonFromStorage<T>(key) ?? initialValue,\n );\n\n useEffect(() => {\n try {\n window.localStorage.setItem(key, JSON.stringify(value));\n } catch (error) {\n // Free the key and retry once — a stale oversized entry from an older\n // session may be occupying the quota this write needs.\n try {\n window.localStorage.removeItem(key);\n window.localStorage.setItem(key, JSON.stringify(value));\n } catch {\n console.warn(\n `Failed to persist \"${key}\" to localStorage; continuing with in-memory data.`,\n error,\n );\n }\n }\n }, [key, value]);\n\n return [value, setValue] as const;\n}\n\nfunction readJsonFromStorage<T>(key: string): T | null {\n try {\n const raw = window.localStorage.getItem(key);\n return raw === null ? null : (JSON.parse(raw) as T);\n } catch {\n return null;\n }\n}\n\n/**\n * When refreshing a cached entry, promote `\"available\"` to whatever the\n * packageManager now reports — the cached `\"available\"` was almost certainly\n * recorded before `packageManager.addPackages()` finished on a prior session.\n * Never downgrade an already-installed entry, and never overwrite a deliberate\n * `\"dismissed\"` choice.\n */\nfunction promoteStatus(\n cachedStatus: RegistryPackageStatus,\n packageSource: RegistryPackageSource | null,\n): RegistryPackageStatus {\n if (cachedStatus !== \"available\") return cachedStatus;\n if (packageSource === null) return cachedStatus;\n return getPackageStatusFromPackageSource(packageSource);\n}\n\nfunction getPackageStatusFromPackageSource(\n packageSource: RegistryPackageSource | null,\n): RegistryPackageStatus {\n // if we check the package source for a package that came from the api and it doesn't exist yet,\n // then we know the package is available on the api but not installed\n if (packageSource === null) return \"available\";\n // show common package, local project package and locally installed packages as \"local-install\"\n if (\n packageSource === \"local-install\" ||\n packageSource === \"common\" ||\n packageSource === \"project\"\n )\n return \"local-install\";\n // show \"registry-install\" status for package source \"registry-install\"\n if (packageSource === \"registry-install\") return \"registry-install\";\n // fallback to available — we should probably do more checks here\n return \"available\";\n}\n"],"names":[],"mappings":"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","debug_id":"ed326e42-f833-53ca-9b66-6d0b7e627b32"}
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@powerhousedao/connect",
|
|
3
3
|
"productName": "Powerhouse-Connect",
|
|
4
|
-
"version": "6.2.0-dev.
|
|
4
|
+
"version": "6.2.0-dev.20",
|
|
5
5
|
"description": "Powerhouse Connect",
|
|
6
6
|
"main": "dist/index.html",
|
|
7
7
|
"type": "module",
|
|
@@ -60,6 +60,7 @@
|
|
|
60
60
|
"i18next": "^25.5.3",
|
|
61
61
|
"kysely": "0.28.16",
|
|
62
62
|
"kysely-pglite-dialect": "1.2.0",
|
|
63
|
+
"lucide-react": "^1.16.0",
|
|
63
64
|
"pglite-legacy-02": "npm:@electric-sql/pglite@0.2.17",
|
|
64
65
|
"pglite-tools-legacy-02": "npm:@electric-sql/pglite-tools@0.2.4",
|
|
65
66
|
"react-error-boundary": "^4.0.11",
|
|
@@ -69,15 +70,15 @@
|
|
|
69
70
|
"tailwind-merge": "3.4.0",
|
|
70
71
|
"usehooks-ts": "^3.1.1",
|
|
71
72
|
"zod": "4.3.6",
|
|
72
|
-
"@powerhousedao/config": "6.2.0-dev.
|
|
73
|
-
"@powerhousedao/
|
|
74
|
-
"@powerhousedao/
|
|
75
|
-
"@powerhousedao/reactor-
|
|
76
|
-
"@powerhousedao/
|
|
77
|
-
"@powerhousedao/shared": "6.2.0-dev.
|
|
78
|
-
"@
|
|
79
|
-
"
|
|
80
|
-
"
|
|
73
|
+
"@powerhousedao/config": "6.2.0-dev.20",
|
|
74
|
+
"@powerhousedao/design-system": "6.2.0-dev.20",
|
|
75
|
+
"@powerhousedao/powerhouse-vetra-packages": "6.2.0-dev.20",
|
|
76
|
+
"@powerhousedao/reactor-attachments": "6.2.0-dev.20",
|
|
77
|
+
"@powerhousedao/reactor-browser": "6.2.0-dev.20",
|
|
78
|
+
"@powerhousedao/shared": "6.2.0-dev.20",
|
|
79
|
+
"@renown/sdk": "6.2.0-dev.20",
|
|
80
|
+
"document-model": "6.2.0-dev.20",
|
|
81
|
+
"@powerhousedao/vetra": "6.2.0-dev.20"
|
|
81
82
|
},
|
|
82
83
|
"devDependencies": {
|
|
83
84
|
"@tailwindcss/cli": "4.2.2",
|
|
@@ -96,7 +97,7 @@
|
|
|
96
97
|
"react-dom": "19.2.6",
|
|
97
98
|
"tailwindcss": "4.2.2",
|
|
98
99
|
"tsdown": "0.21.1",
|
|
99
|
-
"vite": "8.0.
|
|
100
|
+
"vite": "8.0.10",
|
|
100
101
|
"vite-bundle-analyzer": "^1.3.6",
|
|
101
102
|
"vite-tsconfig-paths": "6.1.1"
|
|
102
103
|
},
|