purecontext-mcp 1.0.1
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/LICENSE +21 -0
- package/README.md +207 -0
- package/dist/adapters/actix-web.d.ts +28 -0
- package/dist/adapters/actix-web.d.ts.map +1 -0
- package/dist/adapters/actix-web.js +362 -0
- package/dist/adapters/actix-web.js.map +1 -0
- package/dist/adapters/adapter-registry.d.ts +34 -0
- package/dist/adapters/adapter-registry.d.ts.map +1 -0
- package/dist/adapters/adapter-registry.js +92 -0
- package/dist/adapters/adapter-registry.js.map +1 -0
- package/dist/adapters/angular.d.ts +13 -0
- package/dist/adapters/angular.d.ts.map +1 -0
- package/dist/adapters/angular.js +348 -0
- package/dist/adapters/angular.js.map +1 -0
- package/dist/adapters/axum.d.ts +16 -0
- package/dist/adapters/axum.d.ts.map +1 -0
- package/dist/adapters/axum.js +299 -0
- package/dist/adapters/axum.js.map +1 -0
- package/dist/adapters/django-orm.d.ts +28 -0
- package/dist/adapters/django-orm.d.ts.map +1 -0
- package/dist/adapters/django-orm.js +464 -0
- package/dist/adapters/django-orm.js.map +1 -0
- package/dist/adapters/django.d.ts +16 -0
- package/dist/adapters/django.d.ts.map +1 -0
- package/dist/adapters/django.js +257 -0
- package/dist/adapters/django.js.map +1 -0
- package/dist/adapters/echo.d.ts +15 -0
- package/dist/adapters/echo.d.ts.map +1 -0
- package/dist/adapters/echo.js +142 -0
- package/dist/adapters/echo.js.map +1 -0
- package/dist/adapters/express.d.ts +15 -0
- package/dist/adapters/express.d.ts.map +1 -0
- package/dist/adapters/express.js +126 -0
- package/dist/adapters/express.js.map +1 -0
- package/dist/adapters/fastapi.d.ts +16 -0
- package/dist/adapters/fastapi.d.ts.map +1 -0
- package/dist/adapters/fastapi.js +113 -0
- package/dist/adapters/fastapi.js.map +1 -0
- package/dist/adapters/fastify.d.ts +14 -0
- package/dist/adapters/fastify.d.ts.map +1 -0
- package/dist/adapters/fastify.js +124 -0
- package/dist/adapters/fastify.js.map +1 -0
- package/dist/adapters/fiber.d.ts +15 -0
- package/dist/adapters/fiber.d.ts.map +1 -0
- package/dist/adapters/fiber.js +147 -0
- package/dist/adapters/fiber.js.map +1 -0
- package/dist/adapters/flask.d.ts +18 -0
- package/dist/adapters/flask.d.ts.map +1 -0
- package/dist/adapters/flask.js +175 -0
- package/dist/adapters/flask.js.map +1 -0
- package/dist/adapters/flutter.d.ts +19 -0
- package/dist/adapters/flutter.d.ts.map +1 -0
- package/dist/adapters/flutter.js +251 -0
- package/dist/adapters/flutter.js.map +1 -0
- package/dist/adapters/gin.d.ts +16 -0
- package/dist/adapters/gin.d.ts.map +1 -0
- package/dist/adapters/gin.js +159 -0
- package/dist/adapters/gin.js.map +1 -0
- package/dist/adapters/hibernate.d.ts +24 -0
- package/dist/adapters/hibernate.d.ts.map +1 -0
- package/dist/adapters/hibernate.js +448 -0
- package/dist/adapters/hibernate.js.map +1 -0
- package/dist/adapters/ktor.d.ts +18 -0
- package/dist/adapters/ktor.d.ts.map +1 -0
- package/dist/adapters/ktor.js +219 -0
- package/dist/adapters/ktor.js.map +1 -0
- package/dist/adapters/laravel.d.ts +19 -0
- package/dist/adapters/laravel.d.ts.map +1 -0
- package/dist/adapters/laravel.js +370 -0
- package/dist/adapters/laravel.js.map +1 -0
- package/dist/adapters/micronaut.d.ts +21 -0
- package/dist/adapters/micronaut.d.ts.map +1 -0
- package/dist/adapters/micronaut.js +435 -0
- package/dist/adapters/micronaut.js.map +1 -0
- package/dist/adapters/nestjs.d.ts +12 -0
- package/dist/adapters/nestjs.d.ts.map +1 -0
- package/dist/adapters/nestjs.js +401 -0
- package/dist/adapters/nestjs.js.map +1 -0
- package/dist/adapters/nextjs.d.ts +49 -0
- package/dist/adapters/nextjs.d.ts.map +1 -0
- package/dist/adapters/nextjs.js +412 -0
- package/dist/adapters/nextjs.js.map +1 -0
- package/dist/adapters/nuxt.d.ts +44 -0
- package/dist/adapters/nuxt.d.ts.map +1 -0
- package/dist/adapters/nuxt.js +243 -0
- package/dist/adapters/nuxt.js.map +1 -0
- package/dist/adapters/quarkus.d.ts +21 -0
- package/dist/adapters/quarkus.d.ts.map +1 -0
- package/dist/adapters/quarkus.js +395 -0
- package/dist/adapters/quarkus.js.map +1 -0
- package/dist/adapters/rails.d.ts +18 -0
- package/dist/adapters/rails.d.ts.map +1 -0
- package/dist/adapters/rails.js +363 -0
- package/dist/adapters/rails.js.map +1 -0
- package/dist/adapters/react.d.ts +18 -0
- package/dist/adapters/react.d.ts.map +1 -0
- package/dist/adapters/react.js +95 -0
- package/dist/adapters/react.js.map +1 -0
- package/dist/adapters/rocket.d.ts +19 -0
- package/dist/adapters/rocket.d.ts.map +1 -0
- package/dist/adapters/rocket.js +271 -0
- package/dist/adapters/rocket.js.map +1 -0
- package/dist/adapters/sinatra.d.ts +16 -0
- package/dist/adapters/sinatra.d.ts.map +1 -0
- package/dist/adapters/sinatra.js +114 -0
- package/dist/adapters/sinatra.js.map +1 -0
- package/dist/adapters/spring-boot.d.ts +24 -0
- package/dist/adapters/spring-boot.d.ts.map +1 -0
- package/dist/adapters/spring-boot.js +399 -0
- package/dist/adapters/spring-boot.js.map +1 -0
- package/dist/adapters/spring-kotlin.d.ts +19 -0
- package/dist/adapters/spring-kotlin.d.ts.map +1 -0
- package/dist/adapters/spring-kotlin.js +297 -0
- package/dist/adapters/spring-kotlin.js.map +1 -0
- package/dist/adapters/sqlalchemy.d.ts +28 -0
- package/dist/adapters/sqlalchemy.d.ts.map +1 -0
- package/dist/adapters/sqlalchemy.js +435 -0
- package/dist/adapters/sqlalchemy.js.map +1 -0
- package/dist/adapters/symfony.d.ts +16 -0
- package/dist/adapters/symfony.d.ts.map +1 -0
- package/dist/adapters/symfony.js +242 -0
- package/dist/adapters/symfony.js.map +1 -0
- package/dist/adapters/vapor.d.ts +21 -0
- package/dist/adapters/vapor.d.ts.map +1 -0
- package/dist/adapters/vapor.js +269 -0
- package/dist/adapters/vapor.js.map +1 -0
- package/dist/adapters/vue-preprocessor.d.ts +26 -0
- package/dist/adapters/vue-preprocessor.d.ts.map +1 -0
- package/dist/adapters/vue-preprocessor.js +82 -0
- package/dist/adapters/vue-preprocessor.js.map +1 -0
- package/dist/adapters/vue.d.ts +13 -0
- package/dist/adapters/vue.d.ts.map +1 -0
- package/dist/adapters/vue.js +134 -0
- package/dist/adapters/vue.js.map +1 -0
- package/dist/config/cli.d.ts +26 -0
- package/dist/config/cli.d.ts.map +1 -0
- package/dist/config/cli.js +291 -0
- package/dist/config/cli.js.map +1 -0
- package/dist/config/config-loader.d.ts +26 -0
- package/dist/config/config-loader.d.ts.map +1 -0
- package/dist/config/config-loader.js +168 -0
- package/dist/config/config-loader.js.map +1 -0
- package/dist/config/config-schema.d.ts +228 -0
- package/dist/config/config-schema.d.ts.map +1 -0
- package/dist/config/config-schema.js +400 -0
- package/dist/config/config-schema.js.map +1 -0
- package/dist/config/keys-cli.d.ts +37 -0
- package/dist/config/keys-cli.d.ts.map +1 -0
- package/dist/config/keys-cli.js +179 -0
- package/dist/config/keys-cli.js.map +1 -0
- package/dist/config/workspaces-cli.d.ts +2 -0
- package/dist/config/workspaces-cli.d.ts.map +1 -0
- package/dist/config/workspaces-cli.js +84 -0
- package/dist/config/workspaces-cli.js.map +1 -0
- package/dist/core/db/api-keys.d.ts +95 -0
- package/dist/core/db/api-keys.d.ts.map +1 -0
- package/dist/core/db/api-keys.js +293 -0
- package/dist/core/db/api-keys.js.map +1 -0
- package/dist/core/db/dep-store.d.ts +18 -0
- package/dist/core/db/dep-store.d.ts.map +1 -0
- package/dist/core/db/dep-store.js +155 -0
- package/dist/core/db/dep-store.js.map +1 -0
- package/dist/core/db/embedding-store.d.ts +40 -0
- package/dist/core/db/embedding-store.d.ts.map +1 -0
- package/dist/core/db/embedding-store.js +120 -0
- package/dist/core/db/embedding-store.js.map +1 -0
- package/dist/core/db/file-store.d.ts +27 -0
- package/dist/core/db/file-store.d.ts.map +1 -0
- package/dist/core/db/file-store.js +101 -0
- package/dist/core/db/file-store.js.map +1 -0
- package/dist/core/db/request-log.d.ts +40 -0
- package/dist/core/db/request-log.d.ts.map +1 -0
- package/dist/core/db/request-log.js +90 -0
- package/dist/core/db/request-log.js.map +1 -0
- package/dist/core/db/savings-store.d.ts +9 -0
- package/dist/core/db/savings-store.d.ts.map +1 -0
- package/dist/core/db/savings-store.js +64 -0
- package/dist/core/db/savings-store.js.map +1 -0
- package/dist/core/db/schema.d.ts +14 -0
- package/dist/core/db/schema.d.ts.map +1 -0
- package/dist/core/db/schema.js +235 -0
- package/dist/core/db/schema.js.map +1 -0
- package/dist/core/db/symbol-store.d.ts +64 -0
- package/dist/core/db/symbol-store.d.ts.map +1 -0
- package/dist/core/db/symbol-store.js +243 -0
- package/dist/core/db/symbol-store.js.map +1 -0
- package/dist/core/db/tenants.d.ts +91 -0
- package/dist/core/db/tenants.d.ts.map +1 -0
- package/dist/core/db/tenants.js +219 -0
- package/dist/core/db/tenants.js.map +1 -0
- package/dist/core/errors.d.ts +72 -0
- package/dist/core/errors.d.ts.map +1 -0
- package/dist/core/errors.js +140 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/core/file-discovery.d.ts +17 -0
- package/dist/core/file-discovery.d.ts.map +1 -0
- package/dist/core/file-discovery.js +136 -0
- package/dist/core/file-discovery.js.map +1 -0
- package/dist/core/file-processor.d.ts +21 -0
- package/dist/core/file-processor.d.ts.map +1 -0
- package/dist/core/file-processor.js +110 -0
- package/dist/core/file-processor.js.map +1 -0
- package/dist/core/hash-cache.d.ts +18 -0
- package/dist/core/hash-cache.d.ts.map +1 -0
- package/dist/core/hash-cache.js +35 -0
- package/dist/core/hash-cache.js.map +1 -0
- package/dist/core/index-manager.d.ts +18 -0
- package/dist/core/index-manager.d.ts.map +1 -0
- package/dist/core/index-manager.js +461 -0
- package/dist/core/index-manager.js.map +1 -0
- package/dist/core/indexing-worker.d.ts +60 -0
- package/dist/core/indexing-worker.d.ts.map +1 -0
- package/dist/core/indexing-worker.js +128 -0
- package/dist/core/indexing-worker.js.map +1 -0
- package/dist/core/logger.d.ts +10 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +40 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/parse-dispatcher.d.ts +18 -0
- package/dist/core/parse-dispatcher.d.ts.map +1 -0
- package/dist/core/parse-dispatcher.js +74 -0
- package/dist/core/parse-dispatcher.js.map +1 -0
- package/dist/core/search/query-preprocessor.d.ts +22 -0
- package/dist/core/search/query-preprocessor.d.ts.map +1 -0
- package/dist/core/search/query-preprocessor.js +77 -0
- package/dist/core/search/query-preprocessor.js.map +1 -0
- package/dist/core/search/relevance-ranker.d.ts +34 -0
- package/dist/core/search/relevance-ranker.d.ts.map +1 -0
- package/dist/core/search/relevance-ranker.js +132 -0
- package/dist/core/search/relevance-ranker.js.map +1 -0
- package/dist/core/security.d.ts +40 -0
- package/dist/core/security.d.ts.map +1 -0
- package/dist/core/security.js +133 -0
- package/dist/core/security.js.map +1 -0
- package/dist/core/telemetry.d.ts +19 -0
- package/dist/core/telemetry.d.ts.map +1 -0
- package/dist/core/telemetry.js +101 -0
- package/dist/core/telemetry.js.map +1 -0
- package/dist/core/tenant-context.d.ts +22 -0
- package/dist/core/tenant-context.d.ts.map +1 -0
- package/dist/core/tenant-context.js +16 -0
- package/dist/core/tenant-context.js.map +1 -0
- package/dist/core/token-tracker.d.ts +39 -0
- package/dist/core/token-tracker.d.ts.map +1 -0
- package/dist/core/token-tracker.js +153 -0
- package/dist/core/token-tracker.js.map +1 -0
- package/dist/core/types.d.ts +145 -0
- package/dist/core/types.d.ts.map +1 -0
- package/dist/core/types.js +2 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/watcher/file-watcher.d.ts +18 -0
- package/dist/core/watcher/file-watcher.d.ts.map +1 -0
- package/dist/core/watcher/file-watcher.js +123 -0
- package/dist/core/watcher/file-watcher.js.map +1 -0
- package/dist/core/worker-pool.d.ts +24 -0
- package/dist/core/worker-pool.d.ts.map +1 -0
- package/dist/core/worker-pool.js +92 -0
- package/dist/core/worker-pool.js.map +1 -0
- package/dist/graph/graph-builder.d.ts +16 -0
- package/dist/graph/graph-builder.d.ts.map +1 -0
- package/dist/graph/graph-builder.js +51 -0
- package/dist/graph/graph-builder.js.map +1 -0
- package/dist/graph/graph-traversal.d.ts +57 -0
- package/dist/graph/graph-traversal.d.ts.map +1 -0
- package/dist/graph/graph-traversal.js +144 -0
- package/dist/graph/graph-traversal.js.map +1 -0
- package/dist/graph/path-resolver.d.ts +13 -0
- package/dist/graph/path-resolver.d.ts.map +1 -0
- package/dist/graph/path-resolver.js +161 -0
- package/dist/graph/path-resolver.js.map +1 -0
- package/dist/handlers/c.d.ts +3 -0
- package/dist/handlers/c.d.ts.map +1 -0
- package/dist/handlers/c.js +473 -0
- package/dist/handlers/c.js.map +1 -0
- package/dist/handlers/cpp.d.ts +3 -0
- package/dist/handlers/cpp.d.ts.map +1 -0
- package/dist/handlers/cpp.js +682 -0
- package/dist/handlers/cpp.js.map +1 -0
- package/dist/handlers/csharp.d.ts +3 -0
- package/dist/handlers/csharp.d.ts.map +1 -0
- package/dist/handlers/csharp.js +369 -0
- package/dist/handlers/csharp.js.map +1 -0
- package/dist/handlers/dart.d.ts +3 -0
- package/dist/handlers/dart.d.ts.map +1 -0
- package/dist/handlers/dart.js +596 -0
- package/dist/handlers/dart.js.map +1 -0
- package/dist/handlers/elixir.d.ts +8 -0
- package/dist/handlers/elixir.d.ts.map +1 -0
- package/dist/handlers/elixir.js +412 -0
- package/dist/handlers/elixir.js.map +1 -0
- package/dist/handlers/go.d.ts +3 -0
- package/dist/handlers/go.d.ts.map +1 -0
- package/dist/handlers/go.js +262 -0
- package/dist/handlers/go.js.map +1 -0
- package/dist/handlers/handler-registry.d.ts +20 -0
- package/dist/handlers/handler-registry.d.ts.map +1 -0
- package/dist/handlers/handler-registry.js +69 -0
- package/dist/handlers/handler-registry.js.map +1 -0
- package/dist/handlers/haskell.d.ts +17 -0
- package/dist/handlers/haskell.d.ts.map +1 -0
- package/dist/handlers/haskell.js +356 -0
- package/dist/handlers/haskell.js.map +1 -0
- package/dist/handlers/java.d.ts +3 -0
- package/dist/handlers/java.d.ts.map +1 -0
- package/dist/handlers/java.js +350 -0
- package/dist/handlers/java.js.map +1 -0
- package/dist/handlers/javascript.d.ts +3 -0
- package/dist/handlers/javascript.d.ts.map +1 -0
- package/dist/handlers/javascript.js +232 -0
- package/dist/handlers/javascript.js.map +1 -0
- package/dist/handlers/kotlin.d.ts +3 -0
- package/dist/handlers/kotlin.d.ts.map +1 -0
- package/dist/handlers/kotlin.js +317 -0
- package/dist/handlers/kotlin.js.map +1 -0
- package/dist/handlers/lua.d.ts +3 -0
- package/dist/handlers/lua.d.ts.map +1 -0
- package/dist/handlers/lua.js +328 -0
- package/dist/handlers/lua.js.map +1 -0
- package/dist/handlers/php.d.ts +3 -0
- package/dist/handlers/php.d.ts.map +1 -0
- package/dist/handlers/php.js +343 -0
- package/dist/handlers/php.js.map +1 -0
- package/dist/handlers/python.d.ts +3 -0
- package/dist/handlers/python.d.ts.map +1 -0
- package/dist/handlers/python.js +343 -0
- package/dist/handlers/python.js.map +1 -0
- package/dist/handlers/r.d.ts +9 -0
- package/dist/handlers/r.d.ts.map +1 -0
- package/dist/handlers/r.js +452 -0
- package/dist/handlers/r.js.map +1 -0
- package/dist/handlers/ruby.d.ts +8 -0
- package/dist/handlers/ruby.d.ts.map +1 -0
- package/dist/handlers/ruby.js +271 -0
- package/dist/handlers/ruby.js.map +1 -0
- package/dist/handlers/rust.d.ts +3 -0
- package/dist/handlers/rust.d.ts.map +1 -0
- package/dist/handlers/rust.js +420 -0
- package/dist/handlers/rust.js.map +1 -0
- package/dist/handlers/scala.d.ts +9 -0
- package/dist/handlers/scala.d.ts.map +1 -0
- package/dist/handlers/scala.js +480 -0
- package/dist/handlers/scala.js.map +1 -0
- package/dist/handlers/swift.d.ts +3 -0
- package/dist/handlers/swift.d.ts.map +1 -0
- package/dist/handlers/swift.js +553 -0
- package/dist/handlers/swift.js.map +1 -0
- package/dist/handlers/typescript.d.ts +4 -0
- package/dist/handlers/typescript.d.ts.map +1 -0
- package/dist/handlers/typescript.js +325 -0
- package/dist/handlers/typescript.js.map +1 -0
- package/dist/index.d.ts +44 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +266 -0
- package/dist/index.js.map +1 -0
- package/dist/semantic/embedding-provider.d.ts +66 -0
- package/dist/semantic/embedding-provider.d.ts.map +1 -0
- package/dist/semantic/embedding-provider.js +176 -0
- package/dist/semantic/embedding-provider.js.map +1 -0
- package/dist/semantic/hnsw-index.d.ts +96 -0
- package/dist/semantic/hnsw-index.d.ts.map +1 -0
- package/dist/semantic/hnsw-index.js +364 -0
- package/dist/semantic/hnsw-index.js.map +1 -0
- package/dist/semantic/hybrid-search.d.ts +46 -0
- package/dist/semantic/hybrid-search.d.ts.map +1 -0
- package/dist/semantic/hybrid-search.js +164 -0
- package/dist/semantic/hybrid-search.js.map +1 -0
- package/dist/semantic/query-expansion.d.ts +25 -0
- package/dist/semantic/query-expansion.d.ts.map +1 -0
- package/dist/semantic/query-expansion.js +150 -0
- package/dist/semantic/query-expansion.js.map +1 -0
- package/dist/semantic/semantic-indexer.d.ts +64 -0
- package/dist/semantic/semantic-indexer.d.ts.map +1 -0
- package/dist/semantic/semantic-indexer.js +162 -0
- package/dist/semantic/semantic-indexer.js.map +1 -0
- package/dist/semantic/text-preparation.d.ts +23 -0
- package/dist/semantic/text-preparation.d.ts.map +1 -0
- package/dist/semantic/text-preparation.js +56 -0
- package/dist/semantic/text-preparation.js.map +1 -0
- package/dist/semantic/vector-store.d.ts +81 -0
- package/dist/semantic/vector-store.d.ts.map +1 -0
- package/dist/semantic/vector-store.js +157 -0
- package/dist/semantic/vector-store.js.map +1 -0
- package/dist/server/admin-api.d.ts +88 -0
- package/dist/server/admin-api.d.ts.map +1 -0
- package/dist/server/admin-api.js +661 -0
- package/dist/server/admin-api.js.map +1 -0
- package/dist/server/auth/api-key.d.ts +105 -0
- package/dist/server/auth/api-key.d.ts.map +1 -0
- package/dist/server/auth/api-key.js +276 -0
- package/dist/server/auth/api-key.js.map +1 -0
- package/dist/server/http-server.d.ts +100 -0
- package/dist/server/http-server.d.ts.map +1 -0
- package/dist/server/http-server.js +823 -0
- package/dist/server/http-server.js.map +1 -0
- package/dist/server/mcp-server.d.ts +23 -0
- package/dist/server/mcp-server.d.ts.map +1 -0
- package/dist/server/mcp-server.js +214 -0
- package/dist/server/mcp-server.js.map +1 -0
- package/dist/server/rate-limit-store.d.ts +25 -0
- package/dist/server/rate-limit-store.d.ts.map +1 -0
- package/dist/server/rate-limit-store.js +46 -0
- package/dist/server/rate-limit-store.js.map +1 -0
- package/dist/server/rate-limiter.d.ts +59 -0
- package/dist/server/rate-limiter.d.ts.map +1 -0
- package/dist/server/rate-limiter.js +85 -0
- package/dist/server/rate-limiter.js.map +1 -0
- package/dist/server/tools/_meta.d.ts +24 -0
- package/dist/server/tools/_meta.d.ts.map +1 -0
- package/dist/server/tools/_meta.js +35 -0
- package/dist/server/tools/_meta.js.map +1 -0
- package/dist/server/tools/find-dead-code.d.ts +11 -0
- package/dist/server/tools/find-dead-code.d.ts.map +1 -0
- package/dist/server/tools/find-dead-code.js +45 -0
- package/dist/server/tools/find-dead-code.js.map +1 -0
- package/dist/server/tools/find-importers.d.ts +13 -0
- package/dist/server/tools/find-importers.d.ts.map +1 -0
- package/dist/server/tools/find-importers.js +46 -0
- package/dist/server/tools/find-importers.js.map +1 -0
- package/dist/server/tools/get-blast-radius.d.ts +15 -0
- package/dist/server/tools/get-blast-radius.d.ts.map +1 -0
- package/dist/server/tools/get-blast-radius.js +51 -0
- package/dist/server/tools/get-blast-radius.js.map +1 -0
- package/dist/server/tools/get-context-bundle.d.ts +15 -0
- package/dist/server/tools/get-context-bundle.d.ts.map +1 -0
- package/dist/server/tools/get-context-bundle.js +53 -0
- package/dist/server/tools/get-context-bundle.js.map +1 -0
- package/dist/server/tools/get-file-outline.d.ts +13 -0
- package/dist/server/tools/get-file-outline.d.ts.map +1 -0
- package/dist/server/tools/get-file-outline.js +42 -0
- package/dist/server/tools/get-file-outline.js.map +1 -0
- package/dist/server/tools/get-file-tree.d.ts +11 -0
- package/dist/server/tools/get-file-tree.d.ts.map +1 -0
- package/dist/server/tools/get-file-tree.js +63 -0
- package/dist/server/tools/get-file-tree.js.map +1 -0
- package/dist/server/tools/get-graph.d.ts +46 -0
- package/dist/server/tools/get-graph.d.ts.map +1 -0
- package/dist/server/tools/get-graph.js +168 -0
- package/dist/server/tools/get-graph.js.map +1 -0
- package/dist/server/tools/get-layer-violations.d.ts +37 -0
- package/dist/server/tools/get-layer-violations.d.ts.map +1 -0
- package/dist/server/tools/get-layer-violations.js +138 -0
- package/dist/server/tools/get-layer-violations.js.map +1 -0
- package/dist/server/tools/get-repo-outline.d.ts +13 -0
- package/dist/server/tools/get-repo-outline.d.ts.map +1 -0
- package/dist/server/tools/get-repo-outline.js +72 -0
- package/dist/server/tools/get-repo-outline.js.map +1 -0
- package/dist/server/tools/get-savings-stats.d.ts +11 -0
- package/dist/server/tools/get-savings-stats.d.ts.map +1 -0
- package/dist/server/tools/get-savings-stats.js +56 -0
- package/dist/server/tools/get-savings-stats.js.map +1 -0
- package/dist/server/tools/get-symbol-source.d.ts +13 -0
- package/dist/server/tools/get-symbol-source.d.ts.map +1 -0
- package/dist/server/tools/get-symbol-source.js +93 -0
- package/dist/server/tools/get-symbol-source.js.map +1 -0
- package/dist/server/tools/index-folder.d.ts +15 -0
- package/dist/server/tools/index-folder.d.ts.map +1 -0
- package/dist/server/tools/index-folder.js +45 -0
- package/dist/server/tools/index-folder.js.map +1 -0
- package/dist/server/tools/index-repo.d.ts +26 -0
- package/dist/server/tools/index-repo.d.ts.map +1 -0
- package/dist/server/tools/index-repo.js +164 -0
- package/dist/server/tools/index-repo.js.map +1 -0
- package/dist/server/tools/list-repos.d.ts +11 -0
- package/dist/server/tools/list-repos.d.ts.map +1 -0
- package/dist/server/tools/list-repos.js +60 -0
- package/dist/server/tools/list-repos.js.map +1 -0
- package/dist/server/tools/resolve-repo.d.ts +11 -0
- package/dist/server/tools/resolve-repo.d.ts.map +1 -0
- package/dist/server/tools/resolve-repo.js +44 -0
- package/dist/server/tools/resolve-repo.js.map +1 -0
- package/dist/server/tools/search-semantic.d.ts +56 -0
- package/dist/server/tools/search-semantic.d.ts.map +1 -0
- package/dist/server/tools/search-semantic.js +187 -0
- package/dist/server/tools/search-semantic.js.map +1 -0
- package/dist/server/tools/search-symbols.d.ts +43 -0
- package/dist/server/tools/search-symbols.d.ts.map +1 -0
- package/dist/server/tools/search-symbols.js +197 -0
- package/dist/server/tools/search-symbols.js.map +1 -0
- package/dist/server/tools/search-text.d.ts +21 -0
- package/dist/server/tools/search-text.d.ts.map +1 -0
- package/dist/server/tools/search-text.js +131 -0
- package/dist/server/tools/search-text.js.map +1 -0
- package/dist/server/transport.d.ts +59 -0
- package/dist/server/transport.d.ts.map +1 -0
- package/dist/server/transport.js +92 -0
- package/dist/server/transport.js.map +1 -0
- package/dist/summarizer/ai-summarizer.d.ts +71 -0
- package/dist/summarizer/ai-summarizer.d.ts.map +1 -0
- package/dist/summarizer/ai-summarizer.js +228 -0
- package/dist/summarizer/ai-summarizer.js.map +1 -0
- package/dist/summarizer/docstring-extractor.d.ts +9 -0
- package/dist/summarizer/docstring-extractor.d.ts.map +1 -0
- package/dist/summarizer/docstring-extractor.js +89 -0
- package/dist/summarizer/docstring-extractor.js.map +1 -0
- package/dist/summarizer/embeddings.d.ts +29 -0
- package/dist/summarizer/embeddings.d.ts.map +1 -0
- package/dist/summarizer/embeddings.js +133 -0
- package/dist/summarizer/embeddings.js.map +1 -0
- package/dist/summarizer/summarizer.d.ts +22 -0
- package/dist/summarizer/summarizer.d.ts.map +1 -0
- package/dist/summarizer/summarizer.js +86 -0
- package/dist/summarizer/summarizer.js.map +1 -0
- package/dist/ui/assets/BlastRadius-ccQ3ktbv.js +1 -0
- package/dist/ui/assets/DependencyGraph-BZV40eAE.css +1 -0
- package/dist/ui/assets/DependencyGraph-Ca84q89b.js +27 -0
- package/dist/ui/assets/NotFound-C1wVKP-6.js +1 -0
- package/dist/ui/assets/RepoDetail-B_8qid2Y.js +1 -0
- package/dist/ui/assets/RepoList-BuPu-ODM.js +1 -0
- package/dist/ui/assets/Search-BzZ9Og8E.js +1 -0
- package/dist/ui/assets/SymbolView-FdpKlzql.js +14 -0
- package/dist/ui/assets/client-BANW_tIp.js +1 -0
- package/dist/ui/assets/index-CnwwQi_S.js +61 -0
- package/dist/ui/assets/index-SS2IXr8I.css +1 -0
- package/dist/ui/assets/repoStore-SbGR6YI6.js +1 -0
- package/dist/ui/assets/useSearch-DHyve22h.js +1 -0
- package/dist/ui/index.html +13 -0
- package/dist/version.d.ts +2 -0
- package/dist/version.d.ts.map +1 -0
- package/dist/version.js +2 -0
- package/dist/version.js.map +1 -0
- package/grammars/tree-sitter-c.wasm +0 -0
- package/grammars/tree-sitter-cpp.wasm +0 -0
- package/grammars/tree-sitter-csharp.wasm +0 -0
- package/grammars/tree-sitter-dart.wasm +0 -0
- package/grammars/tree-sitter-elixir.wasm +0 -0
- package/grammars/tree-sitter-go.wasm +0 -0
- package/grammars/tree-sitter-haskell.wasm +0 -0
- package/grammars/tree-sitter-java.wasm +0 -0
- package/grammars/tree-sitter-javascript.wasm +0 -0
- package/grammars/tree-sitter-kotlin.wasm +0 -0
- package/grammars/tree-sitter-lua.wasm +0 -0
- package/grammars/tree-sitter-php.wasm +0 -0
- package/grammars/tree-sitter-python.wasm +0 -0
- package/grammars/tree-sitter-r.wasm +0 -0
- package/grammars/tree-sitter-ruby.wasm +0 -0
- package/grammars/tree-sitter-rust.wasm +0 -0
- package/grammars/tree-sitter-scala.wasm +0 -0
- package/grammars/tree-sitter-swift.wasm +0 -0
- package/grammars/tree-sitter-tsx.wasm +0 -0
- package/grammars/tree-sitter-typescript.wasm +0 -0
- package/package.json +104 -0
- package/scripts/check-sqlite.js +41 -0
|
@@ -0,0 +1,661 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AdminApi — REST endpoints for managing tenants, API keys, and monitoring.
|
|
3
|
+
*
|
|
4
|
+
* All /admin/* routes require a valid API key with `admin` permission.
|
|
5
|
+
* This is enforced by the HTTP server before delegating to AdminApi.handle().
|
|
6
|
+
*
|
|
7
|
+
* Routes:
|
|
8
|
+
* POST /admin/tenants — create tenant
|
|
9
|
+
* GET /admin/tenants — list tenants
|
|
10
|
+
* GET /admin/tenants/:id — get tenant
|
|
11
|
+
* PATCH /admin/tenants/:id — update tenant
|
|
12
|
+
* DELETE /admin/tenants/:id — delete tenant + data
|
|
13
|
+
* POST /admin/tenants/:id/keys — create API key
|
|
14
|
+
* GET /admin/tenants/:id/keys — list API keys
|
|
15
|
+
* DELETE /admin/keys/:prefix — revoke API key
|
|
16
|
+
* GET /admin/stats — global stats
|
|
17
|
+
* GET /admin/tenants/:id/stats — per-tenant stats
|
|
18
|
+
*/
|
|
19
|
+
import { readdirSync, existsSync, unlinkSync } from 'node:fs';
|
|
20
|
+
import { join } from 'node:path';
|
|
21
|
+
import Database from 'better-sqlite3';
|
|
22
|
+
import { ApiKeyValidator } from './auth/api-key.js';
|
|
23
|
+
import { TenantStore } from '../core/db/tenants.js';
|
|
24
|
+
import { ApiKeyStore } from '../core/db/api-keys.js';
|
|
25
|
+
import { RequestLogStore } from '../core/db/request-log.js';
|
|
26
|
+
import { getIndexDir } from '../core/db/schema.js';
|
|
27
|
+
import { logger } from '../core/logger.js';
|
|
28
|
+
// ─── Constants ────────────────────────────────────────────────────────────────
|
|
29
|
+
const MAX_ADMIN_BODY_BYTES = 65_536; // 64 KB — admin bodies are small
|
|
30
|
+
function matchRoute(method, path) {
|
|
31
|
+
if (method === 'GET' && path === '/admin/stats') {
|
|
32
|
+
return { route: 'globalStats', params: {} };
|
|
33
|
+
}
|
|
34
|
+
if (method === 'POST' && path === '/admin/tenants') {
|
|
35
|
+
return { route: 'createTenant', params: {} };
|
|
36
|
+
}
|
|
37
|
+
if (method === 'GET' && path === '/admin/tenants') {
|
|
38
|
+
return { route: 'listTenants', params: {} };
|
|
39
|
+
}
|
|
40
|
+
// /admin/tenants/:id/keys
|
|
41
|
+
const keysMatch = /^\/admin\/tenants\/([^/]+)\/keys$/.exec(path);
|
|
42
|
+
if (keysMatch) {
|
|
43
|
+
const id = keysMatch[1];
|
|
44
|
+
if (method === 'POST')
|
|
45
|
+
return { route: 'createKey', params: { id } };
|
|
46
|
+
if (method === 'GET')
|
|
47
|
+
return { route: 'listKeys', params: { id } };
|
|
48
|
+
}
|
|
49
|
+
// /admin/tenants/:id/stats
|
|
50
|
+
const tenantStatsMatch = /^\/admin\/tenants\/([^/]+)\/stats$/.exec(path);
|
|
51
|
+
if (tenantStatsMatch && method === 'GET') {
|
|
52
|
+
return { route: 'tenantStats', params: { id: tenantStatsMatch[1] } };
|
|
53
|
+
}
|
|
54
|
+
// /admin/tenants/:id
|
|
55
|
+
const tenantMatch = /^\/admin\/tenants\/([^/]+)$/.exec(path);
|
|
56
|
+
if (tenantMatch) {
|
|
57
|
+
const id = tenantMatch[1];
|
|
58
|
+
if (method === 'GET')
|
|
59
|
+
return { route: 'getTenant', params: { id } };
|
|
60
|
+
if (method === 'PATCH')
|
|
61
|
+
return { route: 'updateTenant', params: { id } };
|
|
62
|
+
if (method === 'DELETE')
|
|
63
|
+
return { route: 'deleteTenant', params: { id } };
|
|
64
|
+
}
|
|
65
|
+
// Flat /admin/keys routes (Phase 18)
|
|
66
|
+
if (method === 'POST' && path === '/admin/keys') {
|
|
67
|
+
return { route: 'createKeyFlat', params: {} };
|
|
68
|
+
}
|
|
69
|
+
if (method === 'GET' && path === '/admin/keys') {
|
|
70
|
+
return { route: 'listKeysFlat', params: {} };
|
|
71
|
+
}
|
|
72
|
+
// /admin/keys/:id (Phase 18 flat routes + legacy revoke)
|
|
73
|
+
const keyMatch = /^\/admin\/keys\/([^/]+)$/.exec(path);
|
|
74
|
+
if (keyMatch) {
|
|
75
|
+
const id = keyMatch[1];
|
|
76
|
+
if (method === 'GET')
|
|
77
|
+
return { route: 'keyUsage', params: { id } };
|
|
78
|
+
if (method === 'DELETE')
|
|
79
|
+
return { route: 'revokeKey', params: { prefix: id } };
|
|
80
|
+
}
|
|
81
|
+
// /admin/keys/:id/usage (explicit usage route)
|
|
82
|
+
const keyUsageMatch = /^\/admin\/keys\/([^/]+)\/usage$/.exec(path);
|
|
83
|
+
if (keyUsageMatch && method === 'GET') {
|
|
84
|
+
return { route: 'keyUsage', params: { id: keyUsageMatch[1] } };
|
|
85
|
+
}
|
|
86
|
+
// /admin/workspaces routes
|
|
87
|
+
if (method === 'POST' && path === '/admin/workspaces') {
|
|
88
|
+
return { route: 'createWorkspace', params: {} };
|
|
89
|
+
}
|
|
90
|
+
if (method === 'GET' && path === '/admin/workspaces') {
|
|
91
|
+
return { route: 'listWorkspaces', params: {} };
|
|
92
|
+
}
|
|
93
|
+
const wsMatch = /^\/admin\/workspaces\/([^/]+)$/.exec(path);
|
|
94
|
+
if (wsMatch) {
|
|
95
|
+
const id = wsMatch[1];
|
|
96
|
+
if (method === 'GET')
|
|
97
|
+
return { route: 'getWorkspace', params: { id } };
|
|
98
|
+
if (method === 'PATCH')
|
|
99
|
+
return { route: 'updateWorkspace', params: { id } };
|
|
100
|
+
}
|
|
101
|
+
const wsMembersMatch = /^\/admin\/workspaces\/([^/]+)\/members$/.exec(path);
|
|
102
|
+
if (wsMembersMatch) {
|
|
103
|
+
const id = wsMembersMatch[1];
|
|
104
|
+
if (method === 'POST')
|
|
105
|
+
return { route: 'addWorkspaceMember', params: { id } };
|
|
106
|
+
if (method === 'GET')
|
|
107
|
+
return { route: 'listWorkspaceMembers', params: { id } };
|
|
108
|
+
}
|
|
109
|
+
const wsMemberMatch = /^\/admin\/workspaces\/([^/]+)\/members\/([^/]+)$/.exec(path);
|
|
110
|
+
if (wsMemberMatch && method === 'DELETE') {
|
|
111
|
+
return { route: 'removeWorkspaceMember', params: { id: wsMemberMatch[1], userId: wsMemberMatch[2] } };
|
|
112
|
+
}
|
|
113
|
+
const wsUsageMatch = /^\/admin\/workspaces\/([^/]+)\/usage$/.exec(path);
|
|
114
|
+
if (wsUsageMatch && method === 'GET') {
|
|
115
|
+
return { route: 'workspaceUsage', params: { id: wsUsageMatch[1] } };
|
|
116
|
+
}
|
|
117
|
+
return null;
|
|
118
|
+
}
|
|
119
|
+
// ─── Body reader ──────────────────────────────────────────────────────────────
|
|
120
|
+
function readAdminBody(req) {
|
|
121
|
+
return new Promise((resolve, reject) => {
|
|
122
|
+
const chunks = [];
|
|
123
|
+
let total = 0;
|
|
124
|
+
req.on('data', (chunk) => {
|
|
125
|
+
total += chunk.length;
|
|
126
|
+
if (total > MAX_ADMIN_BODY_BYTES) {
|
|
127
|
+
req.destroy(new Error('Request body too large'));
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
chunks.push(chunk);
|
|
131
|
+
});
|
|
132
|
+
req.on('end', () => resolve(Buffer.concat(chunks)));
|
|
133
|
+
req.on('error', reject);
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
async function parseJsonBody(req) {
|
|
137
|
+
try {
|
|
138
|
+
const raw = await readAdminBody(req);
|
|
139
|
+
if (raw.length === 0)
|
|
140
|
+
return {};
|
|
141
|
+
return JSON.parse(raw.toString('utf8'));
|
|
142
|
+
}
|
|
143
|
+
catch {
|
|
144
|
+
return null;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
// ─── Response helpers ─────────────────────────────────────────────────────────
|
|
148
|
+
function jsonResponse(res, status, body) {
|
|
149
|
+
const payload = JSON.stringify(body);
|
|
150
|
+
res.writeHead(status, {
|
|
151
|
+
'Content-Type': 'application/json',
|
|
152
|
+
'Content-Length': Buffer.byteLength(payload),
|
|
153
|
+
});
|
|
154
|
+
res.end(payload);
|
|
155
|
+
}
|
|
156
|
+
// ─── Serialisers ──────────────────────────────────────────────────────────────
|
|
157
|
+
function tenantToJson(t) {
|
|
158
|
+
return {
|
|
159
|
+
id: t.id,
|
|
160
|
+
name: t.name,
|
|
161
|
+
created_at: t.createdAt,
|
|
162
|
+
settings: t.settings,
|
|
163
|
+
storage_quota_bytes: t.storageQuotaBytes,
|
|
164
|
+
storage_used_bytes: t.storageUsedBytes,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
function workspaceToJson(t) {
|
|
168
|
+
return {
|
|
169
|
+
id: t.id,
|
|
170
|
+
name: t.name,
|
|
171
|
+
created_at: t.createdAt,
|
|
172
|
+
plan: t.plan,
|
|
173
|
+
member_count: t.memberCount ?? 0,
|
|
174
|
+
settings: t.settings,
|
|
175
|
+
storage_quota_bytes: t.storageQuotaBytes,
|
|
176
|
+
storage_used_bytes: t.storageUsedBytes,
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
function memberToJson(m) {
|
|
180
|
+
return {
|
|
181
|
+
workspace_id: m.workspaceId,
|
|
182
|
+
user_id: m.userId,
|
|
183
|
+
role: m.role,
|
|
184
|
+
added_at: m.addedAt,
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
function keyToJson(k) {
|
|
188
|
+
return {
|
|
189
|
+
key_hash_prefix: k.keyHash.slice(0, 8),
|
|
190
|
+
tenant_id: k.tenantId,
|
|
191
|
+
permissions: k.permissions,
|
|
192
|
+
rate_limit_tier: k.rateLimitTier,
|
|
193
|
+
created_at: k.createdAt,
|
|
194
|
+
last_used_at: k.lastUsedAt,
|
|
195
|
+
revoked: k.revokedAt !== null,
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
// ─── AdminApi ─────────────────────────────────────────────────────────────────
|
|
199
|
+
export class AdminApi {
|
|
200
|
+
tenantStore;
|
|
201
|
+
keyStore;
|
|
202
|
+
validator;
|
|
203
|
+
logStore;
|
|
204
|
+
startTime;
|
|
205
|
+
indexDir;
|
|
206
|
+
constructor(opts) {
|
|
207
|
+
this.tenantStore = new TenantStore(opts.db);
|
|
208
|
+
this.keyStore = new ApiKeyStore(opts.db);
|
|
209
|
+
this.validator = new ApiKeyValidator(opts.db);
|
|
210
|
+
this.logStore = new RequestLogStore(opts.db);
|
|
211
|
+
this.startTime = opts.startTime;
|
|
212
|
+
this.indexDir = opts.indexDir ?? getIndexDir();
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Log a request to the request_log table.
|
|
216
|
+
* Called by the HTTP server after each MCP tool call.
|
|
217
|
+
*/
|
|
218
|
+
logRequest(entry) {
|
|
219
|
+
this.logStore.log(entry);
|
|
220
|
+
}
|
|
221
|
+
/**
|
|
222
|
+
* Handle an /admin/* request.
|
|
223
|
+
* Caller is responsible for verifying admin permission before calling this.
|
|
224
|
+
*/
|
|
225
|
+
async handle(req, res, _authResult) {
|
|
226
|
+
const url = req.url ?? '/';
|
|
227
|
+
const method = req.method ?? 'GET';
|
|
228
|
+
const path = url.split('?')[0];
|
|
229
|
+
const match = matchRoute(method, path);
|
|
230
|
+
if (match === null) {
|
|
231
|
+
jsonResponse(res, 404, { error: 'Not found' });
|
|
232
|
+
return;
|
|
233
|
+
}
|
|
234
|
+
try {
|
|
235
|
+
await this.dispatch(req, res, match.route, match.params);
|
|
236
|
+
}
|
|
237
|
+
catch (err) {
|
|
238
|
+
logger.warn(`Admin API error [${method} ${path}]: ${err}`);
|
|
239
|
+
if (!res.headersSent) {
|
|
240
|
+
jsonResponse(res, 500, { error: 'Internal server error' });
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
// ─── Dispatcher ─────────────────────────────────────────────────────────────
|
|
245
|
+
async dispatch(req, res, route, params) {
|
|
246
|
+
switch (route) {
|
|
247
|
+
case 'createTenant': return this.createTenant(req, res);
|
|
248
|
+
case 'listTenants': return this.listTenants(res);
|
|
249
|
+
case 'getTenant': return this.getTenant(res, params['id']);
|
|
250
|
+
case 'updateTenant': return this.updateTenant(req, res, params['id']);
|
|
251
|
+
case 'deleteTenant': return this.deleteTenant(res, params['id']);
|
|
252
|
+
case 'createKey': return this.createKey(req, res, params['id']);
|
|
253
|
+
case 'listKeys': return this.listKeys(res, params['id']);
|
|
254
|
+
case 'revokeKey': return this.revokeKey(res, params['prefix']);
|
|
255
|
+
case 'globalStats': return this.globalStats(res);
|
|
256
|
+
case 'tenantStats': return this.tenantStats(res, params['id']);
|
|
257
|
+
// Phase 18 flat key routes
|
|
258
|
+
case 'createKeyFlat': return this.createKeyFlat(req, res);
|
|
259
|
+
case 'listKeysFlat': return this.listKeysFlat(res);
|
|
260
|
+
case 'keyUsage': return this.keyUsage(res, params['id']);
|
|
261
|
+
// Task 128 workspace routes
|
|
262
|
+
case 'createWorkspace': return this.createWorkspace(req, res);
|
|
263
|
+
case 'listWorkspaces': return this.listWorkspaces(res);
|
|
264
|
+
case 'getWorkspace': return this.getWorkspace(res, params['id']);
|
|
265
|
+
case 'updateWorkspace': return this.updateWorkspace(req, res, params['id']);
|
|
266
|
+
case 'addWorkspaceMember': return this.addWorkspaceMember(req, res, params['id']);
|
|
267
|
+
case 'listWorkspaceMembers': return this.listWorkspaceMembers(res, params['id']);
|
|
268
|
+
case 'removeWorkspaceMember': return this.removeWorkspaceMember(res, params['id'], params['userId']);
|
|
269
|
+
case 'workspaceUsage': return this.workspaceUsage(res, params['id']);
|
|
270
|
+
default:
|
|
271
|
+
jsonResponse(res, 404, { error: 'Not found' });
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
// ─── Tenant CRUD ────────────────────────────────────────────────────────────
|
|
275
|
+
async createTenant(req, res) {
|
|
276
|
+
const body = await parseJsonBody(req);
|
|
277
|
+
if (body === null) {
|
|
278
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
279
|
+
return;
|
|
280
|
+
}
|
|
281
|
+
const name = body['name'];
|
|
282
|
+
if (typeof name !== 'string' || !name.trim()) {
|
|
283
|
+
jsonResponse(res, 400, { error: 'name is required and must be a non-empty string' });
|
|
284
|
+
return;
|
|
285
|
+
}
|
|
286
|
+
const storageQuotaBytes = body['storage_quota_bytes'];
|
|
287
|
+
const settings = body['settings'];
|
|
288
|
+
const tenant = this.tenantStore.create(name.trim(), {
|
|
289
|
+
storageQuotaBytes: typeof storageQuotaBytes === 'number' ? storageQuotaBytes : undefined,
|
|
290
|
+
settings: typeof settings === 'object' && settings !== null && !Array.isArray(settings)
|
|
291
|
+
? settings
|
|
292
|
+
: undefined,
|
|
293
|
+
});
|
|
294
|
+
jsonResponse(res, 201, tenantToJson(tenant));
|
|
295
|
+
}
|
|
296
|
+
listTenants(res) {
|
|
297
|
+
const tenants = this.tenantStore.list();
|
|
298
|
+
jsonResponse(res, 200, tenants.map(tenantToJson));
|
|
299
|
+
}
|
|
300
|
+
getTenant(res, id) {
|
|
301
|
+
const tenant = this.tenantStore.get(id);
|
|
302
|
+
if (tenant === null) {
|
|
303
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
304
|
+
return;
|
|
305
|
+
}
|
|
306
|
+
jsonResponse(res, 200, tenantToJson(tenant));
|
|
307
|
+
}
|
|
308
|
+
async updateTenant(req, res, id) {
|
|
309
|
+
const existing = this.tenantStore.get(id);
|
|
310
|
+
if (existing === null) {
|
|
311
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
312
|
+
return;
|
|
313
|
+
}
|
|
314
|
+
const body = await parseJsonBody(req);
|
|
315
|
+
if (body === null) {
|
|
316
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
317
|
+
return;
|
|
318
|
+
}
|
|
319
|
+
const updates = {};
|
|
320
|
+
if (typeof body['name'] === 'string' && body['name'].trim()) {
|
|
321
|
+
updates.name = body['name'].trim();
|
|
322
|
+
}
|
|
323
|
+
if (typeof body['storage_quota_bytes'] === 'number') {
|
|
324
|
+
updates.storageQuotaBytes = body['storage_quota_bytes'];
|
|
325
|
+
}
|
|
326
|
+
if (body['settings'] !== undefined &&
|
|
327
|
+
typeof body['settings'] === 'object' &&
|
|
328
|
+
!Array.isArray(body['settings'])) {
|
|
329
|
+
updates.settings = body['settings'];
|
|
330
|
+
}
|
|
331
|
+
this.tenantStore.update(id, updates);
|
|
332
|
+
const updated = this.tenantStore.get(id);
|
|
333
|
+
jsonResponse(res, 200, tenantToJson(updated));
|
|
334
|
+
}
|
|
335
|
+
deleteTenant(res, id) {
|
|
336
|
+
const existing = this.tenantStore.get(id);
|
|
337
|
+
if (existing === null) {
|
|
338
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
339
|
+
return;
|
|
340
|
+
}
|
|
341
|
+
// Remove per-repo index databases belonging to this tenant
|
|
342
|
+
this.deleteTenantRepoDatabases(id);
|
|
343
|
+
// Delete from auth DB — cascades to api_keys rows
|
|
344
|
+
this.tenantStore.delete(id);
|
|
345
|
+
jsonResponse(res, 200, { deleted: true, tenant_id: id });
|
|
346
|
+
}
|
|
347
|
+
deleteTenantRepoDatabases(tenantId) {
|
|
348
|
+
if (!existsSync(this.indexDir))
|
|
349
|
+
return;
|
|
350
|
+
let files;
|
|
351
|
+
try {
|
|
352
|
+
files = readdirSync(this.indexDir).filter((f) => f.endsWith('.db'));
|
|
353
|
+
}
|
|
354
|
+
catch {
|
|
355
|
+
return;
|
|
356
|
+
}
|
|
357
|
+
for (const file of files) {
|
|
358
|
+
const dbPath = join(this.indexDir, file);
|
|
359
|
+
try {
|
|
360
|
+
const db = new Database(dbPath, { readonly: true });
|
|
361
|
+
const row = db
|
|
362
|
+
.prepare('SELECT id FROM repos WHERE tenant_id = ? LIMIT 1')
|
|
363
|
+
.get(tenantId);
|
|
364
|
+
db.close();
|
|
365
|
+
if (row !== undefined) {
|
|
366
|
+
unlinkSync(dbPath);
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
catch {
|
|
370
|
+
// Skip unreadable / corrupted databases
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
// ─── API key management ──────────────────────────────────────────────────────
|
|
375
|
+
async createKey(req, res, tenantId) {
|
|
376
|
+
const existing = this.tenantStore.get(tenantId);
|
|
377
|
+
if (existing === null) {
|
|
378
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
379
|
+
return;
|
|
380
|
+
}
|
|
381
|
+
const body = await parseJsonBody(req);
|
|
382
|
+
if (body === null) {
|
|
383
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
384
|
+
return;
|
|
385
|
+
}
|
|
386
|
+
const rawPerms = body['permissions'];
|
|
387
|
+
const permissions = Array.isArray(rawPerms)
|
|
388
|
+
? (rawPerms.filter((p) => p === 'read' || p === 'write' || p === 'admin'))
|
|
389
|
+
: ['read'];
|
|
390
|
+
if (permissions.length === 0)
|
|
391
|
+
permissions.push('read');
|
|
392
|
+
const rateLimitTier = typeof body['rate_limit_tier'] === 'string' ? body['rate_limit_tier'] : 'default';
|
|
393
|
+
const isTest = body['is_test'] === true;
|
|
394
|
+
const rawKey = this.validator.generate(tenantId, permissions, {
|
|
395
|
+
rateLimitTier,
|
|
396
|
+
isTest,
|
|
397
|
+
tenantName: existing.name,
|
|
398
|
+
});
|
|
399
|
+
jsonResponse(res, 201, {
|
|
400
|
+
key: rawKey,
|
|
401
|
+
tenant_id: tenantId,
|
|
402
|
+
permissions,
|
|
403
|
+
rate_limit_tier: rateLimitTier,
|
|
404
|
+
});
|
|
405
|
+
}
|
|
406
|
+
listKeys(res, tenantId) {
|
|
407
|
+
const existing = this.tenantStore.get(tenantId);
|
|
408
|
+
if (existing === null) {
|
|
409
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
410
|
+
return;
|
|
411
|
+
}
|
|
412
|
+
const keys = this.keyStore.listByTenant(tenantId);
|
|
413
|
+
jsonResponse(res, 200, keys.map(keyToJson));
|
|
414
|
+
}
|
|
415
|
+
revokeKey(res, prefix) {
|
|
416
|
+
this.validator.revoke(prefix);
|
|
417
|
+
jsonResponse(res, 200, { revoked: true, prefix });
|
|
418
|
+
}
|
|
419
|
+
// ─── Stats ──────────────────────────────────────────────────────────────────
|
|
420
|
+
globalStats(res) {
|
|
421
|
+
const tenants = this.tenantStore.list();
|
|
422
|
+
const since24h = new Date(Date.now() - 24 * 60 * 60 * 1000).toISOString();
|
|
423
|
+
const logStats = this.logStore.getStats(since24h);
|
|
424
|
+
const storageUsedBytes = tenants.reduce((sum, t) => sum + t.storageUsedBytes, 0);
|
|
425
|
+
jsonResponse(res, 200, {
|
|
426
|
+
tenants: tenants.length,
|
|
427
|
+
total_repos: this.countRepoFiles(),
|
|
428
|
+
total_symbols: 0, // Requires scanning all per-repo DBs — omitted for performance
|
|
429
|
+
storage_used_bytes: storageUsedBytes,
|
|
430
|
+
requests_24h: logStats.requests,
|
|
431
|
+
rate_limit_hits_24h: logStats.rateLimited,
|
|
432
|
+
uptime_seconds: Math.floor((Date.now() - this.startTime) / 1000),
|
|
433
|
+
});
|
|
434
|
+
}
|
|
435
|
+
tenantStats(res, tenantId) {
|
|
436
|
+
const tenant = this.tenantStore.get(tenantId);
|
|
437
|
+
if (tenant === null) {
|
|
438
|
+
jsonResponse(res, 404, { error: 'Tenant not found' });
|
|
439
|
+
return;
|
|
440
|
+
}
|
|
441
|
+
const since24h = new Date(Date.now() - 24 * 60 * 60 * 1000).toISOString();
|
|
442
|
+
const logStats = this.logStore.getTenantStats(tenantId, since24h);
|
|
443
|
+
jsonResponse(res, 200, {
|
|
444
|
+
repos: 0, // Requires scanning per-repo DBs — omitted for performance
|
|
445
|
+
symbols: 0,
|
|
446
|
+
storage_used_bytes: tenant.storageUsedBytes,
|
|
447
|
+
requests_24h: logStats.requests,
|
|
448
|
+
rate_limit_hits_24h: logStats.rateLimited,
|
|
449
|
+
});
|
|
450
|
+
}
|
|
451
|
+
// ─── Phase 18 flat key routes ────────────────────────────────────────────────
|
|
452
|
+
/**
|
|
453
|
+
* POST /admin/keys
|
|
454
|
+
* Create a new API key. Body: { tenant_id?, label?, permissions?, rate_limit_tier?, is_test? }
|
|
455
|
+
* When tenant_id is omitted, creates or reuses a 'default' tenant.
|
|
456
|
+
*/
|
|
457
|
+
async createKeyFlat(req, res) {
|
|
458
|
+
const body = await parseJsonBody(req);
|
|
459
|
+
if (body === null) {
|
|
460
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
461
|
+
return;
|
|
462
|
+
}
|
|
463
|
+
const tenantId = typeof body['tenant_id'] === 'string' && body['tenant_id'].trim()
|
|
464
|
+
? body['tenant_id'].trim()
|
|
465
|
+
: 'default';
|
|
466
|
+
const label = typeof body['label'] === 'string' ? body['label'].trim() || null : null;
|
|
467
|
+
const rawPerms = body['permissions'];
|
|
468
|
+
const permissions = Array.isArray(rawPerms)
|
|
469
|
+
? rawPerms.filter((p) => p === 'read' || p === 'write' || p === 'admin')
|
|
470
|
+
: ['read'];
|
|
471
|
+
if (permissions.length === 0)
|
|
472
|
+
permissions.push('read');
|
|
473
|
+
const rateLimitTier = typeof body['rate_limit_tier'] === 'string' ? body['rate_limit_tier'] : 'default';
|
|
474
|
+
const isTest = body['is_test'] === true;
|
|
475
|
+
const rawKey = this.validator.generate(tenantId, permissions, {
|
|
476
|
+
rateLimitTier,
|
|
477
|
+
isTest,
|
|
478
|
+
tenantName: tenantId,
|
|
479
|
+
});
|
|
480
|
+
// Patch the label onto the stored record (validator.generate doesn't accept label yet)
|
|
481
|
+
if (label !== null) {
|
|
482
|
+
const { hashApiKey } = await import('../core/db/api-keys.js');
|
|
483
|
+
this.keyStore.updateLabel(hashApiKey(rawKey), label);
|
|
484
|
+
}
|
|
485
|
+
jsonResponse(res, 201, {
|
|
486
|
+
key: rawKey,
|
|
487
|
+
tenant_id: tenantId,
|
|
488
|
+
label,
|
|
489
|
+
permissions,
|
|
490
|
+
rate_limit_tier: rateLimitTier,
|
|
491
|
+
});
|
|
492
|
+
}
|
|
493
|
+
/**
|
|
494
|
+
* GET /admin/keys
|
|
495
|
+
* List all API keys across all tenants (hashes only — never raw values).
|
|
496
|
+
*/
|
|
497
|
+
listKeysFlat(res) {
|
|
498
|
+
const tenants = this.tenantStore.list();
|
|
499
|
+
const allKeys = [];
|
|
500
|
+
for (const t of tenants) {
|
|
501
|
+
const keys = this.keyStore.listByTenant(t.id);
|
|
502
|
+
for (const k of keys) {
|
|
503
|
+
allKeys.push(keyToJson(k));
|
|
504
|
+
}
|
|
505
|
+
}
|
|
506
|
+
allKeys.sort((a, b) => String(b['created_at'] ?? '').localeCompare(String(a['created_at'] ?? '')));
|
|
507
|
+
jsonResponse(res, 200, allKeys);
|
|
508
|
+
}
|
|
509
|
+
/**
|
|
510
|
+
* GET /admin/keys/:id
|
|
511
|
+
* GET /admin/keys/:id/usage
|
|
512
|
+
* Returns last_used_at and request count for a key, identified by its hash prefix.
|
|
513
|
+
*/
|
|
514
|
+
keyUsage(res, hashPrefix) {
|
|
515
|
+
const tenants = this.tenantStore.list();
|
|
516
|
+
for (const t of tenants) {
|
|
517
|
+
const keys = this.keyStore.listByTenant(t.id);
|
|
518
|
+
const match = keys.find((k) => k.keyHash.startsWith(hashPrefix));
|
|
519
|
+
if (match) {
|
|
520
|
+
const since24h = new Date(Date.now() - 24 * 60 * 60 * 1000).toISOString();
|
|
521
|
+
const logStats = this.logStore.getTenantStats(t.id, since24h);
|
|
522
|
+
jsonResponse(res, 200, {
|
|
523
|
+
key_hash_prefix: match.keyHash.slice(0, 8),
|
|
524
|
+
tenant_id: match.tenantId,
|
|
525
|
+
label: match.label,
|
|
526
|
+
permissions: match.permissions,
|
|
527
|
+
rate_limit_tier: match.rateLimitTier,
|
|
528
|
+
created_at: match.createdAt,
|
|
529
|
+
last_used_at: match.lastUsedAt,
|
|
530
|
+
revoked: match.revokedAt !== null,
|
|
531
|
+
requests_24h: logStats.requests,
|
|
532
|
+
});
|
|
533
|
+
return;
|
|
534
|
+
}
|
|
535
|
+
}
|
|
536
|
+
jsonResponse(res, 404, { error: `No key found with hash prefix: ${hashPrefix}` });
|
|
537
|
+
}
|
|
538
|
+
// ─── Workspace management (Task 128) ────────────────────────────────────────
|
|
539
|
+
async createWorkspace(req, res) {
|
|
540
|
+
const body = await parseJsonBody(req);
|
|
541
|
+
if (body === null) {
|
|
542
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
543
|
+
return;
|
|
544
|
+
}
|
|
545
|
+
const name = body['name'];
|
|
546
|
+
if (typeof name !== 'string' || !name.trim()) {
|
|
547
|
+
jsonResponse(res, 400, { error: 'name is required and must be a non-empty string' });
|
|
548
|
+
return;
|
|
549
|
+
}
|
|
550
|
+
const rawPlan = body['plan'];
|
|
551
|
+
const plan = rawPlan === 'free' || rawPlan === 'team' || rawPlan === 'enterprise'
|
|
552
|
+
? rawPlan
|
|
553
|
+
: 'team';
|
|
554
|
+
const workspace = this.tenantStore.create(name.trim(), { plan });
|
|
555
|
+
jsonResponse(res, 201, workspaceToJson(workspace));
|
|
556
|
+
}
|
|
557
|
+
listWorkspaces(res) {
|
|
558
|
+
const workspaces = this.tenantStore.list();
|
|
559
|
+
jsonResponse(res, 200, workspaces.map(workspaceToJson));
|
|
560
|
+
}
|
|
561
|
+
getWorkspace(res, id) {
|
|
562
|
+
const workspace = this.tenantStore.get(id);
|
|
563
|
+
if (workspace === null) {
|
|
564
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
565
|
+
return;
|
|
566
|
+
}
|
|
567
|
+
jsonResponse(res, 200, workspaceToJson(workspace));
|
|
568
|
+
}
|
|
569
|
+
async updateWorkspace(req, res, id) {
|
|
570
|
+
const existing = this.tenantStore.get(id);
|
|
571
|
+
if (existing === null) {
|
|
572
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
573
|
+
return;
|
|
574
|
+
}
|
|
575
|
+
const body = await parseJsonBody(req);
|
|
576
|
+
if (body === null) {
|
|
577
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
578
|
+
return;
|
|
579
|
+
}
|
|
580
|
+
const updates = {};
|
|
581
|
+
if (typeof body['name'] === 'string' && body['name'].trim()) {
|
|
582
|
+
updates.name = body['name'].trim();
|
|
583
|
+
}
|
|
584
|
+
const rawPlan = body['plan'];
|
|
585
|
+
if (rawPlan === 'free' || rawPlan === 'team' || rawPlan === 'enterprise') {
|
|
586
|
+
updates.plan = rawPlan;
|
|
587
|
+
}
|
|
588
|
+
this.tenantStore.update(id, updates);
|
|
589
|
+
const updated = this.tenantStore.get(id);
|
|
590
|
+
jsonResponse(res, 200, workspaceToJson(updated));
|
|
591
|
+
}
|
|
592
|
+
async addWorkspaceMember(req, res, id) {
|
|
593
|
+
const workspace = this.tenantStore.get(id);
|
|
594
|
+
if (workspace === null) {
|
|
595
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
596
|
+
return;
|
|
597
|
+
}
|
|
598
|
+
const body = await parseJsonBody(req);
|
|
599
|
+
if (body === null) {
|
|
600
|
+
jsonResponse(res, 400, { error: 'Invalid JSON body' });
|
|
601
|
+
return;
|
|
602
|
+
}
|
|
603
|
+
const userId = body['userId'] ?? body['user_id'];
|
|
604
|
+
if (typeof userId !== 'string' || !userId.trim()) {
|
|
605
|
+
jsonResponse(res, 400, { error: 'userId is required' });
|
|
606
|
+
return;
|
|
607
|
+
}
|
|
608
|
+
const rawRole = body['role'];
|
|
609
|
+
const role = rawRole === 'owner' || rawRole === 'admin' || rawRole === 'member' || rawRole === 'readonly'
|
|
610
|
+
? rawRole
|
|
611
|
+
: 'member';
|
|
612
|
+
this.tenantStore.addMember(id, userId.trim(), role);
|
|
613
|
+
jsonResponse(res, 201, { workspaceId: id, userId: userId.trim(), role });
|
|
614
|
+
}
|
|
615
|
+
listWorkspaceMembers(res, id) {
|
|
616
|
+
const workspace = this.tenantStore.get(id);
|
|
617
|
+
if (workspace === null) {
|
|
618
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
619
|
+
return;
|
|
620
|
+
}
|
|
621
|
+
const members = this.tenantStore.listMembers(id);
|
|
622
|
+
jsonResponse(res, 200, members.map(memberToJson));
|
|
623
|
+
}
|
|
624
|
+
removeWorkspaceMember(res, id, userId) {
|
|
625
|
+
const workspace = this.tenantStore.get(id);
|
|
626
|
+
if (workspace === null) {
|
|
627
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
628
|
+
return;
|
|
629
|
+
}
|
|
630
|
+
this.tenantStore.removeMember(id, userId);
|
|
631
|
+
jsonResponse(res, 200, { removed: true, workspaceId: id, userId });
|
|
632
|
+
}
|
|
633
|
+
workspaceUsage(res, id) {
|
|
634
|
+
const workspace = this.tenantStore.get(id);
|
|
635
|
+
if (workspace === null) {
|
|
636
|
+
jsonResponse(res, 404, { error: 'Workspace not found' });
|
|
637
|
+
return;
|
|
638
|
+
}
|
|
639
|
+
const since30d = new Date(Date.now() - 30 * 24 * 60 * 60 * 1000).toISOString();
|
|
640
|
+
const logStats = this.logStore.getTenantStats(id, since30d);
|
|
641
|
+
jsonResponse(res, 200, {
|
|
642
|
+
workspace_id: id,
|
|
643
|
+
file_count: 0, // Requires scanning per-repo DBs — omitted for performance
|
|
644
|
+
symbol_count: 0, // Requires scanning per-repo DBs — omitted for performance
|
|
645
|
+
storage_used_bytes: workspace.storageUsedBytes,
|
|
646
|
+
api_calls_month: logStats.requests,
|
|
647
|
+
plan: workspace.plan,
|
|
648
|
+
});
|
|
649
|
+
}
|
|
650
|
+
countRepoFiles() {
|
|
651
|
+
if (!existsSync(this.indexDir))
|
|
652
|
+
return 0;
|
|
653
|
+
try {
|
|
654
|
+
return readdirSync(this.indexDir).filter((f) => f.endsWith('.db')).length;
|
|
655
|
+
}
|
|
656
|
+
catch {
|
|
657
|
+
return 0;
|
|
658
|
+
}
|
|
659
|
+
}
|
|
660
|
+
}
|
|
661
|
+
//# sourceMappingURL=admin-api.js.map
|