@happyvertical/smrt-features 0.30.0

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.
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sources":["../src/__smrt-register__.ts","../src/utils.ts","../src/feature-definition.ts","../src/feature-definitions.ts","../src/feature-override.ts","../src/feature-overrides.ts","../src/feature-resolver.ts","../src/feature-sync.ts"],"sourcesContent":["/**\n * Self-registers this package's build-time manifest before any @smrt() decorator\n * in the package fires. Fixes issue #1132: in consumer runtimes (tsx, SvelteKit\n * SSR, plain `vite dev`) the decorator's synchronous manifest lookup previously\n * missed because no step populated the global manifest cache — classes got\n * registered with zero fields and `save()` / `toJSON()` silently dropped every\n * declared property.\n *\n * Import this module as the first statement in `src/index.ts` so its top-level\n * side effect runs ahead of any class module's @smrt() decorator.\n *\n * Silent no-op in dev/test, where the vitest plugin already populates manifests\n * via a different path. Only needs to succeed in the published dist output.\n *\n * @see https://github.com/happyvertical/smrt/issues/1132\n */\nimport { ObjectRegistry } from '@happyvertical/smrt-core';\n\n// `new URL('./manifest.json', import.meta.url)` resolves at runtime to the\n// manifest sitting next to this module's compiled output. Vite warns at build\n// time that it cannot pre-resolve the URL; that is the intended behavior —\n// the URL must resolve to dist/manifest.json at runtime, not be inlined.\nObjectRegistry.registerPackageManifest(\n new URL('./manifest.json', import.meta.url),\n);\n","import {\n createQualifiedName,\n isQualifiedName,\n ObjectRegistry,\n parseQualifiedName,\n type SmartObjectConfig,\n} from '@happyvertical/smrt-core';\nimport type {\n SmartObjectDefinition,\n SmartObjectManifest,\n} from '@happyvertical/smrt-core/manifest';\nimport type { FeatureDefinitionSeed, FeatureMetadata } from './types.js';\n\ntype RegistryEntryLike = {\n name: string;\n qualifiedName?: string;\n packageName?: string;\n config: SmartObjectConfig;\n visibility?: string;\n};\n\nexport function createFeatureKey(\n qualifiedClassName: string,\n localId: string,\n): string {\n if (!isQualifiedName(qualifiedClassName)) {\n throw new Error(\n `Feature keys require a qualified class name. Received \"${qualifiedClassName}\".`,\n );\n }\n\n if (!localId) {\n throw new Error('Feature localId is required.');\n }\n\n if (localId.includes('#')) {\n throw new Error(\n `Feature localId \"${localId}\" cannot contain \"#\". This character is reserved for canonical feature keys.`,\n );\n }\n\n return `${qualifiedClassName}#${localId}`;\n}\n\nexport function parseFeatureKey(featureKey: string): {\n qualifiedClassName: string;\n localId: string;\n} {\n const separatorIndex = featureKey.lastIndexOf('#');\n if (separatorIndex <= 0 || separatorIndex === featureKey.length - 1) {\n throw new Error(\n `Invalid feature key \"${featureKey}\". Expected format \"<qualifiedClassName>#<localId>\".`,\n );\n }\n\n const qualifiedClassName = featureKey.slice(0, separatorIndex);\n const localId = featureKey.slice(separatorIndex + 1);\n\n if (!isQualifiedName(qualifiedClassName)) {\n throw new Error(\n `Invalid feature key \"${featureKey}\". \"${qualifiedClassName}\" is not a qualified class name.`,\n );\n }\n\n return { qualifiedClassName, localId };\n}\n\nexport function serializeFeatureMetadata(\n metadata: FeatureMetadata | string | null | undefined,\n): string {\n if (!metadata) {\n return '';\n }\n\n if (typeof metadata === 'string') {\n const parsed = parseFeatureMetadataString(metadata);\n return JSON.stringify(parsed);\n }\n\n if (!isFeatureMetadataRecord(metadata)) {\n throw new Error(\n 'Feature metadata must be a plain object or a JSON string representing a plain object.',\n );\n }\n\n return JSON.stringify(metadata);\n}\n\nexport function parseFeatureMetadata(\n metadata: string | null | undefined,\n): FeatureMetadata {\n if (!metadata) {\n return {};\n }\n\n try {\n return parseFeatureMetadataString(metadata);\n } catch {\n return {};\n }\n}\n\nfunction parseFeatureMetadataString(metadata: string): FeatureMetadata {\n let parsed: unknown;\n\n try {\n parsed = JSON.parse(metadata);\n } catch (error) {\n throw new Error(\n 'Feature metadata must be a plain object or a JSON string representing a plain object.',\n { cause: error },\n );\n }\n\n if (!isFeatureMetadataRecord(parsed)) {\n throw new Error(\n 'Feature metadata must be a plain object or a JSON string representing a plain object.',\n );\n }\n\n return parsed;\n}\n\nfunction isFeatureMetadataRecord(value: unknown): value is FeatureMetadata {\n if (!value || typeof value !== 'object' || Array.isArray(value)) {\n return false;\n }\n\n const prototype = Object.getPrototypeOf(value);\n return prototype === Object.prototype || prototype === null;\n}\n\nexport function getQualifiedClassNameFromRegistry(\n registration: RegistryEntryLike,\n): string | null {\n if (\n registration.qualifiedName &&\n isQualifiedName(registration.qualifiedName)\n ) {\n return registration.qualifiedName;\n }\n\n if (registration.packageName) {\n return createQualifiedName(registration.packageName, registration.name);\n }\n\n return null;\n}\n\nexport function getPackageNameFromRegistry(\n registration: RegistryEntryLike,\n): string | null {\n if (registration.packageName) {\n return registration.packageName;\n }\n\n const qualifiedClassName = getQualifiedClassNameFromRegistry(registration);\n if (!qualifiedClassName) {\n return null;\n }\n\n return parseQualifiedName(qualifiedClassName).packageName;\n}\n\nexport function extractFeatureSeedsFromRegistryEntry(\n registration: RegistryEntryLike,\n): FeatureDefinitionSeed[] {\n if (registration.visibility === 'test') {\n return [];\n }\n\n const qualifiedClassName = getQualifiedClassNameFromRegistry(registration);\n if (!qualifiedClassName) {\n return [];\n }\n\n const featureConfig = registration.config.features;\n if (!featureConfig) {\n return [];\n }\n\n const packageName =\n registration.packageName ||\n parseQualifiedName(qualifiedClassName).packageName;\n\n return Object.entries(featureConfig).map(([localId, feature]) => ({\n featureKey: createFeatureKey(qualifiedClassName, localId),\n packageName,\n qualifiedClassName,\n className: registration.name,\n localId,\n defaultEnabled: feature.defaultEnabled,\n label: feature.label,\n description: feature.description,\n metadata: feature.metadata,\n visibility: registration.visibility,\n }));\n}\n\nexport function extractFeatureSeedsFromManifest(\n manifest: SmartObjectManifest,\n): FeatureDefinitionSeed[] {\n const seeds: FeatureDefinitionSeed[] = [];\n\n for (const [objectKey, objectDef] of Object.entries(manifest.objects)) {\n seeds.push(...extractFeatureSeedsFromManifestObject(objectKey, objectDef));\n }\n\n return seeds;\n}\n\nexport function extractTouchedPackagesFromManifest(\n manifest: SmartObjectManifest,\n): Set<string> {\n const packages = new Set<string>();\n\n if (manifest.packageName) {\n packages.add(manifest.packageName);\n }\n\n for (const [objectKey, objectDef] of Object.entries(manifest.objects)) {\n if (objectDef.visibility === 'test') {\n continue;\n }\n\n if (objectDef.packageName) {\n packages.add(objectDef.packageName);\n continue;\n }\n\n const qualifiedClassName = resolveManifestQualifiedClassName(\n objectKey,\n objectDef,\n );\n if (qualifiedClassName) {\n packages.add(parseQualifiedName(qualifiedClassName).packageName);\n }\n }\n\n return packages;\n}\n\nfunction extractFeatureSeedsFromManifestObject(\n objectKey: string,\n objectDef: SmartObjectDefinition,\n): FeatureDefinitionSeed[] {\n if (objectDef.visibility === 'test') {\n return [];\n }\n\n const features = objectDef.decoratorConfig.features;\n if (!features) {\n return [];\n }\n\n const qualifiedClassName = resolveManifestQualifiedClassName(\n objectKey,\n objectDef,\n );\n if (!qualifiedClassName) {\n return [];\n }\n\n const packageName =\n objectDef.packageName || parseQualifiedName(qualifiedClassName).packageName;\n\n return Object.entries(features).map(([localId, feature]) => ({\n featureKey: createFeatureKey(qualifiedClassName, localId),\n packageName,\n qualifiedClassName,\n className: objectDef.className,\n localId,\n defaultEnabled: feature.defaultEnabled,\n label: feature.label,\n description: feature.description,\n metadata:\n feature.metadata && typeof feature.metadata === 'object'\n ? (feature.metadata as FeatureMetadata)\n : undefined,\n visibility: objectDef.visibility,\n }));\n}\n\nfunction resolveManifestQualifiedClassName(\n objectKey: string,\n objectDef: SmartObjectDefinition,\n): string | null {\n if (objectDef.qualifiedName && isQualifiedName(objectDef.qualifiedName)) {\n return objectDef.qualifiedName;\n }\n\n if (isQualifiedName(objectKey)) {\n return objectKey;\n }\n\n if (objectDef.packageName) {\n return createQualifiedName(objectDef.packageName, objectDef.className);\n }\n\n return null;\n}\n\nexport function resolveFeatureKeyForTarget(\n classOrInstance: object | (new (...args: any[]) => any),\n localId: string,\n): {\n featureKey: string;\n defaultEnabled: boolean;\n} {\n const ctor =\n typeof classOrInstance === 'function'\n ? (classOrInstance as new (\n ...args: any[]\n ) => any)\n : (classOrInstance as any).constructor;\n\n const registration = ObjectRegistry.getClassByConstructor(ctor);\n if (!registration) {\n throw new Error(\n `Cannot resolve feature \"${localId}\" because ${ctor?.name || 'the target class'} is not registered with @smrt().`,\n );\n }\n\n const qualifiedClassName = getQualifiedClassNameFromRegistry(registration);\n if (!qualifiedClassName) {\n throw new Error(\n `Cannot resolve feature \"${localId}\" for ${registration.name} because the class has no qualified package identity.`,\n );\n }\n\n const feature = registration.config.features?.[localId];\n if (!feature) {\n throw new Error(\n `Feature \"${localId}\" is not declared on ${qualifiedClassName}.`,\n );\n }\n\n return {\n featureKey: createFeatureKey(qualifiedClassName, localId),\n defaultEnabled: feature.defaultEnabled,\n };\n}\n\nexport function findFeatureDefaultInRegistry(\n featureKey: string,\n): boolean | undefined {\n const { qualifiedClassName, localId } = parseFeatureKey(featureKey);\n const registration =\n ObjectRegistry.getClassByQualifiedName(qualifiedClassName) ||\n ObjectRegistry.getClass(qualifiedClassName);\n\n const feature = registration?.config.features?.[localId];\n return feature?.defaultEnabled;\n}\n","import { SmrtObject, smrt } from '@happyvertical/smrt-core';\nimport type { FeatureDefinitionOptions, FeatureMetadata } from './types.js';\nimport { parseFeatureMetadata, serializeFeatureMetadata } from './utils.js';\n\n@smrt({\n tableName: '_smrt_feature_definitions',\n api: { include: ['list', 'get'] },\n cli: { include: ['list', 'get'], exclude: ['getMetadata', 'setMetadata'] },\n mcp: { include: ['list', 'get'], exclude: ['getMetadata', 'setMetadata'] },\n conflictColumns: ['feature_key'],\n})\nexport class FeatureDefinition extends SmrtObject {\n featureKey: string = '';\n packageName: string = '';\n qualifiedClassName: string = '';\n className: string = '';\n localId: string = '';\n defaultEnabled: boolean = false;\n label: string = '';\n description: string = '';\n metadata: string = '';\n visibility: string = 'public';\n\n constructor(options: FeatureDefinitionOptions = {}) {\n super(options);\n\n if (options.featureKey !== undefined) this.featureKey = options.featureKey;\n if (options.packageName !== undefined)\n this.packageName = options.packageName;\n if (options.qualifiedClassName !== undefined) {\n this.qualifiedClassName = options.qualifiedClassName;\n }\n if (options.className !== undefined) this.className = options.className;\n if (options.localId !== undefined) this.localId = options.localId;\n if (options.defaultEnabled !== undefined) {\n this.defaultEnabled = options.defaultEnabled;\n }\n if (options.label !== undefined) this.label = options.label;\n if (options.description !== undefined)\n this.description = options.description;\n if (options.visibility !== undefined) this.visibility = options.visibility;\n if (options.metadata !== undefined) {\n this.metadata = serializeFeatureMetadata(options.metadata);\n }\n }\n\n getMetadata(): FeatureMetadata {\n return parseFeatureMetadata(this.metadata);\n }\n\n setMetadata(metadata: FeatureMetadata | null): void {\n this.metadata = serializeFeatureMetadata(metadata);\n }\n}\n","import { SmrtCollection } from '@happyvertical/smrt-core';\nimport { FeatureDefinition } from './feature-definition.js';\nimport type { FeatureDefinitionSeed } from './types.js';\nimport { serializeFeatureMetadata } from './utils.js';\n\nexport class FeatureDefinitionCollection extends SmrtCollection<FeatureDefinition> {\n static readonly _itemClass = FeatureDefinition;\n\n async findByFeatureKey(\n featureKey: string,\n ): Promise<FeatureDefinition | null> {\n const results = await this.list({\n where: { featureKey },\n limit: 1,\n });\n return results[0] ?? null;\n }\n\n async findByPackageName(packageName: string): Promise<FeatureDefinition[]> {\n return this.list({\n where: { packageName },\n });\n }\n\n async upsertDefinition(seed: FeatureDefinitionSeed): Promise<{\n definition: FeatureDefinition;\n status: 'created' | 'updated' | 'unchanged';\n }> {\n const existing = await this.findByFeatureKey(seed.featureKey);\n\n if (!existing) {\n const created = await this.create({\n ...seed,\n metadata: serializeFeatureMetadata(seed.metadata),\n });\n await created.save();\n return { definition: created, status: 'created' };\n }\n\n const nextMetadata = serializeFeatureMetadata(seed.metadata);\n const changed =\n existing.packageName !== seed.packageName ||\n existing.qualifiedClassName !== seed.qualifiedClassName ||\n existing.className !== seed.className ||\n existing.localId !== seed.localId ||\n existing.defaultEnabled !== seed.defaultEnabled ||\n existing.label !== (seed.label ?? '') ||\n existing.description !== (seed.description ?? '') ||\n existing.metadata !== nextMetadata ||\n existing.visibility !== (seed.visibility ?? 'public');\n\n if (!changed) {\n return { definition: existing, status: 'unchanged' };\n }\n\n existing.packageName = seed.packageName;\n existing.qualifiedClassName = seed.qualifiedClassName;\n existing.className = seed.className;\n existing.localId = seed.localId;\n existing.defaultEnabled = seed.defaultEnabled;\n existing.label = seed.label ?? '';\n existing.description = seed.description ?? '';\n existing.metadata = nextMetadata;\n existing.visibility = seed.visibility ?? 'public';\n await existing.save();\n\n return { definition: existing, status: 'updated' };\n }\n}\n","import { SmrtObject, smrt } from '@happyvertical/smrt-core';\nimport {\n FeatureOverrideEffect,\n type FeatureOverrideOptions,\n type FeatureScopeType,\n GLOBAL_FEATURE_SCOPE_ID,\n} from './types.js';\n\n@smrt({\n tableName: '_smrt_feature_overrides',\n api: { include: ['list', 'get', 'create', 'update', 'delete'] },\n cli: {\n exclude: ['isInherit', 'isEnabled', 'isDisabled'],\n },\n mcp: {\n exclude: ['isInherit', 'isEnabled', 'isDisabled'],\n },\n conflictColumns: ['feature_key', 'scope_type', 'scope_id'],\n})\nexport class FeatureOverride extends SmrtObject {\n featureKey: string = '';\n scopeType: FeatureScopeType = 'global';\n scopeId: string = GLOBAL_FEATURE_SCOPE_ID;\n effect: FeatureOverrideEffect = FeatureOverrideEffect.INHERIT;\n\n constructor(options: FeatureOverrideOptions = {}) {\n super(options);\n\n if (options.featureKey !== undefined) this.featureKey = options.featureKey;\n if (options.scopeType !== undefined) this.scopeType = options.scopeType;\n if (options.scopeId !== undefined) this.scopeId = options.scopeId;\n if (options.effect !== undefined) this.effect = options.effect;\n }\n\n isInherit(): boolean {\n return this.effect === FeatureOverrideEffect.INHERIT;\n }\n\n isEnabled(): boolean {\n return this.effect === FeatureOverrideEffect.ENABLE;\n }\n\n isDisabled(): boolean {\n return this.effect === FeatureOverrideEffect.DISABLE;\n }\n}\n","import { SmrtCollection } from '@happyvertical/smrt-core';\nimport { FeatureOverride } from './feature-override.js';\nimport {\n type FeatureOverrideEffect,\n type FeatureScopeType,\n GLOBAL_FEATURE_SCOPE_ID,\n} from './types.js';\n\nexport class FeatureOverrideCollection extends SmrtCollection<FeatureOverride> {\n static readonly _itemClass = FeatureOverride;\n\n async findByFeatureKey(featureKey: string): Promise<FeatureOverride[]> {\n return this.list({\n where: { featureKey },\n });\n }\n\n async findByFeatureAndScope(\n featureKey: string,\n scopeType: FeatureScopeType,\n scopeId: string,\n ): Promise<FeatureOverride | null> {\n const results = await this.list({\n where: { featureKey, scopeType, scopeId },\n limit: 1,\n });\n return results[0] ?? null;\n }\n\n async getGlobalOverride(featureKey: string): Promise<FeatureOverride | null> {\n return this.findByFeatureAndScope(\n featureKey,\n 'global',\n GLOBAL_FEATURE_SCOPE_ID,\n );\n }\n\n async getTenantOverride(\n featureKey: string,\n tenantId: string,\n ): Promise<FeatureOverride | null> {\n return this.findByFeatureAndScope(featureKey, 'tenant', tenantId);\n }\n\n async getOverrideMap(\n featureKey: string,\n scopeType: FeatureScopeType,\n scopeIds: string[],\n ): Promise<Map<string, FeatureOverride>> {\n const result = new Map<string, FeatureOverride>();\n if (scopeIds.length === 0) {\n return result;\n }\n\n const overrides = await this.list({\n where: { featureKey, scopeType, scopeId: scopeIds },\n });\n\n for (const override of overrides) {\n result.set(override.scopeId, override);\n }\n\n return result;\n }\n\n async setOverride(\n featureKey: string,\n scopeType: FeatureScopeType,\n scopeId: string,\n effect: FeatureOverrideEffect,\n ): Promise<FeatureOverride> {\n const existing = await this.findByFeatureAndScope(\n featureKey,\n scopeType,\n scopeId,\n );\n\n if (existing) {\n existing.effect = effect;\n await existing.save();\n return existing;\n }\n\n const created = await this.create({\n featureKey,\n scopeType,\n scopeId,\n effect,\n });\n await created.save();\n return created;\n }\n\n async setGlobalOverride(\n featureKey: string,\n effect: FeatureOverrideEffect,\n ): Promise<FeatureOverride> {\n return this.setOverride(\n featureKey,\n 'global',\n GLOBAL_FEATURE_SCOPE_ID,\n effect,\n );\n }\n\n async setTenantOverride(\n featureKey: string,\n tenantId: string,\n effect: FeatureOverrideEffect,\n ): Promise<FeatureOverride> {\n return this.setOverride(featureKey, 'tenant', tenantId, effect);\n }\n\n async removeOverride(\n featureKey: string,\n scopeType: FeatureScopeType,\n scopeId: string,\n ): Promise<boolean> {\n const existing = await this.findByFeatureAndScope(\n featureKey,\n scopeType,\n scopeId,\n );\n\n if (!existing) {\n return false;\n }\n\n await existing.delete();\n return true;\n }\n}\n","import type { SmrtClassOptions } from '@happyvertical/smrt-core';\nimport { importWorkspaceModule } from '@happyvertical/smrt-core/utils/import-workspace-module';\nimport { FeatureDefinitionCollection } from './feature-definitions.js';\nimport { FeatureOverrideCollection } from './feature-overrides.js';\nimport {\n FeatureOverrideEffect,\n type FeatureResolutionContext,\n type FeatureResolverOptions,\n type FeatureTenantHierarchyProvider,\n type FeatureTenantNode,\n type FeatureUsersModule,\n} from './types.js';\nimport {\n findFeatureDefaultInRegistry,\n resolveFeatureKeyForTarget,\n} from './utils.js';\n\nexport class FeatureResolver {\n private readonly options: SmrtClassOptions;\n private readonly resolverOptions: FeatureResolverOptions;\n private featureDefinitions!: FeatureDefinitionCollection;\n private featureOverrides!: FeatureOverrideCollection;\n private initializationPromise: Promise<void> | null = null;\n private tenantHierarchyPromise: Promise<FeatureTenantHierarchyProvider | null> | null =\n null;\n\n constructor(\n options: SmrtClassOptions = {},\n resolverOptions: FeatureResolverOptions = {},\n ) {\n this.options = options;\n this.resolverOptions = resolverOptions;\n }\n\n async isEnabled(\n featureKey: string,\n context: FeatureResolutionContext = {},\n ): Promise<boolean> {\n await this.ensureInitialized();\n\n const baseState = await this.resolveBaseState(featureKey);\n const globalOverride =\n await this.featureOverrides.getGlobalOverride(featureKey);\n const globalState = this.applyOverride(baseState, globalOverride?.effect);\n\n if (!context.tenantId) {\n return globalState;\n }\n\n const tenantHierarchy = await this.getTenantHierarchy();\n if (!tenantHierarchy) {\n const directOverride = await this.featureOverrides.getTenantOverride(\n featureKey,\n context.tenantId,\n );\n return this.applyOverride(globalState, directOverride?.effect);\n }\n\n const chain = await tenantHierarchy.getChain(context.tenantId);\n if (chain.length === 0) {\n const directOverride = await this.featureOverrides.getTenantOverride(\n featureKey,\n context.tenantId,\n );\n return this.applyOverride(globalState, directOverride?.effect);\n }\n\n const overrides = await this.featureOverrides.getOverrideMap(\n featureKey,\n 'tenant',\n chain.map((node) => node.id),\n );\n\n let inheritedState = globalState;\n\n for (let index = 0; index < chain.length; index++) {\n const current = chain[index];\n const previous = index > 0 ? chain[index - 1] : null;\n const shouldInherit =\n index === 0 ||\n (!!previous?.cascadePermissions && current.inheritPermissions);\n\n const baseline = shouldInherit ? inheritedState : globalState;\n const override = overrides.get(current.id);\n inheritedState = this.applyOverride(baseline, override?.effect);\n }\n\n return inheritedState;\n }\n\n async isEnabledFor(\n classOrInstance: object | (new (...args: any[]) => any),\n localId: string,\n context: FeatureResolutionContext = {},\n ): Promise<boolean> {\n const { featureKey } = resolveFeatureKeyForTarget(classOrInstance, localId);\n return this.isEnabled(featureKey, context);\n }\n\n private async ensureInitialized(): Promise<void> {\n if (!this.initializationPromise) {\n this.initializationPromise = (async () => {\n this.featureDefinitions = await (\n FeatureDefinitionCollection as any\n ).create(this.options);\n this.featureOverrides = await (FeatureOverrideCollection as any).create(\n this.options,\n );\n })();\n }\n\n await this.initializationPromise;\n }\n\n private async resolveBaseState(featureKey: string): Promise<boolean> {\n const registryDefault = findFeatureDefaultInRegistry(featureKey);\n if (registryDefault !== undefined) {\n return registryDefault;\n }\n\n const definition =\n await this.featureDefinitions.findByFeatureKey(featureKey);\n if (!definition) {\n throw new Error(\n `Feature \"${featureKey}\" is not declared in the registry or synced feature definitions.`,\n );\n }\n\n return definition.defaultEnabled;\n }\n\n private applyOverride(\n currentState: boolean,\n effect: FeatureOverrideEffect | undefined,\n ): boolean {\n switch (effect) {\n case FeatureOverrideEffect.ENABLE:\n return true;\n case FeatureOverrideEffect.DISABLE:\n return false;\n default:\n return currentState;\n }\n }\n\n private async getTenantHierarchy(): Promise<FeatureTenantHierarchyProvider | null> {\n if (!this.tenantHierarchyPromise) {\n const loader =\n this.resolverOptions.tenantHierarchyLoader ||\n defaultTenantHierarchyLoader;\n this.tenantHierarchyPromise = loader(this.options);\n }\n\n return this.tenantHierarchyPromise;\n }\n}\n\nasync function defaultTenantHierarchyLoader(\n options: SmrtClassOptions,\n): Promise<FeatureTenantHierarchyProvider | null> {\n try {\n const usersModule = await importWorkspaceModule<FeatureUsersModule>({\n packageName: '@happyvertical/smrt-users',\n sourceEntry: 'packages/users/src/collections/index.ts',\n purpose: 'tenant-aware feature flag resolution',\n });\n\n const tenantCollection = await usersModule.TenantCollection.create(options);\n return {\n async getChain(tenantId: string): Promise<FeatureTenantNode[]> {\n const tenant = await tenantCollection.get({ id: tenantId });\n if (!tenant) {\n return [];\n }\n\n const ancestors = await tenantCollection.getAncestorsFromRoot(tenantId);\n return [...ancestors, tenant].map((node: any) => ({\n id: String(node.id),\n inheritPermissions: Boolean(node.inheritPermissions),\n cascadePermissions: Boolean(node.cascadePermissions),\n }));\n },\n };\n } catch (error) {\n if (isMissingUsersDependency(error)) {\n return null;\n }\n throw error;\n }\n}\n\nfunction isMissingUsersDependency(error: unknown): boolean {\n if (!(error instanceof Error)) {\n return false;\n }\n\n const causeMessage =\n error.cause instanceof Error ? ` ${error.cause.message}` : '';\n const text = `${error.message}${causeMessage}`;\n\n return (\n text.includes('@happyvertical/smrt-users') &&\n (text.includes('Cannot find module') ||\n text.includes('Failed to load') ||\n text.includes('could not find'))\n );\n}\n","import {\n ObjectRegistry,\n type SmrtClassOptions,\n} from '@happyvertical/smrt-core';\nimport type { SmartObjectManifest } from '@happyvertical/smrt-core/manifest';\nimport { FeatureDefinitionCollection } from './feature-definitions.js';\nimport type {\n FeatureDefinitionSeed,\n FeatureSyncResult,\n SyncDefinitionsOptions,\n SyncManifestOptions,\n} from './types.js';\nimport {\n extractFeatureSeedsFromManifest,\n extractFeatureSeedsFromRegistryEntry,\n extractTouchedPackagesFromManifest,\n getPackageNameFromRegistry,\n} from './utils.js';\n\nexport class FeatureSyncService {\n private readonly options: SmrtClassOptions;\n private featureDefinitions!: FeatureDefinitionCollection;\n private initializationPromise: Promise<void> | null = null;\n\n constructor(options: SmrtClassOptions = {}) {\n this.options = options;\n }\n\n async syncDefinitions(\n options: SyncDefinitionsOptions = {},\n ): Promise<FeatureSyncResult> {\n await this.ensureInitialized();\n const registrations = this.collectRegistrations(options);\n const definitions = this.collectDefinitionsFromRegistrations(registrations);\n const touchedPackages =\n this.collectTouchedPackagesFromRegistrations(registrations);\n const isFilteredSync =\n Boolean(options.classNames?.length) ||\n Boolean(options.constructors?.length);\n const pruneStale = isFilteredSync ? false : (options.pruneStale ?? true);\n\n return this.applyDefinitions(definitions, touchedPackages, pruneStale);\n }\n\n async syncManifest(\n manifest: SmartObjectManifest,\n options: SyncManifestOptions = {},\n ): Promise<FeatureSyncResult> {\n await this.ensureInitialized();\n const definitions = extractFeatureSeedsFromManifest(manifest);\n const touchedPackages = extractTouchedPackagesFromManifest(manifest);\n return this.applyDefinitions(\n definitions,\n touchedPackages,\n options.pruneStale ?? true,\n );\n }\n\n private async ensureInitialized(): Promise<void> {\n if (!this.initializationPromise) {\n this.initializationPromise = (async () => {\n this.featureDefinitions = await (\n FeatureDefinitionCollection as any\n ).create(this.options);\n })();\n }\n\n await this.initializationPromise;\n }\n\n private collectDefinitionsFromRegistrations(\n registrations: Array<any>,\n ): FeatureDefinitionSeed[] {\n const definitions: FeatureDefinitionSeed[] = [];\n\n for (const registration of registrations) {\n definitions.push(...extractFeatureSeedsFromRegistryEntry(registration));\n }\n\n return definitions;\n }\n\n private collectTouchedPackagesFromRegistrations(\n registrations: Array<any>,\n ): Set<string> {\n const packages = new Set<string>();\n\n for (const registration of registrations) {\n if (registration.visibility === 'test') {\n continue;\n }\n\n const packageName = getPackageNameFromRegistry(registration);\n if (packageName) {\n packages.add(packageName);\n }\n }\n\n return packages;\n }\n\n private collectRegistrations(options: SyncDefinitionsOptions): Array<any> {\n if (options.classNames?.length) {\n return options.classNames\n .map((name) => ObjectRegistry.getClass(name))\n .filter((value): value is NonNullable<typeof value> => !!value);\n }\n\n if (options.constructors?.length) {\n return options.constructors\n .map((ctor) => ObjectRegistry.getClassByConstructor(ctor as any))\n .filter((value): value is NonNullable<typeof value> => !!value);\n }\n\n const registrations: any[] = [];\n const seen = new Set<any>();\n\n for (const registration of ObjectRegistry.getAllClasses().values()) {\n if (seen.has(registration)) {\n continue;\n }\n seen.add(registration);\n registrations.push(registration);\n }\n\n return registrations;\n }\n\n private async applyDefinitions(\n definitions: FeatureDefinitionSeed[],\n touchedPackages: Set<string>,\n pruneStale: boolean,\n ): Promise<FeatureSyncResult> {\n const deduped = new Map<string, FeatureDefinitionSeed>();\n for (const definition of definitions) {\n deduped.set(definition.featureKey, definition);\n }\n\n let created = 0;\n let updated = 0;\n let unchanged = 0;\n let deleted = 0;\n\n const dedupedDefinitions = Array.from(deduped.values());\n const currentKeys = new Set(\n dedupedDefinitions.map((definition) => definition.featureKey),\n );\n const upsertResults = await Promise.all(\n dedupedDefinitions.map((definition) =>\n this.featureDefinitions.upsertDefinition(definition),\n ),\n );\n for (const result of upsertResults) {\n if (result.status === 'created') created++;\n if (result.status === 'updated') updated++;\n if (result.status === 'unchanged') unchanged++;\n }\n\n if (pruneStale) {\n const existingByPackage = await Promise.all(\n Array.from(touchedPackages, (packageName) =>\n this.featureDefinitions.findByPackageName(packageName),\n ),\n );\n\n const staleDefinitions = existingByPackage.flatMap((existing) =>\n existing.filter(\n (definition) => !currentKeys.has(definition.featureKey),\n ),\n );\n\n await Promise.all(\n staleDefinitions.map((definition) => definition.delete()),\n );\n deleted = staleDefinitions.length;\n }\n\n return {\n total: deduped.size,\n created,\n updated,\n unchanged,\n deleted,\n featureKeys: Array.from(currentKeys).sort(),\n };\n }\n}\n"],"names":["__decorateClass"],"mappings":";;;AAsBA,eAAe;AAAA,EACb,IAAA,IAAA,mBAAA,YAAA,GAAA;AACF;ACHO,SAAS,iBACd,oBACA,SACQ;AACR,MAAI,CAAC,gBAAgB,kBAAkB,GAAG;AACxC,UAAM,IAAI;AAAA,MACR,0DAA0D,kBAAkB;AAAA,IAAA;AAAA,EAEhF;AAEA,MAAI,CAAC,SAAS;AACZ,UAAM,IAAI,MAAM,8BAA8B;AAAA,EAChD;AAEA,MAAI,QAAQ,SAAS,GAAG,GAAG;AACzB,UAAM,IAAI;AAAA,MACR,oBAAoB,OAAO;AAAA,IAAA;AAAA,EAE/B;AAEA,SAAO,GAAG,kBAAkB,IAAI,OAAO;AACzC;AAEO,SAAS,gBAAgB,YAG9B;AACA,QAAM,iBAAiB,WAAW,YAAY,GAAG;AACjD,MAAI,kBAAkB,KAAK,mBAAmB,WAAW,SAAS,GAAG;AACnE,UAAM,IAAI;AAAA,MACR,wBAAwB,UAAU;AAAA,IAAA;AAAA,EAEtC;AAEA,QAAM,qBAAqB,WAAW,MAAM,GAAG,cAAc;AAC7D,QAAM,UAAU,WAAW,MAAM,iBAAiB,CAAC;AAEnD,MAAI,CAAC,gBAAgB,kBAAkB,GAAG;AACxC,UAAM,IAAI;AAAA,MACR,wBAAwB,UAAU,OAAO,kBAAkB;AAAA,IAAA;AAAA,EAE/D;AAEA,SAAO,EAAE,oBAAoB,QAAA;AAC/B;AAEO,SAAS,yBACd,UACQ;AACR,MAAI,CAAC,UAAU;AACb,WAAO;AAAA,EACT;AAEA,MAAI,OAAO,aAAa,UAAU;AAChC,UAAM,SAAS,2BAA2B,QAAQ;AAClD,WAAO,KAAK,UAAU,MAAM;AAAA,EAC9B;AAEA,MAAI,CAAC,wBAAwB,QAAQ,GAAG;AACtC,UAAM,IAAI;AAAA,MACR;AAAA,IAAA;AAAA,EAEJ;AAEA,SAAO,KAAK,UAAU,QAAQ;AAChC;AAEO,SAAS,qBACd,UACiB;AACjB,MAAI,CAAC,UAAU;AACb,WAAO,CAAA;AAAA,EACT;AAEA,MAAI;AACF,WAAO,2BAA2B,QAAQ;AAAA,EAC5C,QAAQ;AACN,WAAO,CAAA;AAAA,EACT;AACF;AAEA,SAAS,2BAA2B,UAAmC;AACrE,MAAI;AAEJ,MAAI;AACF,aAAS,KAAK,MAAM,QAAQ;AAAA,EAC9B,SAAS,OAAO;AACd,UAAM,IAAI;AAAA,MACR;AAAA,MACA,EAAE,OAAO,MAAA;AAAA,IAAM;AAAA,EAEnB;AAEA,MAAI,CAAC,wBAAwB,MAAM,GAAG;AACpC,UAAM,IAAI;AAAA,MACR;AAAA,IAAA;AAAA,EAEJ;AAEA,SAAO;AACT;AAEA,SAAS,wBAAwB,OAA0C;AACzE,MAAI,CAAC,SAAS,OAAO,UAAU,YAAY,MAAM,QAAQ,KAAK,GAAG;AAC/D,WAAO;AAAA,EACT;AAEA,QAAM,YAAY,OAAO,eAAe,KAAK;AAC7C,SAAO,cAAc,OAAO,aAAa,cAAc;AACzD;AAEO,SAAS,kCACd,cACe;AACf,MACE,aAAa,iBACb,gBAAgB,aAAa,aAAa,GAC1C;AACA,WAAO,aAAa;AAAA,EACtB;AAEA,MAAI,aAAa,aAAa;AAC5B,WAAO,oBAAoB,aAAa,aAAa,aAAa,IAAI;AAAA,EACxE;AAEA,SAAO;AACT;AAEO,SAAS,2BACd,cACe;AACf,MAAI,aAAa,aAAa;AAC5B,WAAO,aAAa;AAAA,EACtB;AAEA,QAAM,qBAAqB,kCAAkC,YAAY;AACzE,MAAI,CAAC,oBAAoB;AACvB,WAAO;AAAA,EACT;AAEA,SAAO,mBAAmB,kBAAkB,EAAE;AAChD;AAEO,SAAS,qCACd,cACyB;AACzB,MAAI,aAAa,eAAe,QAAQ;AACtC,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,qBAAqB,kCAAkC,YAAY;AACzE,MAAI,CAAC,oBAAoB;AACvB,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,gBAAgB,aAAa,OAAO;AAC1C,MAAI,CAAC,eAAe;AAClB,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,cACJ,aAAa,eACb,mBAAmB,kBAAkB,EAAE;AAEzC,SAAO,OAAO,QAAQ,aAAa,EAAE,IAAI,CAAC,CAAC,SAAS,OAAO,OAAO;AAAA,IAChE,YAAY,iBAAiB,oBAAoB,OAAO;AAAA,IACxD;AAAA,IACA;AAAA,IACA,WAAW,aAAa;AAAA,IACxB;AAAA,IACA,gBAAgB,QAAQ;AAAA,IACxB,OAAO,QAAQ;AAAA,IACf,aAAa,QAAQ;AAAA,IACrB,UAAU,QAAQ;AAAA,IAClB,YAAY,aAAa;AAAA,EAAA,EACzB;AACJ;AAEO,SAAS,gCACd,UACyB;AACzB,QAAM,QAAiC,CAAA;AAEvC,aAAW,CAAC,WAAW,SAAS,KAAK,OAAO,QAAQ,SAAS,OAAO,GAAG;AACrE,UAAM,KAAK,GAAG,sCAAsC,WAAW,SAAS,CAAC;AAAA,EAC3E;AAEA,SAAO;AACT;AAEO,SAAS,mCACd,UACa;AACb,QAAM,+BAAe,IAAA;AAErB,MAAI,SAAS,aAAa;AACxB,aAAS,IAAI,SAAS,WAAW;AAAA,EACnC;AAEA,aAAW,CAAC,WAAW,SAAS,KAAK,OAAO,QAAQ,SAAS,OAAO,GAAG;AACrE,QAAI,UAAU,eAAe,QAAQ;AACnC;AAAA,IACF;AAEA,QAAI,UAAU,aAAa;AACzB,eAAS,IAAI,UAAU,WAAW;AAClC;AAAA,IACF;AAEA,UAAM,qBAAqB;AAAA,MACzB;AAAA,MACA;AAAA,IAAA;AAEF,QAAI,oBAAoB;AACtB,eAAS,IAAI,mBAAmB,kBAAkB,EAAE,WAAW;AAAA,IACjE;AAAA,EACF;AAEA,SAAO;AACT;AAEA,SAAS,sCACP,WACA,WACyB;AACzB,MAAI,UAAU,eAAe,QAAQ;AACnC,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,WAAW,UAAU,gBAAgB;AAC3C,MAAI,CAAC,UAAU;AACb,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,qBAAqB;AAAA,IACzB;AAAA,IACA;AAAA,EAAA;AAEF,MAAI,CAAC,oBAAoB;AACvB,WAAO,CAAA;AAAA,EACT;AAEA,QAAM,cACJ,UAAU,eAAe,mBAAmB,kBAAkB,EAAE;AAElE,SAAO,OAAO,QAAQ,QAAQ,EAAE,IAAI,CAAC,CAAC,SAAS,OAAO,OAAO;AAAA,IAC3D,YAAY,iBAAiB,oBAAoB,OAAO;AAAA,IACxD;AAAA,IACA;AAAA,IACA,WAAW,UAAU;AAAA,IACrB;AAAA,IACA,gBAAgB,QAAQ;AAAA,IACxB,OAAO,QAAQ;AAAA,IACf,aAAa,QAAQ;AAAA,IACrB,UACE,QAAQ,YAAY,OAAO,QAAQ,aAAa,WAC3C,QAAQ,WACT;AAAA,IACN,YAAY,UAAU;AAAA,EAAA,EACtB;AACJ;AAEA,SAAS,kCACP,WACA,WACe;AACf,MAAI,UAAU,iBAAiB,gBAAgB,UAAU,aAAa,GAAG;AACvE,WAAO,UAAU;AAAA,EACnB;AAEA,MAAI,gBAAgB,SAAS,GAAG;AAC9B,WAAO;AAAA,EACT;AAEA,MAAI,UAAU,aAAa;AACzB,WAAO,oBAAoB,UAAU,aAAa,UAAU,SAAS;AAAA,EACvE;AAEA,SAAO;AACT;AAEO,SAAS,2BACd,iBACA,SAIA;AACA,QAAM,OACJ,OAAO,oBAAoB,aACtB,kBAGA,gBAAwB;AAE/B,QAAM,eAAe,eAAe,sBAAsB,IAAI;AAC9D,MAAI,CAAC,cAAc;AACjB,UAAM,IAAI;AAAA,MACR,2BAA2B,OAAO,aAAa,MAAM,QAAQ,kBAAkB;AAAA,IAAA;AAAA,EAEnF;AAEA,QAAM,qBAAqB,kCAAkC,YAAY;AACzE,MAAI,CAAC,oBAAoB;AACvB,UAAM,IAAI;AAAA,MACR,2BAA2B,OAAO,SAAS,aAAa,IAAI;AAAA,IAAA;AAAA,EAEhE;AAEA,QAAM,UAAU,aAAa,OAAO,WAAW,OAAO;AACtD,MAAI,CAAC,SAAS;AACZ,UAAM,IAAI;AAAA,MACR,YAAY,OAAO,wBAAwB,kBAAkB;AAAA,IAAA;AAAA,EAEjE;AAEA,SAAO;AAAA,IACL,YAAY,iBAAiB,oBAAoB,OAAO;AAAA,IACxD,gBAAgB,QAAQ;AAAA,EAAA;AAE5B;AAEO,SAAS,6BACd,YACqB;AACrB,QAAM,EAAE,oBAAoB,YAAY,gBAAgB,UAAU;AAClE,QAAM,eACJ,eAAe,wBAAwB,kBAAkB,KACzD,eAAe,SAAS,kBAAkB;AAE5C,QAAM,UAAU,cAAc,OAAO,WAAW,OAAO;AACvD,SAAO,SAAS;AAClB;;;;;;;;;ACtVO,IAAM,oBAAN,cAAgC,WAAW;AAAA,EAChD,aAAqB;AAAA,EACrB,cAAsB;AAAA,EACtB,qBAA6B;AAAA,EAC7B,YAAoB;AAAA,EACpB,UAAkB;AAAA,EAClB,iBAA0B;AAAA,EAC1B,QAAgB;AAAA,EAChB,cAAsB;AAAA,EACtB,WAAmB;AAAA,EACnB,aAAqB;AAAA,EAErB,YAAY,UAAoC,IAAI;AAClD,UAAM,OAAO;AAEb,QAAI,QAAQ,eAAe,OAAW,MAAK,aAAa,QAAQ;AAChE,QAAI,QAAQ,gBAAgB;AAC1B,WAAK,cAAc,QAAQ;AAC7B,QAAI,QAAQ,uBAAuB,QAAW;AAC5C,WAAK,qBAAqB,QAAQ;AAAA,IACpC;AACA,QAAI,QAAQ,cAAc,OAAW,MAAK,YAAY,QAAQ;AAC9D,QAAI,QAAQ,YAAY,OAAW,MAAK,UAAU,QAAQ;AAC1D,QAAI,QAAQ,mBAAmB,QAAW;AACxC,WAAK,iBAAiB,QAAQ;AAAA,IAChC;AACA,QAAI,QAAQ,UAAU,OAAW,MAAK,QAAQ,QAAQ;AACtD,QAAI,QAAQ,gBAAgB;AAC1B,WAAK,cAAc,QAAQ;AAC7B,QAAI,QAAQ,eAAe,OAAW,MAAK,aAAa,QAAQ;AAChE,QAAI,QAAQ,aAAa,QAAW;AAClC,WAAK,WAAW,yBAAyB,QAAQ,QAAQ;AAAA,IAC3D;AAAA,EACF;AAAA,EAEA,cAA+B;AAC7B,WAAO,qBAAqB,KAAK,QAAQ;AAAA,EAC3C;AAAA,EAEA,YAAY,UAAwC;AAClD,SAAK,WAAW,yBAAyB,QAAQ;AAAA,EACnD;AACF;AA1Ca,oBAANA,kBAAA;AAAA,EAPN,KAAK;AAAA,IACJ,WAAW;AAAA,IACX,KAAK,EAAE,SAAS,CAAC,QAAQ,KAAK,EAAA;AAAA,IAC9B,KAAK,EAAE,SAAS,CAAC,QAAQ,KAAK,GAAG,SAAS,CAAC,eAAe,aAAa,EAAA;AAAA,IACvE,KAAK,EAAE,SAAS,CAAC,QAAQ,KAAK,GAAG,SAAS,CAAC,eAAe,aAAa,EAAA;AAAA,IACvE,iBAAiB,CAAC,aAAa;AAAA,EAAA,CAChC;AAAA,GACY,iBAAA;ACNN,MAAM,oCAAoC,eAAkC;AAAA,EACjF,OAAgB,aAAa;AAAA,EAE7B,MAAM,iBACJ,YACmC;AACnC,UAAM,UAAU,MAAM,KAAK,KAAK;AAAA,MAC9B,OAAO,EAAE,WAAA;AAAA,MACT,OAAO;AAAA,IAAA,CACR;AACD,WAAO,QAAQ,CAAC,KAAK;AAAA,EACvB;AAAA,EAEA,MAAM,kBAAkB,aAAmD;AACzE,WAAO,KAAK,KAAK;AAAA,MACf,OAAO,EAAE,YAAA;AAAA,IAAY,CACtB;AAAA,EACH;AAAA,EAEA,MAAM,iBAAiB,MAGpB;AACD,UAAM,WAAW,MAAM,KAAK,iBAAiB,KAAK,UAAU;AAE5D,QAAI,CAAC,UAAU;AACb,YAAM,UAAU,MAAM,KAAK,OAAO;AAAA,QAChC,GAAG;AAAA,QACH,UAAU,yBAAyB,KAAK,QAAQ;AAAA,MAAA,CACjD;AACD,YAAM,QAAQ,KAAA;AACd,aAAO,EAAE,YAAY,SAAS,QAAQ,UAAA;AAAA,IACxC;AAEA,UAAM,eAAe,yBAAyB,KAAK,QAAQ;AAC3D,UAAM,UACJ,SAAS,gBAAgB,KAAK,eAC9B,SAAS,uBAAuB,KAAK,sBACrC,SAAS,cAAc,KAAK,aAC5B,SAAS,YAAY,KAAK,WAC1B,SAAS,mBAAmB,KAAK,kBACjC,SAAS,WAAW,KAAK,SAAS,OAClC,SAAS,iBAAiB,KAAK,eAAe,OAC9C,SAAS,aAAa,gBACtB,SAAS,gBAAgB,KAAK,cAAc;AAE9C,QAAI,CAAC,SAAS;AACZ,aAAO,EAAE,YAAY,UAAU,QAAQ,YAAA;AAAA,IACzC;AAEA,aAAS,cAAc,KAAK;AAC5B,aAAS,qBAAqB,KAAK;AACnC,aAAS,YAAY,KAAK;AAC1B,aAAS,UAAU,KAAK;AACxB,aAAS,iBAAiB,KAAK;AAC/B,aAAS,QAAQ,KAAK,SAAS;AAC/B,aAAS,cAAc,KAAK,eAAe;AAC3C,aAAS,WAAW;AACpB,aAAS,aAAa,KAAK,cAAc;AACzC,UAAM,SAAS,KAAA;AAEf,WAAO,EAAE,YAAY,UAAU,QAAQ,UAAA;AAAA,EACzC;AACF;;;;;;;;;ACjDO,IAAM,kBAAN,cAA8B,WAAW;AAAA,EAC9C,aAAqB;AAAA,EACrB,YAA8B;AAAA,EAC9B,UAAkB;AAAA,EAClB,SAAgC,sBAAsB;AAAA,EAEtD,YAAY,UAAkC,IAAI;AAChD,UAAM,OAAO;AAEb,QAAI,QAAQ,eAAe,OAAW,MAAK,aAAa,QAAQ;AAChE,QAAI,QAAQ,cAAc,OAAW,MAAK,YAAY,QAAQ;AAC9D,QAAI,QAAQ,YAAY,OAAW,MAAK,UAAU,QAAQ;AAC1D,QAAI,QAAQ,WAAW,OAAW,MAAK,SAAS,QAAQ;AAAA,EAC1D;AAAA,EAEA,YAAqB;AACnB,WAAO,KAAK,WAAW,sBAAsB;AAAA,EAC/C;AAAA,EAEA,YAAqB;AACnB,WAAO,KAAK,WAAW,sBAAsB;AAAA,EAC/C;AAAA,EAEA,aAAsB;AACpB,WAAO,KAAK,WAAW,sBAAsB;AAAA,EAC/C;AACF;AA1Ba,kBAAN,gBAAA;AAAA,EAXN,KAAK;AAAA,IACJ,WAAW;AAAA,IACX,KAAK,EAAE,SAAS,CAAC,QAAQ,OAAO,UAAU,UAAU,QAAQ,EAAA;AAAA,IAC5D,KAAK;AAAA,MACH,SAAS,CAAC,aAAa,aAAa,YAAY;AAAA,IAAA;AAAA,IAElD,KAAK;AAAA,MACH,SAAS,CAAC,aAAa,aAAa,YAAY;AAAA,IAAA;AAAA,IAElD,iBAAiB,CAAC,eAAe,cAAc,UAAU;AAAA,EAAA,CAC1D;AAAA,GACY,eAAA;ACXN,MAAM,kCAAkC,eAAgC;AAAA,EAC7E,OAAgB,aAAa;AAAA,EAE7B,MAAM,iBAAiB,YAAgD;AACrE,WAAO,KAAK,KAAK;AAAA,MACf,OAAO,EAAE,WAAA;AAAA,IAAW,CACrB;AAAA,EACH;AAAA,EAEA,MAAM,sBACJ,YACA,WACA,SACiC;AACjC,UAAM,UAAU,MAAM,KAAK,KAAK;AAAA,MAC9B,OAAO,EAAE,YAAY,WAAW,QAAA;AAAA,MAChC,OAAO;AAAA,IAAA,CACR;AACD,WAAO,QAAQ,CAAC,KAAK;AAAA,EACvB;AAAA,EAEA,MAAM,kBAAkB,YAAqD;AAC3E,WAAO,KAAK;AAAA,MACV;AAAA,MACA;AAAA,MACA;AAAA,IAAA;AAAA,EAEJ;AAAA,EAEA,MAAM,kBACJ,YACA,UACiC;AACjC,WAAO,KAAK,sBAAsB,YAAY,UAAU,QAAQ;AAAA,EAClE;AAAA,EAEA,MAAM,eACJ,YACA,WACA,UACuC;AACvC,UAAM,6BAAa,IAAA;AACnB,QAAI,SAAS,WAAW,GAAG;AACzB,aAAO;AAAA,IACT;AAEA,UAAM,YAAY,MAAM,KAAK,KAAK;AAAA,MAChC,OAAO,EAAE,YAAY,WAAW,SAAS,SAAA;AAAA,IAAS,CACnD;AAED,eAAW,YAAY,WAAW;AAChC,aAAO,IAAI,SAAS,SAAS,QAAQ;AAAA,IACvC;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,YACJ,YACA,WACA,SACA,QAC0B;AAC1B,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B;AAAA,MACA;AAAA,MACA;AAAA,IAAA;AAGF,QAAI,UAAU;AACZ,eAAS,SAAS;AAClB,YAAM,SAAS,KAAA;AACf,aAAO;AAAA,IACT;AAEA,UAAM,UAAU,MAAM,KAAK,OAAO;AAAA,MAChC;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IAAA,CACD;AACD,UAAM,QAAQ,KAAA;AACd,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,kBACJ,YACA,QAC0B;AAC1B,WAAO,KAAK;AAAA,MACV;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IAAA;AAAA,EAEJ;AAAA,EAEA,MAAM,kBACJ,YACA,UACA,QAC0B;AAC1B,WAAO,KAAK,YAAY,YAAY,UAAU,UAAU,MAAM;AAAA,EAChE;AAAA,EAEA,MAAM,eACJ,YACA,WACA,SACkB;AAClB,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B;AAAA,MACA;AAAA,MACA;AAAA,IAAA;AAGF,QAAI,CAAC,UAAU;AACb,aAAO;AAAA,IACT;AAEA,UAAM,SAAS,OAAA;AACf,WAAO;AAAA,EACT;AACF;AClHO,MAAM,gBAAgB;AAAA,EACV;AAAA,EACA;AAAA,EACT;AAAA,EACA;AAAA,EACA,wBAA8C;AAAA,EAC9C,yBACN;AAAA,EAEF,YACE,UAA4B,IAC5B,kBAA0C,CAAA,GAC1C;AACA,SAAK,UAAU;AACf,SAAK,kBAAkB;AAAA,EACzB;AAAA,EAEA,MAAM,UACJ,YACA,UAAoC,IAClB;AAClB,UAAM,KAAK,kBAAA;AAEX,UAAM,YAAY,MAAM,KAAK,iBAAiB,UAAU;AACxD,UAAM,iBACJ,MAAM,KAAK,iBAAiB,kBAAkB,UAAU;AAC1D,UAAM,cAAc,KAAK,cAAc,WAAW,gBAAgB,MAAM;AAExE,QAAI,CAAC,QAAQ,UAAU;AACrB,aAAO;AAAA,IACT;AAEA,UAAM,kBAAkB,MAAM,KAAK,mBAAA;AACnC,QAAI,CAAC,iBAAiB;AACpB,YAAM,iBAAiB,MAAM,KAAK,iBAAiB;AAAA,QACjD;AAAA,QACA,QAAQ;AAAA,MAAA;AAEV,aAAO,KAAK,cAAc,aAAa,gBAAgB,MAAM;AAAA,IAC/D;AAEA,UAAM,QAAQ,MAAM,gBAAgB,SAAS,QAAQ,QAAQ;AAC7D,QAAI,MAAM,WAAW,GAAG;AACtB,YAAM,iBAAiB,MAAM,KAAK,iBAAiB;AAAA,QACjD;AAAA,QACA,QAAQ;AAAA,MAAA;AAEV,aAAO,KAAK,cAAc,aAAa,gBAAgB,MAAM;AAAA,IAC/D;AAEA,UAAM,YAAY,MAAM,KAAK,iBAAiB;AAAA,MAC5C;AAAA,MACA;AAAA,MACA,MAAM,IAAI,CAAC,SAAS,KAAK,EAAE;AAAA,IAAA;AAG7B,QAAI,iBAAiB;AAErB,aAAS,QAAQ,GAAG,QAAQ,MAAM,QAAQ,SAAS;AACjD,YAAM,UAAU,MAAM,KAAK;AAC3B,YAAM,WAAW,QAAQ,IAAI,MAAM,QAAQ,CAAC,IAAI;AAChD,YAAM,gBACJ,UAAU,KACT,CAAC,CAAC,UAAU,sBAAsB,QAAQ;AAE7C,YAAM,WAAW,gBAAgB,iBAAiB;AAClD,YAAM,WAAW,UAAU,IAAI,QAAQ,EAAE;AACzC,uBAAiB,KAAK,cAAc,UAAU,UAAU,MAAM;AAAA,IAChE;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,aACJ,iBACA,SACA,UAAoC,CAAA,GAClB;AAClB,UAAM,EAAE,WAAA,IAAe,2BAA2B,iBAAiB,OAAO;AAC1E,WAAO,KAAK,UAAU,YAAY,OAAO;AAAA,EAC3C;AAAA,EAEA,MAAc,oBAAmC;AAC/C,QAAI,CAAC,KAAK,uBAAuB;AAC/B,WAAK,yBAAyB,YAAY;AACxC,aAAK,qBAAqB,MACxB,4BACA,OAAO,KAAK,OAAO;AACrB,aAAK,mBAAmB,MAAO,0BAAkC;AAAA,UAC/D,KAAK;AAAA,QAAA;AAAA,MAET,GAAA;AAAA,IACF;AAEA,UAAM,KAAK;AAAA,EACb;AAAA,EAEA,MAAc,iBAAiB,YAAsC;AACnE,UAAM,kBAAkB,6BAA6B,UAAU;AAC/D,QAAI,oBAAoB,QAAW;AACjC,aAAO;AAAA,IACT;AAEA,UAAM,aACJ,MAAM,KAAK,mBAAmB,iBAAiB,UAAU;AAC3D,QAAI,CAAC,YAAY;AACf,YAAM,IAAI;AAAA,QACR,YAAY,UAAU;AAAA,MAAA;AAAA,IAE1B;AAEA,WAAO,WAAW;AAAA,EACpB;AAAA,EAEQ,cACN,cACA,QACS;AACT,YAAQ,QAAA;AAAA,MACN,KAAK,sBAAsB;AACzB,eAAO;AAAA,MACT,KAAK,sBAAsB;AACzB,eAAO;AAAA,MACT;AACE,eAAO;AAAA,IAAA;AAAA,EAEb;AAAA,EAEA,MAAc,qBAAqE;AACjF,QAAI,CAAC,KAAK,wBAAwB;AAChC,YAAM,SACJ,KAAK,gBAAgB,yBACrB;AACF,WAAK,yBAAyB,OAAO,KAAK,OAAO;AAAA,IACnD;AAEA,WAAO,KAAK;AAAA,EACd;AACF;AAEA,eAAe,6BACb,SACgD;AAChD,MAAI;AACF,UAAM,cAAc,MAAM,sBAA0C;AAAA,MAClE,aAAa;AAAA,MACb,aAAa;AAAA,MACb,SAAS;AAAA,IAAA,CACV;AAED,UAAM,mBAAmB,MAAM,YAAY,iBAAiB,OAAO,OAAO;AAC1E,WAAO;AAAA,MACL,MAAM,SAAS,UAAgD;AAC7D,cAAM,SAAS,MAAM,iBAAiB,IAAI,EAAE,IAAI,UAAU;AAC1D,YAAI,CAAC,QAAQ;AACX,iBAAO,CAAA;AAAA,QACT;AAEA,cAAM,YAAY,MAAM,iBAAiB,qBAAqB,QAAQ;AACtE,eAAO,CAAC,GAAG,WAAW,MAAM,EAAE,IAAI,CAAC,UAAe;AAAA,UAChD,IAAI,OAAO,KAAK,EAAE;AAAA,UAClB,oBAAoB,QAAQ,KAAK,kBAAkB;AAAA,UACnD,oBAAoB,QAAQ,KAAK,kBAAkB;AAAA,QAAA,EACnD;AAAA,MACJ;AAAA,IAAA;AAAA,EAEJ,SAAS,OAAO;AACd,QAAI,yBAAyB,KAAK,GAAG;AACnC,aAAO;AAAA,IACT;AACA,UAAM;AAAA,EACR;AACF;AAEA,SAAS,yBAAyB,OAAyB;AACzD,MAAI,EAAE,iBAAiB,QAAQ;AAC7B,WAAO;AAAA,EACT;AAEA,QAAM,eACJ,MAAM,iBAAiB,QAAQ,IAAI,MAAM,MAAM,OAAO,KAAK;AAC7D,QAAM,OAAO,GAAG,MAAM,OAAO,GAAG,YAAY;AAE5C,SACE,KAAK,SAAS,2BAA2B,MACxC,KAAK,SAAS,oBAAoB,KACjC,KAAK,SAAS,gBAAgB,KAC9B,KAAK,SAAS,gBAAgB;AAEpC;AC3LO,MAAM,mBAAmB;AAAA,EACb;AAAA,EACT;AAAA,EACA,wBAA8C;AAAA,EAEtD,YAAY,UAA4B,IAAI;AAC1C,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,MAAM,gBACJ,UAAkC,IACN;AAC5B,UAAM,KAAK,kBAAA;AACX,UAAM,gBAAgB,KAAK,qBAAqB,OAAO;AACvD,UAAM,cAAc,KAAK,oCAAoC,aAAa;AAC1E,UAAM,kBACJ,KAAK,wCAAwC,aAAa;AAC5D,UAAM,iBACJ,QAAQ,QAAQ,YAAY,MAAM,KAClC,QAAQ,QAAQ,cAAc,MAAM;AACtC,UAAM,aAAa,iBAAiB,QAAS,QAAQ,cAAc;AAEnE,WAAO,KAAK,iBAAiB,aAAa,iBAAiB,UAAU;AAAA,EACvE;AAAA,EAEA,MAAM,aACJ,UACA,UAA+B,IACH;AAC5B,UAAM,KAAK,kBAAA;AACX,UAAM,cAAc,gCAAgC,QAAQ;AAC5D,UAAM,kBAAkB,mCAAmC,QAAQ;AACnE,WAAO,KAAK;AAAA,MACV;AAAA,MACA;AAAA,MACA,QAAQ,cAAc;AAAA,IAAA;AAAA,EAE1B;AAAA,EAEA,MAAc,oBAAmC;AAC/C,QAAI,CAAC,KAAK,uBAAuB;AAC/B,WAAK,yBAAyB,YAAY;AACxC,aAAK,qBAAqB,MACxB,4BACA,OAAO,KAAK,OAAO;AAAA,MACvB,GAAA;AAAA,IACF;AAEA,UAAM,KAAK;AAAA,EACb;AAAA,EAEQ,oCACN,eACyB;AACzB,UAAM,cAAuC,CAAA;AAE7C,eAAW,gBAAgB,eAAe;AACxC,kBAAY,KAAK,GAAG,qCAAqC,YAAY,CAAC;AAAA,IACxE;AAEA,WAAO;AAAA,EACT;AAAA,EAEQ,wCACN,eACa;AACb,UAAM,+BAAe,IAAA;AAErB,eAAW,gBAAgB,eAAe;AACxC,UAAI,aAAa,eAAe,QAAQ;AACtC;AAAA,MACF;AAEA,YAAM,cAAc,2BAA2B,YAAY;AAC3D,UAAI,aAAa;AACf,iBAAS,IAAI,WAAW;AAAA,MAC1B;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA,EAEQ,qBAAqB,SAA6C;AACxE,QAAI,QAAQ,YAAY,QAAQ;AAC9B,aAAO,QAAQ,WACZ,IAAI,CAAC,SAAS,eAAe,SAAS,IAAI,CAAC,EAC3C,OAAO,CAAC,UAA8C,CAAC,CAAC,KAAK;AAAA,IAClE;AAEA,QAAI,QAAQ,cAAc,QAAQ;AAChC,aAAO,QAAQ,aACZ,IAAI,CAAC,SAAS,eAAe,sBAAsB,IAAW,CAAC,EAC/D,OAAO,CAAC,UAA8C,CAAC,CAAC,KAAK;AAAA,IAClE;AAEA,UAAM,gBAAuB,CAAA;AAC7B,UAAM,2BAAW,IAAA;AAEjB,eAAW,gBAAgB,eAAe,cAAA,EAAgB,UAAU;AAClE,UAAI,KAAK,IAAI,YAAY,GAAG;AAC1B;AAAA,MACF;AACA,WAAK,IAAI,YAAY;AACrB,oBAAc,KAAK,YAAY;AAAA,IACjC;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBACZ,aACA,iBACA,YAC4B;AAC5B,UAAM,8BAAc,IAAA;AACpB,eAAW,cAAc,aAAa;AACpC,cAAQ,IAAI,WAAW,YAAY,UAAU;AAAA,IAC/C;AAEA,QAAI,UAAU;AACd,QAAI,UAAU;AACd,QAAI,YAAY;AAChB,QAAI,UAAU;AAEd,UAAM,qBAAqB,MAAM,KAAK,QAAQ,QAAQ;AACtD,UAAM,cAAc,IAAI;AAAA,MACtB,mBAAmB,IAAI,CAAC,eAAe,WAAW,UAAU;AAAA,IAAA;AAE9D,UAAM,gBAAgB,MAAM,QAAQ;AAAA,MAClC,mBAAmB;AAAA,QAAI,CAAC,eACtB,KAAK,mBAAmB,iBAAiB,UAAU;AAAA,MAAA;AAAA,IACrD;AAEF,eAAW,UAAU,eAAe;AAClC,UAAI,OAAO,WAAW,UAAW;AACjC,UAAI,OAAO,WAAW,UAAW;AACjC,UAAI,OAAO,WAAW,YAAa;AAAA,IACrC;AAEA,QAAI,YAAY;AACd,YAAM,oBAAoB,MAAM,QAAQ;AAAA,QACtC,MAAM;AAAA,UAAK;AAAA,UAAiB,CAAC,gBAC3B,KAAK,mBAAmB,kBAAkB,WAAW;AAAA,QAAA;AAAA,MACvD;AAGF,YAAM,mBAAmB,kBAAkB;AAAA,QAAQ,CAAC,aAClD,SAAS;AAAA,UACP,CAAC,eAAe,CAAC,YAAY,IAAI,WAAW,UAAU;AAAA,QAAA;AAAA,MACxD;AAGF,YAAM,QAAQ;AAAA,QACZ,iBAAiB,IAAI,CAAC,eAAe,WAAW,QAAQ;AAAA,MAAA;AAE1D,gBAAU,iBAAiB;AAAA,IAC7B;AAEA,WAAO;AAAA,MACL,OAAO,QAAQ;AAAA,MACf;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA,aAAa,MAAM,KAAK,WAAW,EAAE,KAAA;AAAA,IAAK;AAAA,EAE9C;AACF;"}