@overlordai/server 1.0.53 → 1.0.55
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/database/migrations/001-init-schema.sql +73 -9
- package/dist/adapters/adapter.interface.d.ts +1 -4
- package/dist/adapters/adapter.interface.d.ts.map +1 -1
- package/dist/adapters/adapter.interface.js.map +1 -1
- package/dist/adapters/adapter.module.d.ts.map +1 -1
- package/dist/adapters/adapter.module.js +8 -7
- package/dist/adapters/adapter.module.js.map +1 -1
- package/dist/adapters/lark/lark-card.builder.d.ts +1 -25
- package/dist/adapters/lark/lark-card.builder.d.ts.map +1 -1
- package/dist/adapters/lark/lark-card.builder.js +6 -110
- package/dist/adapters/lark/lark-card.builder.js.map +1 -1
- package/dist/adapters/lark/lark-message.parser.d.ts +4 -4
- package/dist/adapters/lark/lark-message.parser.d.ts.map +1 -1
- package/dist/adapters/lark/lark-message.parser.js +20 -13
- package/dist/adapters/lark/lark-message.parser.js.map +1 -1
- package/dist/adapters/lark/lark-signature.d.ts.map +1 -1
- package/dist/adapters/lark/lark-signature.js +6 -0
- package/dist/adapters/lark/lark-signature.js.map +1 -1
- package/dist/adapters/lark/lark.adapter.d.ts +14 -9
- package/dist/adapters/lark/lark.adapter.d.ts.map +1 -1
- package/dist/adapters/lark/lark.adapter.js +99 -177
- package/dist/adapters/lark/lark.adapter.js.map +1 -1
- package/dist/adapters/lark/lark.controller.d.ts +10 -1
- package/dist/adapters/lark/lark.controller.d.ts.map +1 -1
- package/dist/adapters/lark/lark.controller.js +48 -14
- package/dist/adapters/lark/lark.controller.js.map +1 -1
- package/dist/adapters/slack/slack-block.builder.d.ts +8 -0
- package/dist/adapters/slack/slack-block.builder.d.ts.map +1 -0
- package/dist/adapters/slack/slack-block.builder.js +117 -0
- package/dist/adapters/slack/slack-block.builder.js.map +1 -0
- package/dist/adapters/slack/slack-message.parser.d.ts +15 -0
- package/dist/adapters/slack/slack-message.parser.d.ts.map +1 -0
- package/dist/adapters/slack/slack-message.parser.js +158 -0
- package/dist/adapters/slack/slack-message.parser.js.map +1 -0
- package/dist/adapters/slack/slack-signature.d.ts +7 -0
- package/dist/adapters/slack/slack-signature.d.ts.map +1 -0
- package/dist/adapters/slack/slack-signature.js +59 -0
- package/dist/adapters/slack/slack-signature.js.map +1 -0
- package/dist/adapters/slack/slack.adapter.d.ts +67 -13
- package/dist/adapters/slack/slack.adapter.d.ts.map +1 -1
- package/dist/adapters/slack/slack.adapter.js +468 -19
- package/dist/adapters/slack/slack.adapter.js.map +1 -1
- package/dist/adapters/slack/slack.controller.d.ts +20 -0
- package/dist/adapters/slack/slack.controller.d.ts.map +1 -0
- package/dist/adapters/slack/slack.controller.js +257 -0
- package/dist/adapters/slack/slack.controller.js.map +1 -0
- package/dist/app.module.d.ts.map +1 -1
- package/dist/app.module.js +4 -0
- package/dist/app.module.js.map +1 -1
- package/dist/auth/auth.controller.d.ts.map +1 -1
- package/dist/auth/auth.controller.js +1 -0
- package/dist/auth/auth.controller.js.map +1 -1
- package/dist/auth/auth.module.d.ts.map +1 -1
- package/dist/auth/auth.module.js +4 -5
- package/dist/auth/auth.module.js.map +1 -1
- package/dist/auth/auth.service.d.ts +9 -2
- package/dist/auth/auth.service.d.ts.map +1 -1
- package/dist/auth/auth.service.js +50 -79
- package/dist/auth/auth.service.js.map +1 -1
- package/dist/auth/authenticated-request.d.ts +10 -0
- package/dist/auth/authenticated-request.d.ts.map +1 -0
- package/dist/auth/authenticated-request.js +3 -0
- package/dist/auth/authenticated-request.js.map +1 -0
- package/dist/auth/extract-user.middleware.d.ts.map +1 -1
- package/dist/auth/extract-user.middleware.js +2 -1
- package/dist/auth/extract-user.middleware.js.map +1 -1
- package/dist/auth/guards/jwt-auth.guard.d.ts.map +1 -1
- package/dist/auth/guards/jwt-auth.guard.js +5 -2
- package/dist/auth/guards/jwt-auth.guard.js.map +1 -1
- package/dist/auth/guards/project-role.guard.d.ts.map +1 -1
- package/dist/auth/guards/project-role.guard.js +6 -8
- package/dist/auth/guards/project-role.guard.js.map +1 -1
- package/dist/auth/jwt.strategy.d.ts +6 -3
- package/dist/auth/jwt.strategy.d.ts.map +1 -1
- package/dist/auth/jwt.strategy.js +15 -5
- package/dist/auth/jwt.strategy.js.map +1 -1
- package/dist/common/command-parser.d.ts +29 -0
- package/dist/common/command-parser.d.ts.map +1 -0
- package/dist/common/command-parser.js +133 -0
- package/dist/common/command-parser.js.map +1 -0
- package/dist/common/config.d.ts +17 -0
- package/dist/common/config.d.ts.map +1 -0
- package/dist/common/config.js +47 -0
- package/dist/common/config.js.map +1 -0
- package/dist/common/crypto.service.d.ts +4 -1
- package/dist/common/crypto.service.d.ts.map +1 -1
- package/dist/common/crypto.service.js +14 -7
- package/dist/common/crypto.service.js.map +1 -1
- package/dist/common/error-filter.d.ts +1 -0
- package/dist/common/error-filter.d.ts.map +1 -1
- package/dist/common/error-filter.js +6 -2
- package/dist/common/error-filter.js.map +1 -1
- package/dist/common/git-utils.d.ts +9 -0
- package/dist/common/git-utils.d.ts.map +1 -0
- package/dist/common/git-utils.js +41 -0
- package/dist/common/git-utils.js.map +1 -0
- package/dist/common/health.controller.d.ts.map +1 -1
- package/dist/common/health.controller.js +3 -5
- package/dist/common/health.controller.js.map +1 -1
- package/dist/common/machine-utils.d.ts +32 -0
- package/dist/common/machine-utils.d.ts.map +1 -0
- package/dist/common/machine-utils.js +12 -0
- package/dist/common/machine-utils.js.map +1 -0
- package/dist/common/pagination.d.ts +12 -5
- package/dist/common/pagination.d.ts.map +1 -1
- package/dist/common/pagination.js +27 -17
- package/dist/common/pagination.js.map +1 -1
- package/dist/common/project-validation.d.ts +7 -0
- package/dist/common/project-validation.d.ts.map +1 -0
- package/dist/common/project-validation.js +86 -0
- package/dist/common/project-validation.js.map +1 -0
- package/dist/common/rate-limit.guard.d.ts +4 -3
- package/dist/common/rate-limit.guard.d.ts.map +1 -1
- package/dist/common/rate-limit.guard.js +14 -5
- package/dist/common/rate-limit.guard.js.map +1 -1
- package/dist/common/sql-utils.d.ts +6 -0
- package/dist/common/sql-utils.d.ts.map +1 -0
- package/dist/common/sql-utils.js +11 -0
- package/dist/common/sql-utils.js.map +1 -0
- package/dist/common/string-utils.d.ts +6 -0
- package/dist/common/string-utils.d.ts.map +1 -0
- package/dist/common/string-utils.js +15 -0
- package/dist/common/string-utils.js.map +1 -0
- package/dist/common/worker-utils.d.ts +31 -0
- package/dist/common/worker-utils.d.ts.map +1 -0
- package/dist/common/worker-utils.js +12 -0
- package/dist/common/worker-utils.js.map +1 -0
- package/dist/database/base.repository.d.ts +56 -0
- package/dist/database/base.repository.d.ts.map +1 -0
- package/dist/database/base.repository.js +82 -0
- package/dist/database/base.repository.js.map +1 -0
- package/dist/database/database.service.d.ts.map +1 -1
- package/dist/database/database.service.js +9 -1
- package/dist/database/database.service.js.map +1 -1
- package/dist/database/migration-runner.d.ts.map +1 -1
- package/dist/database/migration-runner.js +2 -1
- package/dist/database/migration-runner.js.map +1 -1
- package/dist/database/repositories/audit-log.repository.d.ts.map +1 -1
- package/dist/database/repositories/audit-log.repository.js +16 -18
- package/dist/database/repositories/audit-log.repository.js.map +1 -1
- package/dist/database/repositories/bot.repository.d.ts +18 -32
- package/dist/database/repositories/bot.repository.d.ts.map +1 -1
- package/dist/database/repositories/bot.repository.js +42 -21
- package/dist/database/repositories/bot.repository.js.map +1 -1
- package/dist/database/repositories/developer-token.repository.d.ts +7 -17
- package/dist/database/repositories/developer-token.repository.d.ts.map +1 -1
- package/dist/database/repositories/developer-token.repository.js +24 -15
- package/dist/database/repositories/developer-token.repository.js.map +1 -1
- package/dist/database/repositories/developer.repository.d.ts +5 -1
- package/dist/database/repositories/developer.repository.d.ts.map +1 -1
- package/dist/database/repositories/developer.repository.js +60 -49
- package/dist/database/repositories/developer.repository.js.map +1 -1
- package/dist/database/repositories/machine.repository.d.ts.map +1 -1
- package/dist/database/repositories/machine.repository.js +2 -7
- package/dist/database/repositories/machine.repository.js.map +1 -1
- package/dist/database/repositories/notification.repository.d.ts +1 -0
- package/dist/database/repositories/notification.repository.d.ts.map +1 -1
- package/dist/database/repositories/notification.repository.js +25 -20
- package/dist/database/repositories/notification.repository.js.map +1 -1
- package/dist/database/repositories/project-member.repository.d.ts +7 -16
- package/dist/database/repositories/project-member.repository.d.ts.map +1 -1
- package/dist/database/repositories/project-member.repository.js +34 -24
- package/dist/database/repositories/project-member.repository.js.map +1 -1
- package/dist/database/repositories/project.repository.d.ts +2 -1
- package/dist/database/repositories/project.repository.d.ts.map +1 -1
- package/dist/database/repositories/project.repository.js +70 -71
- package/dist/database/repositories/project.repository.js.map +1 -1
- package/dist/database/repositories/session.repository.d.ts.map +1 -1
- package/dist/database/repositories/session.repository.js +22 -25
- package/dist/database/repositories/session.repository.js.map +1 -1
- package/dist/database/repositories/task.repository.d.ts +31 -7
- package/dist/database/repositories/task.repository.d.ts.map +1 -1
- package/dist/database/repositories/task.repository.js +134 -86
- package/dist/database/repositories/task.repository.js.map +1 -1
- package/dist/database/repositories/worker-token.repository.d.ts.map +1 -1
- package/dist/database/repositories/worker-token.repository.js +18 -16
- package/dist/database/repositories/worker-token.repository.js.map +1 -1
- package/dist/database/repositories/worker.repository.d.ts +50 -0
- package/dist/database/repositories/worker.repository.d.ts.map +1 -0
- package/dist/database/repositories/worker.repository.js +215 -0
- package/dist/database/repositories/worker.repository.js.map +1 -0
- package/dist/database/repositories/workspace.repository.d.ts +3 -2
- package/dist/database/repositories/workspace.repository.d.ts.map +1 -1
- package/dist/database/repositories/workspace.repository.js +29 -21
- package/dist/database/repositories/workspace.repository.js.map +1 -1
- package/dist/database/repository.module.d.ts +3 -0
- package/dist/database/repository.module.d.ts.map +1 -0
- package/dist/database/repository.module.js +45 -0
- package/dist/database/repository.module.js.map +1 -0
- package/dist/dispatcher/capability.service.d.ts +19 -14
- package/dist/dispatcher/capability.service.d.ts.map +1 -1
- package/dist/dispatcher/capability.service.js +77 -69
- package/dist/dispatcher/capability.service.js.map +1 -1
- package/dist/dispatcher/cleanup.service.d.ts +1 -1
- package/dist/dispatcher/cleanup.service.d.ts.map +1 -1
- package/dist/dispatcher/cleanup.service.js +13 -13
- package/dist/dispatcher/cleanup.service.js.map +1 -1
- package/dist/dispatcher/dedup.service.d.ts +17 -3
- package/dist/dispatcher/dedup.service.d.ts.map +1 -1
- package/dist/dispatcher/dedup.service.js +76 -82
- package/dist/dispatcher/dedup.service.js.map +1 -1
- package/dist/dispatcher/dispatcher.module.d.ts.map +1 -1
- package/dist/dispatcher/dispatcher.module.js +11 -18
- package/dist/dispatcher/dispatcher.module.js.map +1 -1
- package/dist/dispatcher/dispatcher.service.d.ts +14 -116
- package/dist/dispatcher/dispatcher.service.d.ts.map +1 -1
- package/dist/dispatcher/dispatcher.service.js +62 -940
- package/dist/dispatcher/dispatcher.service.js.map +1 -1
- package/dist/dispatcher/dispatcher.types.d.ts +33 -0
- package/dist/dispatcher/dispatcher.types.d.ts.map +1 -0
- package/dist/dispatcher/dispatcher.types.js +3 -0
- package/dist/dispatcher/dispatcher.types.js.map +1 -0
- package/dist/dispatcher/heartbeat.service.d.ts +17 -10
- package/dist/dispatcher/heartbeat.service.d.ts.map +1 -1
- package/dist/dispatcher/heartbeat.service.js +47 -51
- package/dist/dispatcher/heartbeat.service.js.map +1 -1
- package/dist/dispatcher/pty-relay.service.d.ts.map +1 -1
- package/dist/dispatcher/pty-relay.service.js +7 -15
- package/dist/dispatcher/pty-relay.service.js.map +1 -1
- package/dist/dispatcher/reconciler.d.ts +18 -8
- package/dist/dispatcher/reconciler.d.ts.map +1 -1
- package/dist/dispatcher/reconciler.js +219 -130
- package/dist/dispatcher/reconciler.js.map +1 -1
- package/dist/dispatcher/scheduler.service.d.ts +15 -9
- package/dist/dispatcher/scheduler.service.d.ts.map +1 -1
- package/dist/dispatcher/scheduler.service.js +95 -53
- package/dist/dispatcher/scheduler.service.js.map +1 -1
- package/dist/dispatcher/state-machine.d.ts.map +1 -1
- package/dist/dispatcher/state-machine.js +1 -5
- package/dist/dispatcher/state-machine.js.map +1 -1
- package/dist/dispatcher/task-creation.service.d.ts +30 -0
- package/dist/dispatcher/task-creation.service.d.ts.map +1 -0
- package/dist/dispatcher/task-creation.service.js +242 -0
- package/dist/dispatcher/task-creation.service.js.map +1 -0
- package/dist/dispatcher/task-lifecycle.service.d.ts +63 -0
- package/dist/dispatcher/task-lifecycle.service.d.ts.map +1 -0
- package/dist/dispatcher/task-lifecycle.service.js +584 -0
- package/dist/dispatcher/task-lifecycle.service.js.map +1 -0
- package/dist/dispatcher/task-log-batcher.d.ts.map +1 -1
- package/dist/dispatcher/task-log-batcher.js +4 -11
- package/dist/dispatcher/task-log-batcher.js.map +1 -1
- package/dist/dispatcher/worker-auth.service.d.ts +29 -0
- package/dist/dispatcher/worker-auth.service.d.ts.map +1 -0
- package/dist/dispatcher/worker-auth.service.js +296 -0
- package/dist/dispatcher/worker-auth.service.js.map +1 -0
- package/dist/dispatcher/worker-connection.manager.d.ts +15 -15
- package/dist/dispatcher/worker-connection.manager.d.ts.map +1 -1
- package/dist/dispatcher/worker-connection.manager.js +35 -43
- package/dist/dispatcher/worker-connection.manager.js.map +1 -1
- package/dist/dispatcher/worker-selector.d.ts +18 -0
- package/dist/dispatcher/worker-selector.d.ts.map +1 -0
- package/dist/dispatcher/worker-selector.js +150 -0
- package/dist/dispatcher/worker-selector.js.map +1 -0
- package/dist/events/event-types.d.ts +31 -0
- package/dist/events/event-types.d.ts.map +1 -0
- package/dist/events/event-types.js +16 -0
- package/dist/events/event-types.js.map +1 -0
- package/dist/events/events.module.d.ts +7 -0
- package/dist/events/events.module.d.ts.map +1 -0
- package/dist/events/events.module.js +26 -0
- package/dist/events/events.module.js.map +1 -0
- package/dist/main.js +22 -0
- package/dist/main.js.map +1 -1
- package/dist/notifier/debouncer.d.ts +1 -1
- package/dist/notifier/debouncer.d.ts.map +1 -1
- package/dist/notifier/debouncer.js +2 -1
- package/dist/notifier/debouncer.js.map +1 -1
- package/dist/notifier/notification-consumer.d.ts +1 -1
- package/dist/notifier/notification-consumer.d.ts.map +1 -1
- package/dist/notifier/notification-consumer.js +5 -5
- package/dist/notifier/notification-consumer.js.map +1 -1
- package/dist/notifier/notifier.module.d.ts.map +1 -1
- package/dist/notifier/notifier.module.js +0 -6
- package/dist/notifier/notifier.module.js.map +1 -1
- package/dist/notifier/notifier.service.d.ts +1 -1
- package/dist/notifier/notifier.service.d.ts.map +1 -1
- package/dist/notifier/notifier.service.js +7 -9
- package/dist/notifier/notifier.service.js.map +1 -1
- package/dist/notifier/template.service.d.ts +1 -1
- package/dist/notifier/template.service.d.ts.map +1 -1
- package/dist/notifier/template.service.js +6 -10
- package/dist/notifier/template.service.js.map +1 -1
- package/dist/redis/redis.service.d.ts.map +1 -1
- package/dist/redis/redis.service.js +2 -2
- package/dist/redis/redis.service.js.map +1 -1
- package/dist/web/admin/admin-audit.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-audit.controller.js +2 -1
- package/dist/web/admin/admin-audit.controller.js.map +1 -1
- package/dist/web/admin/admin-bot.controller.d.ts +11 -48
- package/dist/web/admin/admin-bot.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-bot.controller.js +50 -18
- package/dist/web/admin/admin-bot.controller.js.map +1 -1
- package/dist/web/admin/admin-developer.controller.d.ts +14 -27
- package/dist/web/admin/admin-developer.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-developer.controller.js +62 -28
- package/dist/web/admin/admin-developer.controller.js.map +1 -1
- package/dist/web/admin/admin-machine.controller.d.ts +1 -8
- package/dist/web/admin/admin-machine.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-machine.controller.js +3 -6
- package/dist/web/admin/admin-machine.controller.js.map +1 -1
- package/dist/web/admin/admin-project.controller.d.ts +9 -30
- package/dist/web/admin/admin-project.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-project.controller.js +15 -60
- package/dist/web/admin/admin-project.controller.js.map +1 -1
- package/dist/web/admin/admin-settings.controller.d.ts +7 -10
- package/dist/web/admin/admin-settings.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-settings.controller.js +14 -6
- package/dist/web/admin/admin-settings.controller.js.map +1 -1
- package/dist/web/admin/admin-token.controller.d.ts +6 -13
- package/dist/web/admin/admin-token.controller.d.ts.map +1 -1
- package/dist/web/admin/admin-token.controller.js +15 -27
- package/dist/web/admin/admin-token.controller.js.map +1 -1
- package/dist/web/admin/admin-worker.controller.d.ts +26 -0
- package/dist/web/admin/admin-worker.controller.d.ts.map +1 -0
- package/dist/web/admin/admin-worker.controller.js +184 -0
- package/dist/web/admin/admin-worker.controller.js.map +1 -0
- package/dist/web/dashboard.controller.d.ts +6 -12
- package/dist/web/dashboard.controller.d.ts.map +1 -1
- package/dist/web/dashboard.controller.js +30 -18
- package/dist/web/dashboard.controller.js.map +1 -1
- package/dist/web/dashboard.service.d.ts +21 -12
- package/dist/web/dashboard.service.d.ts.map +1 -1
- package/dist/web/dashboard.service.js +169 -119
- package/dist/web/dashboard.service.js.map +1 -1
- package/dist/web/event.gateway.d.ts +32 -0
- package/dist/web/event.gateway.d.ts.map +1 -0
- package/dist/web/event.gateway.js +168 -0
- package/dist/web/event.gateway.js.map +1 -0
- package/dist/web/frame-handlers/frame-handler.interface.d.ts +24 -0
- package/dist/web/frame-handlers/frame-handler.interface.d.ts.map +1 -0
- package/dist/web/frame-handlers/frame-handler.interface.js +3 -0
- package/dist/web/frame-handlers/frame-handler.interface.js.map +1 -0
- package/dist/web/frame-handlers/frame-handler.registry.d.ts +16 -0
- package/dist/web/frame-handlers/frame-handler.registry.d.ts.map +1 -0
- package/dist/web/frame-handlers/frame-handler.registry.js +39 -0
- package/dist/web/frame-handlers/frame-handler.registry.js.map +1 -0
- package/dist/web/frame-handlers/heartbeat.handler.d.ts +13 -0
- package/dist/web/frame-handlers/heartbeat.handler.d.ts.map +1 -0
- package/dist/web/frame-handlers/heartbeat.handler.js +35 -0
- package/dist/web/frame-handlers/heartbeat.handler.js.map +1 -0
- package/dist/web/frame-handlers/index.d.ts +7 -0
- package/dist/web/frame-handlers/index.d.ts.map +1 -0
- package/dist/web/frame-handlers/index.js +14 -0
- package/dist/web/frame-handlers/index.js.map +1 -0
- package/dist/web/frame-handlers/progress.handler.d.ts +25 -0
- package/dist/web/frame-handlers/progress.handler.d.ts.map +1 -0
- package/dist/web/frame-handlers/progress.handler.js +69 -0
- package/dist/web/frame-handlers/progress.handler.js.map +1 -0
- package/dist/web/frame-handlers/stage-confirm.handler.d.ts +15 -0
- package/dist/web/frame-handlers/stage-confirm.handler.d.ts.map +1 -0
- package/dist/web/frame-handlers/stage-confirm.handler.js +39 -0
- package/dist/web/frame-handlers/stage-confirm.handler.js.map +1 -0
- package/dist/web/frame-handlers/tunnel.handler.d.ts +10 -0
- package/dist/web/frame-handlers/tunnel.handler.d.ts.map +1 -0
- package/dist/web/frame-handlers/tunnel.handler.js +31 -0
- package/dist/web/frame-handlers/tunnel.handler.js.map +1 -0
- package/dist/web/interaction.service.d.ts +0 -4
- package/dist/web/interaction.service.d.ts.map +1 -1
- package/dist/web/interaction.service.js +0 -10
- package/dist/web/interaction.service.js.map +1 -1
- package/dist/web/machine.controller.d.ts +1 -8
- package/dist/web/machine.controller.d.ts.map +1 -1
- package/dist/web/machine.controller.js +6 -9
- package/dist/web/machine.controller.js.map +1 -1
- package/dist/web/notification.controller.d.ts +1 -8
- package/dist/web/notification.controller.d.ts.map +1 -1
- package/dist/web/notification.controller.js +3 -2
- package/dist/web/notification.controller.js.map +1 -1
- package/dist/web/profile.controller.d.ts +19 -10
- package/dist/web/profile.controller.d.ts.map +1 -1
- package/dist/web/profile.controller.js +100 -13
- package/dist/web/profile.controller.js.map +1 -1
- package/dist/web/project-member.service.d.ts +16 -0
- package/dist/web/project-member.service.d.ts.map +1 -0
- package/dist/web/project-member.service.js +90 -0
- package/dist/web/project-member.service.js.map +1 -0
- package/dist/web/project.controller.d.ts +43 -26
- package/dist/web/project.controller.d.ts.map +1 -1
- package/dist/web/project.controller.js +73 -46
- package/dist/web/project.controller.js.map +1 -1
- package/dist/web/pty.gateway.d.ts +9 -3
- package/dist/web/pty.gateway.d.ts.map +1 -1
- package/dist/web/pty.gateway.js +46 -18
- package/dist/web/pty.gateway.js.map +1 -1
- package/dist/web/search.service.d.ts +9 -2
- package/dist/web/search.service.d.ts.map +1 -1
- package/dist/web/search.service.js +53 -26
- package/dist/web/search.service.js.map +1 -1
- package/dist/web/task.controller.d.ts +15 -24
- package/dist/web/task.controller.d.ts.map +1 -1
- package/dist/web/task.controller.js +70 -53
- package/dist/web/task.controller.js.map +1 -1
- package/dist/web/tunnel.service.d.ts +74 -0
- package/dist/web/tunnel.service.d.ts.map +1 -0
- package/dist/web/tunnel.service.js +250 -0
- package/dist/web/tunnel.service.js.map +1 -0
- package/dist/web/web-event.service.d.ts +25 -0
- package/dist/web/web-event.service.d.ts.map +1 -0
- package/dist/web/web-event.service.js +116 -0
- package/dist/web/web-event.service.js.map +1 -0
- package/dist/web/web.module.d.ts.map +1 -1
- package/dist/web/web.module.js +13 -28
- package/dist/web/web.module.js.map +1 -1
- package/dist/web/worker-channel.gateway.d.ts +10 -18
- package/dist/web/worker-channel.gateway.d.ts.map +1 -1
- package/dist/web/worker-channel.gateway.js +70 -144
- package/dist/web/worker-channel.gateway.js.map +1 -1
- package/dist/web/worker-web.controller.d.ts +15 -0
- package/dist/web/worker-web.controller.d.ts.map +1 -0
- package/dist/web/worker-web.controller.js +143 -0
- package/dist/web/worker-web.controller.js.map +1 -0
- package/dist/web/worker.controller.d.ts +3 -3
- package/dist/web/worker.controller.d.ts.map +1 -1
- package/dist/web/worker.controller.js +8 -8
- package/dist/web/worker.controller.js.map +1 -1
- package/dist/web/workspace.controller.d.ts +8 -33
- package/dist/web/workspace.controller.d.ts.map +1 -1
- package/dist/web/workspace.controller.js +93 -205
- package/dist/web/workspace.controller.js.map +1 -1
- package/package.json +10 -2
- package/public/apple-touch-icon-120x120.png +0 -0
- package/public/apple-touch-icon-152x152.png +0 -0
- package/public/apple-touch-icon-180x180.png +0 -0
- package/public/assets/AccessTokensPage-DPQB2fbi.js +1 -0
- package/public/assets/AdminPage-BqVelYNu.js +1 -0
- package/public/assets/ApiReferencePage-CiGvbLxL.js +1 -0
- package/public/assets/AuditLogPage-DSo4jVYm.js +6 -0
- package/public/assets/BindPlatformPage-CTqzpOmt.js +1 -0
- package/public/assets/BotManage-CIR0rrK7.js +6 -0
- package/public/assets/CliReferencePage-C8GmlwUz.js +14 -0
- package/public/assets/DeveloperManage-r6y2AoB4.js +16 -0
- package/public/assets/EditProjectPage-7WCsNltj.js +2 -0
- package/public/assets/{EmptyState-CvmhFgWJ.js → EmptyState-D3foEiul.js} +1 -1
- package/public/assets/HomePage-D4yv4orb.js +1 -0
- package/public/assets/InfoRow-DhdTYoY9.js +1 -0
- package/public/assets/LandingPage-CqS0E2eC.js +43 -0
- package/public/assets/LoginPage-DDXkdcz_.js +1 -0
- package/public/assets/MetricBar-DMMHfS0A.js +1 -0
- package/public/assets/{NotFoundPage-BuiAS4g4.js → NotFoundPage-D5x5BrlX.js} +1 -1
- package/public/assets/OnboardingGuide-D8RyPcEd.js +1 -0
- package/public/assets/PipelineEditorPage-y2-Q8ofQ.js +3 -0
- package/public/assets/ProfilePage-DN7usHOi.js +1 -0
- package/public/assets/ProjectDetailPage-DJexg49z.js +7 -0
- package/public/assets/ProjectListPage-Bz7I2D0H.js +6 -0
- package/public/assets/QuickAuth-Dr0Q50ld.js +1 -0
- package/public/assets/{RemoveMemberConfirmDialog-DS9z6jQT.js → RemoveMemberConfirmDialog-BCrue0AP.js} +2 -2
- package/public/assets/Select-BnV8yZlD.js +6 -0
- package/public/assets/SettingsPage-HaUCcsgl.js +6 -0
- package/public/assets/{Skeleton-CcVqz28_.js → Skeleton-DUgWc2LJ.js} +1 -1
- package/public/assets/SkillPage-BInwZTQh.js +1 -0
- package/public/assets/TaskDetailPage-CfwEj1hy.js +31 -0
- package/public/assets/TaskListPage-Dh59ldSZ.js +1 -0
- package/public/assets/TaskStatusBadge-DuOoGIwE.js +1 -0
- package/public/assets/TerminalHomePage-BwXJjr-a.js +16 -0
- package/public/assets/TokenManage-B0Cpv6SO.js +1 -0
- package/public/assets/TotpSetupPage-MSCCURj9.js +9 -0
- package/public/assets/WorkerDetailPage-R2veIzKo.js +1 -0
- package/public/assets/WorkerListPage-CserMjGO.js +6 -0
- package/public/assets/WorkerSetupGuidePage-SqO2lzVa.js +11 -0
- package/public/assets/{arrow-left-CVKez32c.js → arrow-left-DklRsENx.js} +1 -1
- package/public/assets/{arrow-right-g7hTftEi.js → arrow-right-MDrzFe3K.js} +1 -1
- package/public/assets/{bot-DYvBcsZn.js → bot-DPaziJPf.js} +1 -1
- package/public/assets/{chevron-right-COxU2yxz.js → chevron-right-CqyufMDW.js} +1 -1
- package/public/assets/{copy-BGttVgA1.js → copy-BUH7P2Hf.js} +1 -1
- package/public/assets/date-BdNtiQTP.js +1 -0
- package/public/assets/{external-link-DXlCfUjE.js → external-link-ChPgQ7N_.js} +1 -1
- package/public/assets/index-BS0Fbx5V.css +1 -0
- package/public/assets/index-vL7aQJNr.js +225 -0
- package/public/assets/{key-3eDVdGih.js → key-CxvwwHnW.js} +1 -1
- package/public/assets/{loader-circle-DPm92ETj.js → loader-circle-DS5g1-Od.js} +1 -1
- package/public/assets/password-CHk45-jw.js +1 -0
- package/public/assets/{pencil-Bs3PwH2W.js → pencil-B6spIBcw.js} +1 -1
- package/public/assets/{plus-d-PLzbVX.js → plus-Bnd1Vz2Y.js} +1 -1
- package/public/assets/{rotate-ccw-Cus8CABi.js → rotate-ccw-CgcLAXNR.js} +1 -1
- package/public/assets/{scroll-text-CV3wlIy2.js → scroll-text-CecZ0Fk5.js} +1 -1
- package/public/assets/{settings-Ccijf48b.js → settings-C1uOD3PZ.js} +1 -1
- package/public/assets/status-colors-BPEUp90-.js +1 -0
- package/public/assets/string-B39tzdVK.js +1 -0
- package/public/assets/task-constants-BbFyCyKk.js +14 -0
- package/public/assets/task.store-BE6fEPu4.js +1 -0
- package/public/assets/{trash-2-bRJ-xwtq.js → trash-2-A2FsT1yG.js} +1 -1
- package/public/assets/useFetch-vGZMAvGi.js +1 -0
- package/public/assets/{users-LNQqKSEN.js → users-CEdRS_A3.js} +1 -1
- package/public/assets/wifi-D60NkK6F.js +6 -0
- package/public/assets/zap-DXw1NrWz.js +6 -0
- package/public/icon-192x192.png +0 -0
- package/public/icon-512x512.png +0 -0
- package/public/icon-maskable-192x192.png +0 -0
- package/public/icon-maskable-512x512.png +0 -0
- package/public/index.html +21 -4
- package/public/manifest.webmanifest +1 -0
- package/public/og-image.png +0 -0
- package/public/registerSW.js +1 -0
- package/public/sw.js +1 -0
- package/public/workbox-6e9b121d.js +1 -0
- package/database/migrations/002-add-indexes.sql +0 -17
- package/database/migrations/003-add-settings-table.sql +0 -4
- package/database/migrations/004-add-developer-id-index.sql +0 -5
- package/database/migrations/005-add-worker-version.sql +0 -2
- package/database/migrations/006-add-decommission-fields.sql +0 -2
- package/database/migrations/007-add-ssh-url.sql +0 -1
- package/public/assets/AccessTokensPage-Cb5hGBfN.js +0 -1
- package/public/assets/AdminPage-C7Xytkfo.js +0 -1
- package/public/assets/ApiReferencePage-DLGVc4xN.js +0 -1
- package/public/assets/AuditLogPage-BgqsUJ7x.js +0 -6
- package/public/assets/BotManage-Ds9DYQZA.js +0 -6
- package/public/assets/CliReferencePage-WXinn_69.js +0 -8
- package/public/assets/DeveloperManage-saSq3Hfx.js +0 -16
- package/public/assets/EditProjectPage-C4xWYLTo.js +0 -2
- package/public/assets/HomePage-y20pQ52r.js +0 -1
- package/public/assets/LandingPage-CB5BBbdI.js +0 -36
- package/public/assets/LoginPage-DLljhJkQ.js +0 -1
- package/public/assets/MachineDetailPage-CPm9tfdp.js +0 -1
- package/public/assets/MachineListPage-D-nhVz-m.js +0 -6
- package/public/assets/PipelineEditorPage-B9GgqAju.js +0 -3
- package/public/assets/ProfilePage-DASpeJq6.js +0 -1
- package/public/assets/ProjectDetailPage-DWZyNwTV.js +0 -7
- package/public/assets/ProjectListPage-BUMX3Dxa.js +0 -6
- package/public/assets/QuickAuth-B9mvq7ht.js +0 -1
- package/public/assets/Select-A7PXobk_.js +0 -6
- package/public/assets/SettingsPage-WGzxUbGp.js +0 -6
- package/public/assets/SkillPage-BIt2kF3W.js +0 -1
- package/public/assets/TaskDetailPage-CePyNfp6.js +0 -44
- package/public/assets/TaskListPage-MO4_PSve.js +0 -1
- package/public/assets/TaskStatusBadge-QtQUDscM.js +0 -1
- package/public/assets/TokenManage-Sf0RGymw.js +0 -1
- package/public/assets/TotpSetupPage-D-rbEYLf.js +0 -9
- package/public/assets/WorkerSetupGuidePage-D6Fv1MQo.js +0 -16
- package/public/assets/index-CDBuOPx4.js +0 -212
- package/public/assets/index-CQojj7Zu.css +0 -1
- package/public/assets/protocol-C5uQmiiB.js +0 -1
- package/public/assets/task.store-CvjSr507.js +0 -1
- /package/public/assets/{TaskDetailPage-Beg8tuEN.css → task-constants-Beg8tuEN.css} +0 -0
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { TunnelStatus, TunnelClosedFrame } from '@overlordai/protocol';
|
|
2
|
+
import { TaskRepository } from '../database/repositories/task.repository';
|
|
3
|
+
import { WorkerConnectionManager } from '../dispatcher/worker-connection.manager';
|
|
4
|
+
interface TunnelState {
|
|
5
|
+
status: TunnelStatus;
|
|
6
|
+
taskId: number;
|
|
7
|
+
tunnelUrl?: string;
|
|
8
|
+
expiresAt?: string;
|
|
9
|
+
expiryTimer?: NodeJS.Timeout;
|
|
10
|
+
}
|
|
11
|
+
export declare class TunnelService {
|
|
12
|
+
private readonly taskRepo;
|
|
13
|
+
private readonly workerConnectionManager;
|
|
14
|
+
private readonly logger;
|
|
15
|
+
/**
|
|
16
|
+
* In-memory tunnel state per taskId.
|
|
17
|
+
* NOTE: This state is lost on server restart. This is acceptable because
|
|
18
|
+
* tunnels have a TTL and self-expire, and Workers will detect stale tunnels
|
|
19
|
+
* via heartbeat. After a restart, getTunnelStatus returns 'IDLE' which
|
|
20
|
+
* allows clients to start a new tunnel.
|
|
21
|
+
*/
|
|
22
|
+
private readonly tunnels;
|
|
23
|
+
private readonly tunnelLocks;
|
|
24
|
+
constructor(taskRepo: TaskRepository, workerConnectionManager: WorkerConnectionManager);
|
|
25
|
+
getTunnels(): Map<number, TunnelState>;
|
|
26
|
+
getTunnelLocks(): Set<number>;
|
|
27
|
+
/**
|
|
28
|
+
* Handle a tunnel_started event from the Worker (called by WorkerChannelGateway).
|
|
29
|
+
* Transitions the tunnel state to CONNECTED and starts the expiry timer.
|
|
30
|
+
*/
|
|
31
|
+
handleTunnelStarted(taskId: number, tunnelUrl: string): void;
|
|
32
|
+
/**
|
|
33
|
+
* Handle a tunnel_closed event from the Worker (called by WorkerChannelGateway).
|
|
34
|
+
* Worker-side tunnel process exited unexpectedly.
|
|
35
|
+
*/
|
|
36
|
+
handleTunnelClosed(taskId: number, _reason: TunnelClosedFrame['reason']): void;
|
|
37
|
+
/**
|
|
38
|
+
* Handle a tunnel_crashed event from the Worker.
|
|
39
|
+
* If the Worker will retry, keep state unchanged (tunnel may recover).
|
|
40
|
+
* If it won't retry, clean up the tunnel state.
|
|
41
|
+
*/
|
|
42
|
+
handleTunnelCrashed(taskId: number, error: string, willRetry: boolean): void;
|
|
43
|
+
/**
|
|
44
|
+
* Reconcile server-side tunnel state with the worker's reported active tunnels.
|
|
45
|
+
* Called during heartbeat to keep tunnel state in sync.
|
|
46
|
+
*
|
|
47
|
+
* - Restores tunnels the worker has but the server lost (e.g. after restart).
|
|
48
|
+
* - Cleans up tunnels the server tracks but the worker no longer has.
|
|
49
|
+
*/
|
|
50
|
+
reconcileTunnels(workerId: string, activeTunnels: Array<{
|
|
51
|
+
taskId: number;
|
|
52
|
+
tunnelUrl: string;
|
|
53
|
+
}>): void;
|
|
54
|
+
/**
|
|
55
|
+
* Clean up all tunnels for a given worker.
|
|
56
|
+
* Called when a worker disconnects to avoid stale tunnel state.
|
|
57
|
+
*/
|
|
58
|
+
cleanupTunnelsForWorker(workerId: string): void;
|
|
59
|
+
/**
|
|
60
|
+
* Force-close all tunnels for a task (called when task reaches terminal state).
|
|
61
|
+
*/
|
|
62
|
+
cleanupTunnelForTask(taskId: number, workerId?: string | null): Promise<void>;
|
|
63
|
+
getState(taskId: number): TunnelState | undefined;
|
|
64
|
+
transitionToConnected(taskId: number, tunnelUrl: string): void;
|
|
65
|
+
setTunnelState(taskId: number, state: TunnelState): void;
|
|
66
|
+
deleteTunnelState(taskId: number): void;
|
|
67
|
+
resetTunnel(tunnel: TunnelState): void;
|
|
68
|
+
/**
|
|
69
|
+
* Handle tunnel expiry: transition to EXPIRED, then send stop_tunnel to Worker.
|
|
70
|
+
*/
|
|
71
|
+
private handleTunnelExpired;
|
|
72
|
+
}
|
|
73
|
+
export {};
|
|
74
|
+
//# sourceMappingURL=tunnel.service.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tunnel.service.d.ts","sourceRoot":"","sources":["../../src/web/tunnel.service.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,YAAY,EAAE,iBAAiB,EAAmB,MAAM,sBAAsB,CAAC;AACxF,OAAO,EAAE,cAAc,EAAE,MAAM,0CAA0C,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,yCAAyC,CAAC;AAQlF,UAAU,WAAW;IACnB,MAAM,EAAE,YAAY,CAAC;IACrB,MAAM,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,WAAW,CAAC,EAAE,MAAM,CAAC,OAAO,CAAC;CAC9B;AAED,qBACa,aAAa;IAatB,OAAO,CAAC,QAAQ,CAAC,QAAQ;IACzB,OAAO,CAAC,QAAQ,CAAC,uBAAuB;IAb1C,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkC;IACzD;;;;;;OAMG;IACH,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;IAC1D,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAqB;gBAG9B,QAAQ,EAAE,cAAc,EACxB,uBAAuB,EAAE,uBAAuB;IAGnE,UAAU,IAAI,GAAG,CAAC,MAAM,EAAE,WAAW,CAAC;IAItC,cAAc,IAAI,GAAG,CAAC,MAAM,CAAC;IAI7B;;;OAGG;IACH,mBAAmB,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI;IAI5D;;;OAGG;IACH,kBAAkB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,iBAAiB,CAAC,QAAQ,CAAC,GAAG,IAAI;IAQ9E;;;;OAIG;IACH,mBAAmB,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,OAAO,GAAG,IAAI;IAkB5E;;;;;;OAMG;IACH,gBAAgB,CACd,QAAQ,EAAE,MAAM,EAChB,aAAa,EAAE,KAAK,CAAC;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,CAAC,GAC1D,IAAI;IA+BP;;;OAGG;IACH,uBAAuB,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAc/C;;OAEG;IACG,oBAAoB,CAAC,MAAM,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;IA0BnF,QAAQ,CAAC,MAAM,EAAE,MAAM,GAAG,WAAW,GAAG,SAAS;IAIjD,qBAAqB,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI;IAyB9D,cAAc,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,GAAG,IAAI;IAIxD,iBAAiB,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAIvC,WAAW,CAAC,MAAM,EAAE,WAAW,GAAG,IAAI;IAQtC;;OAEG;YACW,mBAAmB;CA+BlC"}
|
|
@@ -0,0 +1,250 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
19
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
20
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
21
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
22
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
23
|
+
};
|
|
24
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
25
|
+
var ownKeys = function(o) {
|
|
26
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
27
|
+
var ar = [];
|
|
28
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
29
|
+
return ar;
|
|
30
|
+
};
|
|
31
|
+
return ownKeys(o);
|
|
32
|
+
};
|
|
33
|
+
return function (mod) {
|
|
34
|
+
if (mod && mod.__esModule) return mod;
|
|
35
|
+
var result = {};
|
|
36
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
37
|
+
__setModuleDefault(result, mod);
|
|
38
|
+
return result;
|
|
39
|
+
};
|
|
40
|
+
})();
|
|
41
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
42
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
43
|
+
};
|
|
44
|
+
var TunnelService_1;
|
|
45
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
46
|
+
exports.TunnelService = void 0;
|
|
47
|
+
const common_1 = require("@nestjs/common");
|
|
48
|
+
const crypto = __importStar(require("node:crypto"));
|
|
49
|
+
const protocol_1 = require("@overlordai/protocol");
|
|
50
|
+
const task_repository_1 = require("../database/repositories/task.repository");
|
|
51
|
+
const worker_connection_manager_1 = require("../dispatcher/worker-connection.manager");
|
|
52
|
+
/** Default tunnel TTL in hours, configurable via TUNNEL_TTL_HOURS env var. */
|
|
53
|
+
const TUNNEL_TTL_HOURS = Math.min(parseInt(process.env.TUNNEL_TTL_HOURS || '4', 10) || 4, 24);
|
|
54
|
+
let TunnelService = TunnelService_1 = class TunnelService {
|
|
55
|
+
taskRepo;
|
|
56
|
+
workerConnectionManager;
|
|
57
|
+
logger = new common_1.Logger(TunnelService_1.name);
|
|
58
|
+
/**
|
|
59
|
+
* In-memory tunnel state per taskId.
|
|
60
|
+
* NOTE: This state is lost on server restart. This is acceptable because
|
|
61
|
+
* tunnels have a TTL and self-expire, and Workers will detect stale tunnels
|
|
62
|
+
* via heartbeat. After a restart, getTunnelStatus returns 'IDLE' which
|
|
63
|
+
* allows clients to start a new tunnel.
|
|
64
|
+
*/
|
|
65
|
+
tunnels = new Map();
|
|
66
|
+
tunnelLocks = new Set();
|
|
67
|
+
constructor(taskRepo, workerConnectionManager) {
|
|
68
|
+
this.taskRepo = taskRepo;
|
|
69
|
+
this.workerConnectionManager = workerConnectionManager;
|
|
70
|
+
}
|
|
71
|
+
getTunnels() {
|
|
72
|
+
return this.tunnels;
|
|
73
|
+
}
|
|
74
|
+
getTunnelLocks() {
|
|
75
|
+
return this.tunnelLocks;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Handle a tunnel_started event from the Worker (called by WorkerChannelGateway).
|
|
79
|
+
* Transitions the tunnel state to CONNECTED and starts the expiry timer.
|
|
80
|
+
*/
|
|
81
|
+
handleTunnelStarted(taskId, tunnelUrl) {
|
|
82
|
+
this.transitionToConnected(taskId, tunnelUrl);
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Handle a tunnel_closed event from the Worker (called by WorkerChannelGateway).
|
|
86
|
+
* Worker-side tunnel process exited unexpectedly.
|
|
87
|
+
*/
|
|
88
|
+
handleTunnelClosed(taskId, _reason) {
|
|
89
|
+
const tunnel = this.tunnels.get(taskId);
|
|
90
|
+
if (!tunnel)
|
|
91
|
+
return;
|
|
92
|
+
this.resetTunnel(tunnel);
|
|
93
|
+
this.logger.log(`Tunnel for task #${taskId} closed (${_reason})`);
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Handle a tunnel_crashed event from the Worker.
|
|
97
|
+
* If the Worker will retry, keep state unchanged (tunnel may recover).
|
|
98
|
+
* If it won't retry, clean up the tunnel state.
|
|
99
|
+
*/
|
|
100
|
+
handleTunnelCrashed(taskId, error, willRetry) {
|
|
101
|
+
if (willRetry) {
|
|
102
|
+
this.logger.warn(`Tunnel for task #${taskId} crashed (will retry): ${error}`);
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
this.logger.error(`Tunnel for task #${taskId} crashed (no retry): ${error}`);
|
|
106
|
+
const tunnel = this.tunnels.get(taskId);
|
|
107
|
+
if (!tunnel)
|
|
108
|
+
return;
|
|
109
|
+
this.resetTunnel(tunnel);
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Reconcile server-side tunnel state with the worker's reported active tunnels.
|
|
113
|
+
* Called during heartbeat to keep tunnel state in sync.
|
|
114
|
+
*
|
|
115
|
+
* - Restores tunnels the worker has but the server lost (e.g. after restart).
|
|
116
|
+
* - Cleans up tunnels the server tracks but the worker no longer has.
|
|
117
|
+
*/
|
|
118
|
+
reconcileTunnels(workerId, activeTunnels) {
|
|
119
|
+
const workerTaskIds = new Set(activeTunnels.map((t) => t.taskId));
|
|
120
|
+
// Restore tunnels the worker has but the server doesn't
|
|
121
|
+
for (const { taskId, tunnelUrl } of activeTunnels) {
|
|
122
|
+
const existing = this.tunnels.get(taskId);
|
|
123
|
+
if (!existing) {
|
|
124
|
+
this.logger.log(`Restoring tunnel for task #${taskId} from worker ${workerId}`);
|
|
125
|
+
const tunnel = { status: protocol_1.TunnelStatus.STARTING, taskId };
|
|
126
|
+
this.tunnels.set(taskId, tunnel);
|
|
127
|
+
this.transitionToConnected(taskId, tunnelUrl);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
// Clean up tunnels the server tracks but the worker no longer reports
|
|
131
|
+
// Collect entries first to avoid mutating the Map during iteration
|
|
132
|
+
const stale = [...this.tunnels.entries()].filter(([taskId]) => {
|
|
133
|
+
if (workerTaskIds.has(taskId))
|
|
134
|
+
return false;
|
|
135
|
+
const task = this.taskRepo.findById(taskId);
|
|
136
|
+
return task?.workerId === workerId;
|
|
137
|
+
});
|
|
138
|
+
for (const [taskId, tunnel] of stale) {
|
|
139
|
+
this.logger.log(`Cleaning up stale tunnel for task #${taskId} (not reported by worker ${workerId})`);
|
|
140
|
+
this.resetTunnel(tunnel);
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Clean up all tunnels for a given worker.
|
|
145
|
+
* Called when a worker disconnects to avoid stale tunnel state.
|
|
146
|
+
*/
|
|
147
|
+
cleanupTunnelsForWorker(workerId) {
|
|
148
|
+
// Collect entries first to avoid mutating the Map during iteration
|
|
149
|
+
const toClean = [...this.tunnels.entries()].filter(([taskId]) => {
|
|
150
|
+
const task = this.taskRepo.findById(taskId);
|
|
151
|
+
return task?.workerId === workerId;
|
|
152
|
+
});
|
|
153
|
+
for (const [taskId, tunnel] of toClean) {
|
|
154
|
+
this.logger.log(`Cleaning up tunnel for task #${taskId} (worker ${workerId} disconnected)`);
|
|
155
|
+
this.resetTunnel(tunnel);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Force-close all tunnels for a task (called when task reaches terminal state).
|
|
160
|
+
*/
|
|
161
|
+
async cleanupTunnelForTask(taskId, workerId) {
|
|
162
|
+
const tunnel = this.tunnels.get(taskId);
|
|
163
|
+
if (!tunnel || tunnel.status === protocol_1.TunnelStatus.IDLE || tunnel.status === protocol_1.TunnelStatus.CLOSED) {
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
// Send stop_tunnel to worker (fire-and-forget)
|
|
167
|
+
if (workerId && this.workerConnectionManager.isConnected(workerId)) {
|
|
168
|
+
const msgId = crypto.randomUUID();
|
|
169
|
+
this.workerConnectionManager.send(workerId, {
|
|
170
|
+
type: 'stop_tunnel',
|
|
171
|
+
msgId,
|
|
172
|
+
taskId,
|
|
173
|
+
reason: 'task_ended',
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
this.resetTunnel(tunnel);
|
|
177
|
+
this.logger.log(`Tunnel for task #${taskId} force-closed (task ended)`);
|
|
178
|
+
}
|
|
179
|
+
// ---------------------------------------------------------------------------
|
|
180
|
+
// Internal helpers used by WorkspaceController REST endpoints
|
|
181
|
+
// ---------------------------------------------------------------------------
|
|
182
|
+
getState(taskId) {
|
|
183
|
+
return this.tunnels.get(taskId);
|
|
184
|
+
}
|
|
185
|
+
transitionToConnected(taskId, tunnelUrl) {
|
|
186
|
+
const expiresAt = new Date(Date.now() + TUNNEL_TTL_HOURS * 60 * 60 * 1000).toISOString();
|
|
187
|
+
let tunnel = this.tunnels.get(taskId);
|
|
188
|
+
if (!tunnel) {
|
|
189
|
+
tunnel = { status: protocol_1.TunnelStatus.STARTING, taskId };
|
|
190
|
+
this.tunnels.set(taskId, tunnel);
|
|
191
|
+
}
|
|
192
|
+
tunnel.status = protocol_1.TunnelStatus.CONNECTED;
|
|
193
|
+
tunnel.tunnelUrl = tunnelUrl;
|
|
194
|
+
tunnel.expiresAt = expiresAt;
|
|
195
|
+
// Start expiry timer
|
|
196
|
+
tunnel.expiryTimer = setTimeout(() => {
|
|
197
|
+
void this.handleTunnelExpired(taskId);
|
|
198
|
+
}, TUNNEL_TTL_HOURS * 60 * 60 * 1000);
|
|
199
|
+
this.logger.log(`Tunnel for task #${taskId} connected: ${tunnelUrl} (expires ${expiresAt})`);
|
|
200
|
+
}
|
|
201
|
+
setTunnelState(taskId, state) {
|
|
202
|
+
this.tunnels.set(taskId, state);
|
|
203
|
+
}
|
|
204
|
+
deleteTunnelState(taskId) {
|
|
205
|
+
this.tunnels.delete(taskId);
|
|
206
|
+
}
|
|
207
|
+
resetTunnel(tunnel) {
|
|
208
|
+
if (tunnel.expiryTimer) {
|
|
209
|
+
clearTimeout(tunnel.expiryTimer);
|
|
210
|
+
tunnel.expiryTimer = undefined;
|
|
211
|
+
}
|
|
212
|
+
this.tunnels.delete(tunnel.taskId);
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Handle tunnel expiry: transition to EXPIRED, then send stop_tunnel to Worker.
|
|
216
|
+
*/
|
|
217
|
+
async handleTunnelExpired(taskId) {
|
|
218
|
+
const tunnel = this.tunnels.get(taskId);
|
|
219
|
+
if (!tunnel || tunnel.status !== protocol_1.TunnelStatus.CONNECTED)
|
|
220
|
+
return;
|
|
221
|
+
tunnel.status = protocol_1.TunnelStatus.EXPIRED;
|
|
222
|
+
this.logger.log(`Tunnel for task #${taskId} expired`);
|
|
223
|
+
// Find the task to get the workerId
|
|
224
|
+
const task = this.taskRepo.findById(taskId);
|
|
225
|
+
if (task?.workerId && this.workerConnectionManager.isConnected(task.workerId)) {
|
|
226
|
+
tunnel.status = protocol_1.TunnelStatus.CLOSING;
|
|
227
|
+
const msgId = crypto.randomUUID();
|
|
228
|
+
try {
|
|
229
|
+
await this.workerConnectionManager.sendWithAck(task.workerId, {
|
|
230
|
+
type: 'stop_tunnel',
|
|
231
|
+
msgId,
|
|
232
|
+
taskId,
|
|
233
|
+
reason: 'expired',
|
|
234
|
+
}, 15_000);
|
|
235
|
+
}
|
|
236
|
+
catch {
|
|
237
|
+
// Best effort
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
this.resetTunnel(tunnel);
|
|
241
|
+
this.logger.log(`Tunnel for task #${taskId} closed after expiry`);
|
|
242
|
+
}
|
|
243
|
+
};
|
|
244
|
+
exports.TunnelService = TunnelService;
|
|
245
|
+
exports.TunnelService = TunnelService = TunnelService_1 = __decorate([
|
|
246
|
+
(0, common_1.Injectable)(),
|
|
247
|
+
__metadata("design:paramtypes", [task_repository_1.TaskRepository,
|
|
248
|
+
worker_connection_manager_1.WorkerConnectionManager])
|
|
249
|
+
], TunnelService);
|
|
250
|
+
//# sourceMappingURL=tunnel.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tunnel.service.js","sourceRoot":"","sources":["../../src/web/tunnel.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { WebSocket } from 'ws';
|
|
2
|
+
import { type TaskStatusChangedPayload, type TaskCreatedPayload, type WorkerStatusChangedPayload } from '../events/event-types';
|
|
3
|
+
/**
|
|
4
|
+
* Lightweight in-process event broadcaster for web clients connected
|
|
5
|
+
* via the EventGateway (/ws/events).
|
|
6
|
+
*
|
|
7
|
+
* Since Overlord runs as a single server instance, we keep an in-memory
|
|
8
|
+
* set of authenticated WebSocket connections and broadcast to all of them.
|
|
9
|
+
*
|
|
10
|
+
* Subscribes to domain events emitted via EventEmitter2 so that producers
|
|
11
|
+
* (TaskCreationService, TaskLifecycleService, etc.) do not need a direct
|
|
12
|
+
* reference to this service.
|
|
13
|
+
*/
|
|
14
|
+
export declare class WebEventService {
|
|
15
|
+
private readonly logger;
|
|
16
|
+
private readonly clients;
|
|
17
|
+
register(ws: WebSocket): void;
|
|
18
|
+
unregister(ws: WebSocket): void;
|
|
19
|
+
get clientCount(): number;
|
|
20
|
+
handleTaskCreated(p: TaskCreatedPayload): void;
|
|
21
|
+
handleTaskStatusChanged(p: TaskStatusChangedPayload): void;
|
|
22
|
+
handleWorkerStatusChanged(p: WorkerStatusChangedPayload): void;
|
|
23
|
+
private broadcast;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=web-event.service.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"web-event.service.d.ts","sourceRoot":"","sources":["../../src/web/web-event.service.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAQ/B,OAAO,EAEL,KAAK,wBAAwB,EAC7B,KAAK,kBAAkB,EACvB,KAAK,0BAA0B,EAChC,MAAM,uBAAuB,CAAC;AAE/B;;;;;;;;;;GAUG;AACH,qBACa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAoC;IAC3D,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAwB;IAEhD,QAAQ,CAAC,EAAE,EAAE,SAAS,GAAG,IAAI;IAI7B,UAAU,CAAC,EAAE,EAAE,SAAS,GAAG,IAAI;IAI/B,IAAI,WAAW,IAAI,MAAM,CAExB;IAKD,iBAAiB,CAAC,CAAC,EAAE,kBAAkB,GAAG,IAAI;IAY9C,uBAAuB,CAAC,CAAC,EAAE,wBAAwB,GAAG,IAAI;IAa1D,yBAAyB,CAAC,CAAC,EAAE,0BAA0B,GAAG,IAAI;IAY9D,OAAO,CAAC,SAAS;CAwBlB"}
|
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
3
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
4
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
5
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
6
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
7
|
+
};
|
|
8
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
9
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
10
|
+
};
|
|
11
|
+
var WebEventService_1;
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.WebEventService = void 0;
|
|
14
|
+
const common_1 = require("@nestjs/common");
|
|
15
|
+
const event_emitter_1 = require("@nestjs/event-emitter");
|
|
16
|
+
const ws_1 = require("ws");
|
|
17
|
+
const event_types_1 = require("../events/event-types");
|
|
18
|
+
/**
|
|
19
|
+
* Lightweight in-process event broadcaster for web clients connected
|
|
20
|
+
* via the EventGateway (/ws/events).
|
|
21
|
+
*
|
|
22
|
+
* Since Overlord runs as a single server instance, we keep an in-memory
|
|
23
|
+
* set of authenticated WebSocket connections and broadcast to all of them.
|
|
24
|
+
*
|
|
25
|
+
* Subscribes to domain events emitted via EventEmitter2 so that producers
|
|
26
|
+
* (TaskCreationService, TaskLifecycleService, etc.) do not need a direct
|
|
27
|
+
* reference to this service.
|
|
28
|
+
*/
|
|
29
|
+
let WebEventService = WebEventService_1 = class WebEventService {
|
|
30
|
+
logger = new common_1.Logger(WebEventService_1.name);
|
|
31
|
+
clients = new Set();
|
|
32
|
+
register(ws) {
|
|
33
|
+
this.clients.add(ws);
|
|
34
|
+
}
|
|
35
|
+
unregister(ws) {
|
|
36
|
+
this.clients.delete(ws);
|
|
37
|
+
}
|
|
38
|
+
get clientCount() {
|
|
39
|
+
return this.clients.size;
|
|
40
|
+
}
|
|
41
|
+
// ─── Domain event subscribers ───────────────────────────────────────
|
|
42
|
+
handleTaskCreated(p) {
|
|
43
|
+
this.broadcast({
|
|
44
|
+
type: 'event',
|
|
45
|
+
event: 'task_created',
|
|
46
|
+
taskId: p.taskId,
|
|
47
|
+
status: p.status,
|
|
48
|
+
projectKey: p.projectKey,
|
|
49
|
+
description: p.description,
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
handleTaskStatusChanged(p) {
|
|
53
|
+
this.broadcast({
|
|
54
|
+
type: 'event',
|
|
55
|
+
event: 'task_status_changed',
|
|
56
|
+
taskId: p.taskId,
|
|
57
|
+
status: p.status,
|
|
58
|
+
previousStatus: p.previousStatus,
|
|
59
|
+
workerId: p.workerId,
|
|
60
|
+
projectKey: p.projectKey,
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
handleWorkerStatusChanged(p) {
|
|
64
|
+
this.broadcast({
|
|
65
|
+
type: 'event',
|
|
66
|
+
event: 'worker_status_changed',
|
|
67
|
+
workerId: p.workerId,
|
|
68
|
+
status: p.status,
|
|
69
|
+
previousStatus: p.previousStatus,
|
|
70
|
+
});
|
|
71
|
+
}
|
|
72
|
+
// ─── Internal ──────────────────────────────────────────────────────
|
|
73
|
+
broadcast(frame) {
|
|
74
|
+
const json = JSON.stringify(frame);
|
|
75
|
+
let sent = 0;
|
|
76
|
+
const stale = [];
|
|
77
|
+
for (const ws of this.clients) {
|
|
78
|
+
if (ws.readyState === ws_1.WebSocket.OPEN) {
|
|
79
|
+
ws.send(json);
|
|
80
|
+
sent++;
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
stale.push(ws);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
for (const ws of stale) {
|
|
87
|
+
this.clients.delete(ws);
|
|
88
|
+
}
|
|
89
|
+
if (sent > 0) {
|
|
90
|
+
this.logger.debug(`Broadcast event to ${sent} client(s): ${frame.event}`);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
exports.WebEventService = WebEventService;
|
|
95
|
+
__decorate([
|
|
96
|
+
(0, event_emitter_1.OnEvent)(event_types_1.DomainEvents.TASK_CREATED),
|
|
97
|
+
__metadata("design:type", Function),
|
|
98
|
+
__metadata("design:paramtypes", [Object]),
|
|
99
|
+
__metadata("design:returntype", void 0)
|
|
100
|
+
], WebEventService.prototype, "handleTaskCreated", null);
|
|
101
|
+
__decorate([
|
|
102
|
+
(0, event_emitter_1.OnEvent)(event_types_1.DomainEvents.TASK_STATUS_CHANGED),
|
|
103
|
+
__metadata("design:type", Function),
|
|
104
|
+
__metadata("design:paramtypes", [Object]),
|
|
105
|
+
__metadata("design:returntype", void 0)
|
|
106
|
+
], WebEventService.prototype, "handleTaskStatusChanged", null);
|
|
107
|
+
__decorate([
|
|
108
|
+
(0, event_emitter_1.OnEvent)(event_types_1.DomainEvents.WORKER_STATUS_CHANGED),
|
|
109
|
+
__metadata("design:type", Function),
|
|
110
|
+
__metadata("design:paramtypes", [Object]),
|
|
111
|
+
__metadata("design:returntype", void 0)
|
|
112
|
+
], WebEventService.prototype, "handleWorkerStatusChanged", null);
|
|
113
|
+
exports.WebEventService = WebEventService = WebEventService_1 = __decorate([
|
|
114
|
+
(0, common_1.Injectable)()
|
|
115
|
+
], WebEventService);
|
|
116
|
+
//# sourceMappingURL=web-event.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"web-event.service.js","sourceRoot":"","sources":["../../src/web/web-event.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"web.module.d.ts","sourceRoot":"","sources":["../../src/web/web.module.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"web.module.d.ts","sourceRoot":"","sources":["../../src/web/web.module.ts"],"names":[],"mappings":"AAsCA,qBAkCa,SAAS;CAAG"}
|
package/dist/web/web.module.js
CHANGED
|
@@ -11,22 +11,10 @@ const common_1 = require("@nestjs/common");
|
|
|
11
11
|
const auth_module_1 = require("../auth/auth.module");
|
|
12
12
|
const dispatcher_module_1 = require("../dispatcher/dispatcher.module");
|
|
13
13
|
const crypto_service_1 = require("../common/crypto.service");
|
|
14
|
-
// Repositories
|
|
15
|
-
const task_repository_1 = require("../database/repositories/task.repository");
|
|
16
|
-
const project_repository_1 = require("../database/repositories/project.repository");
|
|
17
|
-
const machine_repository_1 = require("../database/repositories/machine.repository");
|
|
18
|
-
const developer_repository_1 = require("../database/repositories/developer.repository");
|
|
19
|
-
const notification_repository_1 = require("../database/repositories/notification.repository");
|
|
20
|
-
const workspace_repository_1 = require("../database/repositories/workspace.repository");
|
|
21
|
-
const audit_log_repository_1 = require("../database/repositories/audit-log.repository");
|
|
22
|
-
const bot_repository_1 = require("../database/repositories/bot.repository");
|
|
23
|
-
const project_member_repository_1 = require("../database/repositories/project-member.repository");
|
|
24
|
-
const developer_token_repository_1 = require("../database/repositories/developer-token.repository");
|
|
25
|
-
const worker_token_repository_1 = require("../database/repositories/worker-token.repository");
|
|
26
14
|
// Web controllers
|
|
27
15
|
const task_controller_1 = require("./task.controller");
|
|
28
16
|
const project_controller_1 = require("./project.controller");
|
|
29
|
-
const
|
|
17
|
+
const worker_web_controller_1 = require("./worker-web.controller");
|
|
30
18
|
const workspace_controller_1 = require("./workspace.controller");
|
|
31
19
|
const notification_controller_1 = require("./notification.controller");
|
|
32
20
|
const profile_controller_1 = require("./profile.controller");
|
|
@@ -36,7 +24,7 @@ const health_controller_1 = require("../common/health.controller");
|
|
|
36
24
|
// Admin controllers
|
|
37
25
|
const admin_project_controller_1 = require("./admin/admin-project.controller");
|
|
38
26
|
const admin_developer_controller_1 = require("./admin/admin-developer.controller");
|
|
39
|
-
const
|
|
27
|
+
const admin_worker_controller_1 = require("./admin/admin-worker.controller");
|
|
40
28
|
const admin_token_controller_1 = require("./admin/admin-token.controller");
|
|
41
29
|
const admin_bot_controller_1 = require("./admin/admin-bot.controller");
|
|
42
30
|
const admin_audit_controller_1 = require("./admin/admin-audit.controller");
|
|
@@ -44,10 +32,14 @@ const admin_settings_controller_1 = require("./admin/admin-settings.controller")
|
|
|
44
32
|
// WebSocket gateways
|
|
45
33
|
const worker_channel_gateway_1 = require("./worker-channel.gateway");
|
|
46
34
|
const pty_gateway_1 = require("./pty.gateway");
|
|
35
|
+
const event_gateway_1 = require("./event.gateway");
|
|
47
36
|
// Services
|
|
48
37
|
const interaction_service_1 = require("./interaction.service");
|
|
49
38
|
const dashboard_service_1 = require("./dashboard.service");
|
|
50
39
|
const search_service_1 = require("./search.service");
|
|
40
|
+
const web_event_service_1 = require("./web-event.service");
|
|
41
|
+
const tunnel_service_1 = require("./tunnel.service");
|
|
42
|
+
const project_member_service_1 = require("./project-member.service");
|
|
51
43
|
let WebModule = class WebModule {
|
|
52
44
|
};
|
|
53
45
|
exports.WebModule = WebModule;
|
|
@@ -57,7 +49,7 @@ exports.WebModule = WebModule = __decorate([
|
|
|
57
49
|
controllers: [
|
|
58
50
|
task_controller_1.TaskController,
|
|
59
51
|
project_controller_1.ProjectController,
|
|
60
|
-
|
|
52
|
+
worker_web_controller_1.WorkerWebController,
|
|
61
53
|
workspace_controller_1.WorkspaceController,
|
|
62
54
|
notification_controller_1.NotificationController,
|
|
63
55
|
profile_controller_1.ProfileController,
|
|
@@ -66,7 +58,7 @@ exports.WebModule = WebModule = __decorate([
|
|
|
66
58
|
health_controller_1.HealthController,
|
|
67
59
|
admin_project_controller_1.AdminProjectController,
|
|
68
60
|
admin_developer_controller_1.AdminDeveloperController,
|
|
69
|
-
|
|
61
|
+
admin_worker_controller_1.AdminWorkerController,
|
|
70
62
|
admin_token_controller_1.AdminTokenController,
|
|
71
63
|
admin_bot_controller_1.AdminBotController,
|
|
72
64
|
admin_audit_controller_1.AdminAuditController,
|
|
@@ -74,24 +66,17 @@ exports.WebModule = WebModule = __decorate([
|
|
|
74
66
|
],
|
|
75
67
|
providers: [
|
|
76
68
|
crypto_service_1.CryptoService,
|
|
77
|
-
|
|
78
|
-
project_repository_1.ProjectRepository,
|
|
79
|
-
machine_repository_1.MachineRepository,
|
|
80
|
-
developer_repository_1.DeveloperRepository,
|
|
81
|
-
notification_repository_1.NotificationRepository,
|
|
82
|
-
workspace_repository_1.WorkspaceRepository,
|
|
83
|
-
audit_log_repository_1.AuditLogRepository,
|
|
84
|
-
bot_repository_1.BotRepository,
|
|
85
|
-
project_member_repository_1.ProjectMemberRepository,
|
|
86
|
-
developer_token_repository_1.DeveloperTokenRepository,
|
|
87
|
-
worker_token_repository_1.WorkerTokenRepository,
|
|
69
|
+
tunnel_service_1.TunnelService,
|
|
88
70
|
worker_channel_gateway_1.WorkerChannelGateway,
|
|
89
71
|
pty_gateway_1.PtyGateway,
|
|
72
|
+
event_gateway_1.EventGateway,
|
|
73
|
+
web_event_service_1.WebEventService,
|
|
90
74
|
interaction_service_1.InteractionService,
|
|
91
75
|
dashboard_service_1.DashboardService,
|
|
92
76
|
search_service_1.SearchService,
|
|
77
|
+
project_member_service_1.ProjectMemberService,
|
|
93
78
|
],
|
|
94
|
-
exports: [interaction_service_1.InteractionService, dashboard_service_1.DashboardService, search_service_1.SearchService],
|
|
79
|
+
exports: [interaction_service_1.InteractionService, dashboard_service_1.DashboardService, search_service_1.SearchService, web_event_service_1.WebEventService],
|
|
95
80
|
})
|
|
96
81
|
], WebModule);
|
|
97
82
|
//# sourceMappingURL=web.module.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"web.module.js","sourceRoot":"","sources":["../../src/web/web.module.ts"],"names":[],"mappings":";;;;;;;;;AAAA,2CAAwC;AACxC,qDAAiD;AACjD,uEAAmE;AACnE,6DAAyD;AAEzD,
|
|
1
|
+
{"version":3,"file":"web.module.js","sourceRoot":"","sources":["../../src/web/web.module.ts"],"names":[],"mappings":";;;;;;;;;AAAA,2CAAwC;AACxC,qDAAiD;AACjD,uEAAmE;AACnE,6DAAyD;AAEzD,kBAAkB;AAClB,uDAAmD;AACnD,6DAAyD;AACzD,mEAA8D;AAC9D,iEAA6D;AAC7D,uEAAmE;AACnE,6DAAyD;AACzD,2DAAuD;AACvD,iEAA6D;AAC7D,mEAA+D;AAE/D,oBAAoB;AACpB,+EAA0E;AAC1E,mFAA8E;AAC9E,6EAAwE;AACxE,2EAAsE;AACtE,uEAAkE;AAClE,2EAAsE;AACtE,iFAA4E;AAE5E,qBAAqB;AACrB,qEAAgE;AAChE,+CAA2C;AAC3C,mDAA+C;AAE/C,WAAW;AACX,+DAA2D;AAC3D,2DAAuD;AACvD,qDAAiD;AACjD,2DAAsD;AACtD,qDAAiD;AACjD,qEAAgE;AAoCzD,IAAM,SAAS,GAAf,MAAM,SAAS;CAAG,CAAA;AAAZ,8BAAS;oBAAT,SAAS;IAlCrB,IAAA,eAAM,EAAC;QACN,OAAO,EAAE,CAAC,wBAAU,EAAE,oCAAgB,CAAC;QACvC,WAAW,EAAE;YACX,gCAAc;YACd,sCAAiB;YACjB,2CAAmB;YACnB,0CAAmB;YACnB,gDAAsB;YACtB,sCAAiB;YACjB,oCAAgB;YAChB,0CAAmB;YACnB,oCAAgB;YAChB,iDAAsB;YACtB,qDAAwB;YACxB,+CAAqB;YACrB,6CAAoB;YACpB,yCAAkB;YAClB,6CAAoB;YACpB,mDAAuB;SACxB;QACD,SAAS,EAAE;YACT,8BAAa;YACb,8BAAa;YACb,6CAAoB;YACpB,wBAAU;YACV,4BAAY;YACZ,mCAAe;YACf,wCAAkB;YAClB,oCAAgB;YAChB,8BAAa;YACb,6CAAoB;SACrB;QACD,OAAO,EAAE,CAAC,wCAAkB,EAAE,oCAAgB,EAAE,8BAAa,EAAE,mCAAe,CAAC;KAChF,CAAC;GACW,SAAS,CAAG"}
|