opalserve 2.0.0 → 3.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.
Files changed (118) hide show
  1. package/README.md +493 -67
  2. package/assets/logo.svg +36 -39
  3. package/dist/auth/api-keys.d.ts +7 -0
  4. package/dist/auth/api-keys.d.ts.map +1 -0
  5. package/dist/auth/api-keys.js +12 -0
  6. package/dist/auth/api-keys.js.map +1 -0
  7. package/dist/auth/index.d.ts +4 -0
  8. package/dist/auth/index.d.ts.map +1 -0
  9. package/dist/auth/index.js +4 -0
  10. package/dist/auth/index.js.map +1 -0
  11. package/dist/auth/middleware.d.ts +11 -0
  12. package/dist/auth/middleware.d.ts.map +1 -0
  13. package/dist/auth/middleware.js +46 -0
  14. package/dist/auth/middleware.js.map +1 -0
  15. package/dist/auth/passwords.d.ts +3 -0
  16. package/dist/auth/passwords.d.ts.map +1 -0
  17. package/dist/auth/passwords.js +33 -0
  18. package/dist/auth/passwords.js.map +1 -0
  19. package/dist/cli/commands/admin.d.ts +13 -0
  20. package/dist/cli/commands/admin.d.ts.map +1 -0
  21. package/dist/cli/commands/admin.js +261 -0
  22. package/dist/cli/commands/admin.js.map +1 -0
  23. package/dist/cli/commands/auth.d.ts +4 -0
  24. package/dist/cli/commands/auth.d.ts.map +1 -0
  25. package/dist/cli/commands/auth.js +77 -0
  26. package/dist/cli/commands/auth.js.map +1 -0
  27. package/dist/cli/commands/context.d.ts +5 -0
  28. package/dist/cli/commands/context.d.ts.map +1 -0
  29. package/dist/cli/commands/context.js +162 -0
  30. package/dist/cli/commands/context.js.map +1 -0
  31. package/dist/cli/commands/sync.d.ts +2 -0
  32. package/dist/cli/commands/sync.d.ts.map +1 -0
  33. package/dist/cli/commands/sync.js +58 -0
  34. package/dist/cli/commands/sync.js.map +1 -0
  35. package/dist/cli/index.js +75 -1
  36. package/dist/cli/index.js.map +1 -1
  37. package/dist/cli/ui/banner.js +2 -2
  38. package/dist/config/credentials.d.ts +10 -0
  39. package/dist/config/credentials.d.ts.map +1 -0
  40. package/dist/config/credentials.js +33 -0
  41. package/dist/config/credentials.js.map +1 -0
  42. package/dist/config/defaults.d.ts.map +1 -1
  43. package/dist/config/defaults.js +2 -0
  44. package/dist/config/defaults.js.map +1 -1
  45. package/dist/constants.d.ts +5 -0
  46. package/dist/constants.d.ts.map +1 -0
  47. package/dist/constants.js +5 -0
  48. package/dist/constants.js.map +1 -0
  49. package/dist/context/chunker.d.ts +2 -0
  50. package/dist/context/chunker.d.ts.map +1 -0
  51. package/dist/context/chunker.js +81 -0
  52. package/dist/context/chunker.js.map +1 -0
  53. package/dist/context/index.d.ts +26 -0
  54. package/dist/context/index.d.ts.map +1 -0
  55. package/dist/context/index.js +97 -0
  56. package/dist/context/index.js.map +1 -0
  57. package/dist/core/registry.d.ts +3 -1
  58. package/dist/core/registry.d.ts.map +1 -1
  59. package/dist/core/registry.js +8 -6
  60. package/dist/core/registry.js.map +1 -1
  61. package/dist/core/secrets.d.ts +4 -0
  62. package/dist/core/secrets.d.ts.map +1 -0
  63. package/dist/core/secrets.js +40 -0
  64. package/dist/core/secrets.js.map +1 -0
  65. package/dist/core/server-manager.js +1 -1
  66. package/dist/dashboard/assets/index-BNOtcUPs.js +257 -0
  67. package/dist/dashboard/assets/index-Duwp34GW.css +1 -0
  68. package/dist/dashboard/index.html +14 -0
  69. package/dist/index.d.ts +13 -2
  70. package/dist/index.d.ts.map +1 -1
  71. package/dist/index.js +18 -1
  72. package/dist/index.js.map +1 -1
  73. package/dist/integrations/github.d.ts +5 -0
  74. package/dist/integrations/github.d.ts.map +1 -0
  75. package/dist/integrations/github.js +63 -0
  76. package/dist/integrations/github.js.map +1 -0
  77. package/dist/integrations/slack.d.ts +21 -0
  78. package/dist/integrations/slack.d.ts.map +1 -0
  79. package/dist/integrations/slack.js +61 -0
  80. package/dist/integrations/slack.js.map +1 -0
  81. package/dist/monitoring/tracker.d.ts +31 -0
  82. package/dist/monitoring/tracker.d.ts.map +1 -0
  83. package/dist/monitoring/tracker.js +86 -0
  84. package/dist/monitoring/tracker.js.map +1 -0
  85. package/dist/server/app.d.ts.map +1 -1
  86. package/dist/server/app.js +46 -0
  87. package/dist/server/app.js.map +1 -1
  88. package/dist/server/mcp-gateway.js +1 -1
  89. package/dist/server/routes/auth.d.ts +4 -0
  90. package/dist/server/routes/auth.d.ts.map +1 -0
  91. package/dist/server/routes/auth.js +117 -0
  92. package/dist/server/routes/auth.js.map +1 -0
  93. package/dist/server/routes/context.d.ts +4 -0
  94. package/dist/server/routes/context.d.ts.map +1 -0
  95. package/dist/server/routes/context.js +107 -0
  96. package/dist/server/routes/context.js.map +1 -0
  97. package/dist/server/routes/health.js +1 -1
  98. package/dist/server/routes/stats.d.ts +4 -0
  99. package/dist/server/routes/stats.d.ts.map +1 -0
  100. package/dist/server/routes/stats.js +97 -0
  101. package/dist/server/routes/stats.js.map +1 -0
  102. package/dist/server/routes/team-servers.d.ts +4 -0
  103. package/dist/server/routes/team-servers.d.ts.map +1 -0
  104. package/dist/server/routes/team-servers.js +108 -0
  105. package/dist/server/routes/team-servers.js.map +1 -0
  106. package/dist/server/routes/webhooks.d.ts +4 -0
  107. package/dist/server/routes/webhooks.d.ts.map +1 -0
  108. package/dist/server/routes/webhooks.js +77 -0
  109. package/dist/server/routes/webhooks.js.map +1 -0
  110. package/dist/storage/database.d.ts +10 -3
  111. package/dist/storage/database.d.ts.map +1 -1
  112. package/dist/storage/database.js +269 -115
  113. package/dist/storage/database.js.map +1 -1
  114. package/dist/types/index.d.ts +102 -0
  115. package/dist/types/index.d.ts.map +1 -1
  116. package/dist/types/index.js +8 -0
  117. package/dist/types/index.js.map +1 -1
  118. package/package.json +5 -3
@@ -0,0 +1 @@
1
+ *,:before,:after{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x: 0;--tw-border-spacing-y: 0;--tw-translate-x: 0;--tw-translate-y: 0;--tw-rotate: 0;--tw-skew-x: 0;--tw-skew-y: 0;--tw-scale-x: 1;--tw-scale-y: 1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness: proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width: 0px;--tw-ring-offset-color: #fff;--tw-ring-color: rgb(59 130 246 / .5);--tw-ring-offset-shadow: 0 0 #0000;--tw-ring-shadow: 0 0 #0000;--tw-shadow: 0 0 #0000;--tw-shadow-colored: 0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }*,:before,:after{box-sizing:border-box;border-width:0;border-style:solid;border-color:#e5e7eb}:before,:after{--tw-content: ""}html,:host{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji",Segoe UI Symbol,"Noto Color Emoji";font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dl,dd,h1,h2,h3,h4,h5,h6,hr,figure,p,pre{margin:0}fieldset{margin:0;padding:0}legend{padding:0}ol,ul,menu{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}button,[role=button]{cursor:pointer}:disabled{cursor:default}img,svg,video,canvas,audio,iframe,embed,object{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}.absolute{position:absolute}.relative{position:relative}.left-3{left:.75rem}.top-1\/2{top:50%}.mx-auto{margin-left:auto;margin-right:auto}.mb-1\.5{margin-bottom:.375rem}.mb-3{margin-bottom:.75rem}.mb-4{margin-bottom:1rem}.ml-2{margin-left:.5rem}.mt-0\.5{margin-top:.125rem}.mt-1{margin-top:.25rem}.mt-1\.5{margin-top:.375rem}.mt-2{margin-top:.5rem}.block{display:block}.inline-block{display:inline-block}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.grid{display:grid}.contents{display:contents}.h-1\.5{height:.375rem}.h-11{height:2.75rem}.h-16{height:4rem}.h-8{height:2rem}.h-screen{height:100vh}.w-1\.5{width:.375rem}.w-11{width:2.75rem}.w-64{width:16rem}.w-8{width:2rem}.w-full{width:100%}.min-w-0{min-width:0px}.max-w-3xl{max-width:48rem}.flex-1{flex:1 1 0%}.flex-shrink-0{flex-shrink:0}.-translate-y-1\/2{--tw-translate-y: -50%;transform:translate(var(--tw-translate-x),var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y))}@keyframes pulse{50%{opacity:.5}}.animate-pulse{animation:pulse 2s cubic-bezier(.4,0,.6,1) infinite}@keyframes spin{to{transform:rotate(360deg)}}.animate-spin{animation:spin 1s linear infinite}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.flex-col{flex-direction:column}.items-start{align-items:flex-start}.items-center{align-items:center}.justify-center{justify-content:center}.justify-between{justify-content:space-between}.gap-1{gap:.25rem}.gap-1\.5{gap:.375rem}.gap-2{gap:.5rem}.gap-3{gap:.75rem}.gap-4{gap:1rem}.space-y-1>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.25rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.25rem * var(--tw-space-y-reverse))}.space-y-2>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(.5rem * var(--tw-space-y-reverse))}.space-y-4>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1rem * var(--tw-space-y-reverse))}.space-y-6>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(1.5rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(1.5rem * var(--tw-space-y-reverse))}.space-y-8>:not([hidden])~:not([hidden]){--tw-space-y-reverse: 0;margin-top:calc(2rem * calc(1 - var(--tw-space-y-reverse)));margin-bottom:calc(2rem * var(--tw-space-y-reverse))}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.rounded{border-radius:.25rem}.rounded-full{border-radius:9999px}.rounded-lg{border-radius:.5rem}.rounded-xl{border-radius:.75rem}.border{border-width:1px}.border-b{border-bottom-width:1px}.border-r{border-right-width:1px}.border-t{border-top-width:1px}.border-blue-500\/30{border-color:#3b82f64d}.border-brand-amber\/30{border-color:#f973164d}.border-brand-purple\/30{border-color:#8b5cf64d}.border-emerald-500\/30{border-color:#10b9814d}.border-gray-700{--tw-border-opacity: 1;border-color:rgb(55 65 81 / var(--tw-border-opacity, 1))}.border-gray-700\/30{border-color:#3741514d}.border-gray-700\/50{border-color:#37415180}.border-gray-800{--tw-border-opacity: 1;border-color:rgb(31 41 55 / var(--tw-border-opacity, 1))}.bg-amber-400{--tw-bg-opacity: 1;background-color:rgb(251 191 36 / var(--tw-bg-opacity, 1))}.bg-amber-400\/10{background-color:#fbbf241a}.bg-blue-500\/5{background-color:#3b82f60d}.bg-brand-amber{--tw-bg-opacity: 1;background-color:rgb(249 115 22 / var(--tw-bg-opacity, 1))}.bg-brand-amber\/10{background-color:#f973161a}.bg-brand-amber\/15{background-color:#f9731626}.bg-brand-amber\/5{background-color:#f973160d}.bg-brand-purple\/15{background-color:#8b5cf626}.bg-brand-purple\/5{background-color:#8b5cf60d}.bg-emerald-400{--tw-bg-opacity: 1;background-color:rgb(52 211 153 / var(--tw-bg-opacity, 1))}.bg-emerald-400\/10{background-color:#34d3991a}.bg-emerald-500\/15{background-color:#10b98126}.bg-emerald-500\/5{background-color:#10b9810d}.bg-gray-400{--tw-bg-opacity: 1;background-color:rgb(156 163 175 / var(--tw-bg-opacity, 1))}.bg-gray-400\/10{background-color:#9ca3af1a}.bg-gray-600\/20{background-color:#4b556333}.bg-gray-700{--tw-bg-opacity: 1;background-color:rgb(55 65 81 / var(--tw-bg-opacity, 1))}.bg-gray-700\/50{background-color:#37415180}.bg-gray-800{--tw-bg-opacity: 1;background-color:rgb(31 41 55 / var(--tw-bg-opacity, 1))}.bg-gray-800\/30{background-color:#1f29374d}.bg-gray-800\/60{background-color:#1f293799}.bg-gray-900{--tw-bg-opacity: 1;background-color:rgb(17 24 39 / var(--tw-bg-opacity, 1))}.bg-gray-950{--tw-bg-opacity: 1;background-color:rgb(3 7 18 / var(--tw-bg-opacity, 1))}.bg-gray-950\/80{background-color:#030712cc}.bg-red-400{--tw-bg-opacity: 1;background-color:rgb(248 113 113 / var(--tw-bg-opacity, 1))}.bg-red-400\/10{background-color:#f871711a}.bg-red-500\/15{background-color:#ef444426}.bg-gradient-to-br{background-image:linear-gradient(to bottom right,var(--tw-gradient-stops))}.bg-gradient-to-r{background-image:linear-gradient(to right,var(--tw-gradient-stops))}.from-brand-amber{--tw-gradient-from: #F97316 var(--tw-gradient-from-position);--tw-gradient-to: rgb(249 115 22 / 0) var(--tw-gradient-to-position);--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to)}.to-brand-purple{--tw-gradient-to: #8B5CF6 var(--tw-gradient-to-position)}.bg-clip-text{-webkit-background-clip:text;background-clip:text}.p-2{padding:.5rem}.p-4{padding:1rem}.p-5{padding:1.25rem}.p-6{padding:1.5rem}.px-1\.5{padding-left:.375rem;padding-right:.375rem}.px-2{padding-left:.5rem;padding-right:.5rem}.px-2\.5{padding-left:.625rem;padding-right:.625rem}.px-3{padding-left:.75rem;padding-right:.75rem}.px-4{padding-left:1rem;padding-right:1rem}.px-6{padding-left:1.5rem;padding-right:1.5rem}.py-0\.5{padding-top:.125rem;padding-bottom:.125rem}.py-1{padding-top:.25rem;padding-bottom:.25rem}.py-1\.5{padding-top:.375rem;padding-bottom:.375rem}.py-12{padding-top:3rem;padding-bottom:3rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-2\.5{padding-top:.625rem;padding-bottom:.625rem}.py-3{padding-top:.75rem;padding-bottom:.75rem}.py-4{padding-top:1rem;padding-bottom:1rem}.py-8{padding-top:2rem;padding-bottom:2rem}.pl-10{padding-left:2.5rem}.pr-4{padding-right:1rem}.text-left{text-align:left}.text-center{text-align:center}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace}.text-2xl{font-size:1.5rem;line-height:2rem}.text-\[10px\]{font-size:10px}.text-lg{font-size:1.125rem;line-height:1.75rem}.text-sm{font-size:.875rem;line-height:1.25rem}.text-xl{font-size:1.25rem;line-height:1.75rem}.text-xs{font-size:.75rem;line-height:1rem}.font-bold{font-weight:700}.font-medium{font-weight:500}.font-normal{font-weight:400}.font-semibold{font-weight:600}.uppercase{text-transform:uppercase}.leading-relaxed{line-height:1.625}.tracking-wider{letter-spacing:.05em}.text-amber-400{--tw-text-opacity: 1;color:rgb(251 191 36 / var(--tw-text-opacity, 1))}.text-brand-amber{--tw-text-opacity: 1;color:rgb(249 115 22 / var(--tw-text-opacity, 1))}.text-brand-amber\/60{color:#f9731699}.text-brand-purple{--tw-text-opacity: 1;color:rgb(139 92 246 / var(--tw-text-opacity, 1))}.text-emerald-400{--tw-text-opacity: 1;color:rgb(52 211 153 / var(--tw-text-opacity, 1))}.text-gray-100{--tw-text-opacity: 1;color:rgb(243 244 246 / var(--tw-text-opacity, 1))}.text-gray-200{--tw-text-opacity: 1;color:rgb(229 231 235 / var(--tw-text-opacity, 1))}.text-gray-300{--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.text-gray-400{--tw-text-opacity: 1;color:rgb(156 163 175 / var(--tw-text-opacity, 1))}.text-gray-500{--tw-text-opacity: 1;color:rgb(107 114 128 / var(--tw-text-opacity, 1))}.text-gray-600{--tw-text-opacity: 1;color:rgb(75 85 99 / var(--tw-text-opacity, 1))}.text-red-400{--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.text-transparent{color:transparent}.text-white{--tw-text-opacity: 1;color:rgb(255 255 255 / var(--tw-text-opacity, 1))}.placeholder-gray-500::-moz-placeholder{--tw-placeholder-opacity: 1;color:rgb(107 114 128 / var(--tw-placeholder-opacity, 1))}.placeholder-gray-500::placeholder{--tw-placeholder-opacity: 1;color:rgb(107 114 128 / var(--tw-placeholder-opacity, 1))}.opacity-50{opacity:.5}.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.backdrop-blur{--tw-backdrop-blur: blur(8px);-webkit-backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia);backdrop-filter:var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia)}.transition-colors{transition-property:color,background-color,border-color,text-decoration-color,fill,stroke;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.15s}.hover\:bg-brand-orange:hover{--tw-bg-opacity: 1;background-color:rgb(234 88 12 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-600:hover{--tw-bg-opacity: 1;background-color:rgb(75 85 99 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-700:hover{--tw-bg-opacity: 1;background-color:rgb(55 65 81 / var(--tw-bg-opacity, 1))}.hover\:bg-gray-700\/20:hover{background-color:#37415133}.hover\:bg-gray-800\/50:hover{background-color:#1f293780}.hover\:bg-gray-800\/80:hover{background-color:#1f2937cc}.hover\:text-gray-200:hover{--tw-text-opacity: 1;color:rgb(229 231 235 / var(--tw-text-opacity, 1))}.hover\:text-gray-300:hover{--tw-text-opacity: 1;color:rgb(209 213 219 / var(--tw-text-opacity, 1))}.hover\:text-red-400:hover{--tw-text-opacity: 1;color:rgb(248 113 113 / var(--tw-text-opacity, 1))}.focus\:border-brand-amber\/50:focus{border-color:#f9731680}.focus\:outline-none:focus{outline:2px solid transparent;outline-offset:2px}.focus\:ring-1:focus{--tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow, 0 0 #0000)}.focus\:ring-brand-amber\/25:focus{--tw-ring-color: rgb(249 115 22 / .25)}.disabled\:opacity-50:disabled{opacity:.5}@media(min-width:640px){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}}@media(min-width:768px){.md\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}}@media(min-width:1024px){.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}}
@@ -0,0 +1,14 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>OpalServe Dashboard</title>
7
+ <link rel="icon" type="image/svg+xml" href="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 100 100'><text y='.9em' font-size='90'>&#x1F4A0;</text></svg>" />
8
+ <script type="module" crossorigin src="/dashboard/assets/index-BNOtcUPs.js"></script>
9
+ <link rel="stylesheet" crossorigin href="/dashboard/assets/index-Duwp34GW.css">
10
+ </head>
11
+ <body class="bg-gray-950 text-gray-100">
12
+ <div id="root"></div>
13
+ </body>
14
+ </html>
package/dist/index.d.ts CHANGED
@@ -4,9 +4,20 @@ export { DiscoveryEngine } from './core/discovery.js';
4
4
  export { ToolProxy } from './core/proxy.js';
5
5
  export { createServer, startServer } from './server/app.js';
6
6
  export { McpGateway } from './server/mcp-gateway.js';
7
+ export { hashPassword, verifyPassword } from './auth/passwords.js';
8
+ export { generateApiKey, hashApiKey } from './auth/api-keys.js';
9
+ export { createAuthMiddleware } from './auth/middleware.js';
10
+ export { KnowledgeBase } from './context/index.js';
11
+ export { chunkDocument } from './context/chunker.js';
12
+ export { UsageTracker } from './monitoring/tracker.js';
13
+ export { createGitHubServerConfig, handleGitHubWebhook } from './integrations/github.js';
14
+ export { createSlackServerConfig, handleSlackCommand, handleSlackEvent } from './integrations/slack.js';
7
15
  export { loadConfig, loadConfigFromPath } from './config/loader.js';
8
16
  export { getConfigDir, getDefaultDbPath, getDefaultConfig } from './config/defaults.js';
17
+ export { saveCredentials, loadCredentials, clearCredentials } from './config/credentials.js';
18
+ export { encrypt, decrypt, getMasterKey } from './core/secrets.js';
9
19
  export { Database } from './storage/database.js';
10
- export type { McpServerConfig, TransportConfig, ServerStatus, ConnectedServer, IndexedTool, ToolAnnotations, SearchResult, OpalServeConfig, HealthResult, ApiResponse, } from './types/index.js';
11
- export { McpServerConfigSchema, OpalServeConfigSchema, TransportConfigSchema, } from './types/index.js';
20
+ export { VERSION, APP_NAME } from './constants.js';
21
+ export type { McpServerConfig, TransportConfig, ServerStatus, ConnectedServer, IndexedTool, ToolAnnotations, SearchResult, OpalServeConfig, OpalServeMode, HealthResult, ApiResponse, User, Team, TeamMember, ApiKey, ContextDocument, UsageEvent, UsageStats, } from './types/index.js';
22
+ export { McpServerConfigSchema, OpalServeConfigSchema, TransportConfigSchema, ModeSchema, TeamConfigSchema, } from './types/index.js';
12
23
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAG5C,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC5D,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AAGrD,OAAO,EAAE,UAAU,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAGxF,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AAGjD,YAAY,EACV,eAAe,EACf,eAAe,EACf,YAAY,EACZ,eAAe,EACf,WAAW,EACX,eAAe,EACf,YAAY,EACZ,eAAe,EACf,YAAY,EACZ,WAAW,GACZ,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EACL,qBAAqB,EACrB,qBAAqB,EACrB,qBAAqB,GACtB,MAAM,kBAAkB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAG5C,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC5D,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AAGrD,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACnE,OAAO,EAAE,cAAc,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChE,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAG5D,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAGrD,OAAO,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AAGvD,OAAO,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AACzF,OAAO,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAC;AAGxG,OAAO,EAAE,UAAU,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxF,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAC;AAG7F,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAGnE,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AAGjD,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAGnD,YAAY,EACV,eAAe,EACf,eAAe,EACf,YAAY,EACZ,eAAe,EACf,WAAW,EACX,eAAe,EACf,YAAY,EACZ,eAAe,EACf,aAAa,EACb,YAAY,EACZ,WAAW,EACX,IAAI,EACJ,IAAI,EACJ,UAAU,EACV,MAAM,EACN,eAAe,EACf,UAAU,EACV,UAAU,GACX,MAAM,kBAAkB,CAAC;AAE1B,OAAO,EACL,qBAAqB,EACrB,qBAAqB,EACrB,qBAAqB,EACrB,UAAU,EACV,gBAAgB,GACjB,MAAM,kBAAkB,CAAC"}
package/dist/index.js CHANGED
@@ -6,10 +6,27 @@ export { ToolProxy } from './core/proxy.js';
6
6
  // Server
7
7
  export { createServer, startServer } from './server/app.js';
8
8
  export { McpGateway } from './server/mcp-gateway.js';
9
+ // Auth
10
+ export { hashPassword, verifyPassword } from './auth/passwords.js';
11
+ export { generateApiKey, hashApiKey } from './auth/api-keys.js';
12
+ export { createAuthMiddleware } from './auth/middleware.js';
13
+ // Context
14
+ export { KnowledgeBase } from './context/index.js';
15
+ export { chunkDocument } from './context/chunker.js';
16
+ // Monitoring
17
+ export { UsageTracker } from './monitoring/tracker.js';
18
+ // Integrations
19
+ export { createGitHubServerConfig, handleGitHubWebhook } from './integrations/github.js';
20
+ export { createSlackServerConfig, handleSlackCommand, handleSlackEvent } from './integrations/slack.js';
9
21
  // Config
10
22
  export { loadConfig, loadConfigFromPath } from './config/loader.js';
11
23
  export { getConfigDir, getDefaultDbPath, getDefaultConfig } from './config/defaults.js';
24
+ export { saveCredentials, loadCredentials, clearCredentials } from './config/credentials.js';
25
+ // Secrets
26
+ export { encrypt, decrypt, getMasterKey } from './core/secrets.js';
12
27
  // Storage
13
28
  export { Database } from './storage/database.js';
14
- export { McpServerConfigSchema, OpalServeConfigSchema, TransportConfigSchema, } from './types/index.js';
29
+ // Constants
30
+ export { VERSION, APP_NAME } from './constants.js';
31
+ export { McpServerConfigSchema, OpalServeConfigSchema, TransportConfigSchema, ModeSchema, TeamConfigSchema, } from './types/index.js';
15
32
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO;AACP,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,SAAS;AACT,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC5D,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AAErD,SAAS;AACT,OAAO,EAAE,UAAU,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AAExF,UAAU;AACV,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AAgBjD,OAAO,EACL,qBAAqB,EACrB,qBAAqB,EACrB,qBAAqB,GACtB,MAAM,kBAAkB,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO;AACP,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,SAAS;AACT,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC5D,OAAO,EAAE,UAAU,EAAE,MAAM,yBAAyB,CAAC;AAErD,OAAO;AACP,OAAO,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACnE,OAAO,EAAE,cAAc,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChE,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAE5D,UAAU;AACV,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AACnD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,aAAa;AACb,OAAO,EAAE,YAAY,EAAE,MAAM,yBAAyB,CAAC;AAEvD,eAAe;AACf,OAAO,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AACzF,OAAO,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAC;AAExG,SAAS;AACT,OAAO,EAAE,UAAU,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AACpE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxF,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,gBAAgB,EAAE,MAAM,yBAAyB,CAAC;AAE7F,UAAU;AACV,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAEnE,UAAU;AACV,OAAO,EAAE,QAAQ,EAAE,MAAM,uBAAuB,CAAC;AAEjD,YAAY;AACZ,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAwBnD,OAAO,EACL,qBAAqB,EACrB,qBAAqB,EACrB,qBAAqB,EACrB,UAAU,EACV,gBAAgB,GACjB,MAAM,kBAAkB,CAAC"}
@@ -0,0 +1,5 @@
1
+ import type { McpServerConfig } from '../types/index.js';
2
+ import type { Database } from '../storage/database.js';
3
+ export declare function createGitHubServerConfig(name: string, token: string): McpServerConfig;
4
+ export declare function handleGitHubWebhook(payload: unknown, db: Database): void;
5
+ //# sourceMappingURL=github.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"github.d.ts","sourceRoot":"","sources":["../../src/integrations/github.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAC;AAEvD,wBAAgB,wBAAwB,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,eAAe,CAerF;AAoBD,wBAAgB,mBAAmB,CAAC,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,QAAQ,GAAG,IAAI,CA6DxE"}
@@ -0,0 +1,63 @@
1
+ export function createGitHubServerConfig(name, token) {
2
+ return {
3
+ name,
4
+ transport: {
5
+ type: 'stdio',
6
+ command: 'npx',
7
+ args: ['-y', '@modelcontextprotocol/server-github'],
8
+ env: {
9
+ GITHUB_PERSONAL_ACCESS_TOKEN: token,
10
+ },
11
+ },
12
+ enabled: true,
13
+ tags: ['github', 'vcs', 'code'],
14
+ autoConnect: true,
15
+ };
16
+ }
17
+ export function handleGitHubWebhook(payload, db) {
18
+ if (!payload || typeof payload !== 'object')
19
+ return;
20
+ const data = payload;
21
+ // Push event
22
+ if (Array.isArray(data.commits) && data.ref) {
23
+ const push = data;
24
+ const commitSummary = push.commits
25
+ .map(c => `- ${c.message} (${c.author.name})`)
26
+ .join('\n');
27
+ const content = `# Push to ${push.repository.full_name}\n\nBranch: ${push.ref}\n\n## Commits\n${commitSummary}`;
28
+ db.run(`INSERT INTO context_documents (id, title, content, content_type, source, tags, created_at, updated_at)
29
+ VALUES (?, ?, ?, 'text/markdown', ?, '["github","push"]', datetime('now'), datetime('now'))`, [
30
+ crypto.randomUUID(),
31
+ `Push to ${push.repository.full_name} (${push.ref})`,
32
+ content,
33
+ push.commits[0]?.url ?? null,
34
+ ]);
35
+ return;
36
+ }
37
+ // PR event
38
+ if (data.pull_request && data.action) {
39
+ const pr = data;
40
+ const content = `# PR: ${pr.pull_request.title}\n\nAction: ${pr.action}\nRepo: ${pr.repository.full_name}\nAuthor: ${pr.pull_request.user.login}\n\n${pr.pull_request.body || ''}`;
41
+ db.run(`INSERT INTO context_documents (id, title, content, content_type, source, tags, created_at, updated_at)
42
+ VALUES (?, ?, ?, 'text/markdown', ?, '["github","pull-request"]', datetime('now'), datetime('now'))`, [
43
+ crypto.randomUUID(),
44
+ `PR: ${pr.pull_request.title}`,
45
+ content,
46
+ pr.pull_request.html_url,
47
+ ]);
48
+ return;
49
+ }
50
+ // Issue event
51
+ if (data.issue && data.action) {
52
+ const issue = data;
53
+ const content = `# Issue: ${issue.issue.title}\n\nAction: ${issue.action}\nRepo: ${issue.repository.full_name}\nAuthor: ${issue.issue.user.login}\n\n${issue.issue.body || ''}`;
54
+ db.run(`INSERT INTO context_documents (id, title, content, content_type, source, tags, created_at, updated_at)
55
+ VALUES (?, ?, ?, 'text/markdown', ?, '["github","issue"]', datetime('now'), datetime('now'))`, [
56
+ crypto.randomUUID(),
57
+ `Issue: ${issue.issue.title}`,
58
+ content,
59
+ issue.issue.html_url,
60
+ ]);
61
+ }
62
+ }
63
+ //# sourceMappingURL=github.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"github.js","sourceRoot":"","sources":["../../src/integrations/github.ts"],"names":[],"mappings":"AAGA,MAAM,UAAU,wBAAwB,CAAC,IAAY,EAAE,KAAa;IAClE,OAAO;QACL,IAAI;QACJ,SAAS,EAAE;YACT,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,KAAK;YACd,IAAI,EAAE,CAAC,IAAI,EAAE,qCAAqC,CAAC;YACnD,GAAG,EAAE;gBACH,4BAA4B,EAAE,KAAK;aACpC;SACF;QACD,OAAO,EAAE,IAAI;QACb,IAAI,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC;QAC/B,WAAW,EAAE,IAAI;KAClB,CAAC;AACJ,CAAC;AAoBD,MAAM,UAAU,mBAAmB,CAAC,OAAgB,EAAE,EAAY;IAChE,IAAI,CAAC,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ;QAAE,OAAO;IAEpD,MAAM,IAAI,GAAG,OAAkC,CAAC;IAEhD,aAAa;IACb,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,GAAG,EAAE,CAAC;QAC5C,MAAM,IAAI,GAAG,IAAoC,CAAC;QAClD,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO;aAC/B,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,OAAO,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC;aAC7C,IAAI,CAAC,IAAI,CAAC,CAAC;QAEd,MAAM,OAAO,GAAG,aAAa,IAAI,CAAC,UAAU,CAAC,SAAS,eAAe,IAAI,CAAC,GAAG,mBAAmB,aAAa,EAAE,CAAC;QAEhH,EAAE,CAAC,GAAG,CACJ;mGAC6F,EAC7F;YACE,MAAM,CAAC,UAAU,EAAE;YACnB,WAAW,IAAI,CAAC,UAAU,CAAC,SAAS,KAAK,IAAI,CAAC,GAAG,GAAG;YACpD,OAAO;YACP,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,GAAG,IAAI,IAAI;SAC7B,CACF,CAAC;QACF,OAAO;IACT,CAAC;IAED,WAAW;IACX,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;QACrC,MAAM,EAAE,GAAG,IAAkC,CAAC;QAC9C,MAAM,OAAO,GAAG,SAAS,EAAE,CAAC,YAAY,CAAC,KAAK,eAAe,EAAE,CAAC,MAAM,WAAW,EAAE,CAAC,UAAU,CAAC,SAAS,aAAa,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,OAAO,EAAE,CAAC,YAAY,CAAC,IAAI,IAAI,EAAE,EAAE,CAAC;QAEnL,EAAE,CAAC,GAAG,CACJ;2GACqG,EACrG;YACE,MAAM,CAAC,UAAU,EAAE;YACnB,OAAO,EAAE,CAAC,YAAY,CAAC,KAAK,EAAE;YAC9B,OAAO;YACP,EAAE,CAAC,YAAY,CAAC,QAAQ;SACzB,CACF,CAAC;QACF,OAAO;IACT,CAAC;IAED,cAAc;IACd,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;QAC9B,MAAM,KAAK,GAAG,IAAqC,CAAC;QACpD,MAAM,OAAO,GAAG,YAAY,KAAK,CAAC,KAAK,CAAC,KAAK,eAAe,KAAK,CAAC,MAAM,WAAW,KAAK,CAAC,UAAU,CAAC,SAAS,aAAa,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,OAAO,KAAK,CAAC,KAAK,CAAC,IAAI,IAAI,EAAE,EAAE,CAAC;QAEhL,EAAE,CAAC,GAAG,CACJ;oGAC8F,EAC9F;YACE,MAAM,CAAC,UAAU,EAAE;YACnB,UAAU,KAAK,CAAC,KAAK,CAAC,KAAK,EAAE;YAC7B,OAAO;YACP,KAAK,CAAC,KAAK,CAAC,QAAQ;SACrB,CACF,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,21 @@
1
+ import type { McpServerConfig } from '../types/index.js';
2
+ import type { Database } from '../storage/database.js';
3
+ export declare function createSlackServerConfig(name: string, botToken: string): McpServerConfig;
4
+ export declare function handleSlackCommand(body: {
5
+ text: string;
6
+ user_name: string;
7
+ channel_name: string;
8
+ }, registry: {
9
+ searchTools?: (query: string, limit?: number) => {
10
+ tool: {
11
+ name: string;
12
+ description: string;
13
+ serverName: string;
14
+ };
15
+ score: number;
16
+ }[];
17
+ }): {
18
+ text: string;
19
+ };
20
+ export declare function handleSlackEvent(event: unknown, db: Database): void;
21
+ //# sourceMappingURL=slack.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.d.ts","sourceRoot":"","sources":["../../src/integrations/slack.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAC;AAEvD,wBAAgB,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,eAAe,CAevF;AAED,wBAAgB,kBAAkB,CAChC,IAAI,EAAE;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAC;IAAC,YAAY,EAAE,MAAM,CAAA;CAAE,EAC/D,QAAQ,EAAE;IAAE,WAAW,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK;QAAE,IAAI,EAAE;YAAE,IAAI,EAAE,MAAM,CAAC;YAAC,WAAW,EAAE,MAAM,CAAC;YAAC,UAAU,EAAE,MAAM,CAAA;SAAE,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,EAAE,CAAA;CAAE,GAClJ;IAAE,IAAI,EAAE,MAAM,CAAA;CAAE,CAgClB;AAUD,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,OAAO,EAAE,EAAE,EAAE,QAAQ,GAAG,IAAI,CAuBnE"}
@@ -0,0 +1,61 @@
1
+ export function createSlackServerConfig(name, botToken) {
2
+ return {
3
+ name,
4
+ transport: {
5
+ type: 'stdio',
6
+ command: 'npx',
7
+ args: ['-y', '@modelcontextprotocol/server-slack'],
8
+ env: {
9
+ SLACK_BOT_TOKEN: botToken,
10
+ },
11
+ },
12
+ enabled: true,
13
+ tags: ['slack', 'messaging', 'communication'],
14
+ autoConnect: true,
15
+ };
16
+ }
17
+ export function handleSlackCommand(body, registry) {
18
+ const query = body.text.trim();
19
+ if (!query || query === 'help') {
20
+ return {
21
+ text: [
22
+ '*OpalServe Slash Command*',
23
+ '`/opalserve <query>` - Search for MCP tools',
24
+ '`/opalserve help` - Show this help message',
25
+ '',
26
+ `Requested by @${body.user_name} in #${body.channel_name}`,
27
+ ].join('\n'),
28
+ };
29
+ }
30
+ if (!registry.searchTools) {
31
+ return { text: 'Tool search is not available.' };
32
+ }
33
+ const results = registry.searchTools(query, 5);
34
+ if (results.length === 0) {
35
+ return { text: `No tools found for "${query}".` };
36
+ }
37
+ const lines = results.map((r, i) => `${i + 1}. *${r.tool.name}* (${r.tool.serverName}) - ${r.tool.description}`);
38
+ return {
39
+ text: `*Tool search results for "${query}":*\n${lines.join('\n')}`,
40
+ };
41
+ }
42
+ export function handleSlackEvent(event, db) {
43
+ if (!event || typeof event !== 'object')
44
+ return;
45
+ const data = event;
46
+ // Handle message events
47
+ if (data.type === 'message' && typeof data.text === 'string') {
48
+ const msg = data;
49
+ // Only ingest substantial messages (more than 50 chars)
50
+ if (msg.text.length < 50)
51
+ return;
52
+ db.run(`INSERT INTO context_documents (id, title, content, content_type, source, tags, created_at, updated_at)
53
+ VALUES (?, ?, ?, 'text/plain', ?, '["slack","message"]', datetime('now'), datetime('now'))`, [
54
+ crypto.randomUUID(),
55
+ `Slack message from ${msg.user} in ${msg.channel}`,
56
+ msg.text,
57
+ `slack://channel/${msg.channel}/message/${msg.ts}`,
58
+ ]);
59
+ }
60
+ }
61
+ //# sourceMappingURL=slack.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"slack.js","sourceRoot":"","sources":["../../src/integrations/slack.ts"],"names":[],"mappings":"AAGA,MAAM,UAAU,uBAAuB,CAAC,IAAY,EAAE,QAAgB;IACpE,OAAO;QACL,IAAI;QACJ,SAAS,EAAE;YACT,IAAI,EAAE,OAAO;YACb,OAAO,EAAE,KAAK;YACd,IAAI,EAAE,CAAC,IAAI,EAAE,oCAAoC,CAAC;YAClD,GAAG,EAAE;gBACH,eAAe,EAAE,QAAQ;aAC1B;SACF;QACD,OAAO,EAAE,IAAI;QACb,IAAI,EAAE,CAAC,OAAO,EAAE,WAAW,EAAE,eAAe,CAAC;QAC7C,WAAW,EAAE,IAAI;KAClB,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,kBAAkB,CAChC,IAA+D,EAC/D,QAAmJ;IAEnJ,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;IAE/B,IAAI,CAAC,KAAK,IAAI,KAAK,KAAK,MAAM,EAAE,CAAC;QAC/B,OAAO;YACL,IAAI,EAAE;gBACJ,2BAA2B;gBAC3B,6CAA6C;gBAC7C,4CAA4C;gBAC5C,EAAE;gBACF,iBAAiB,IAAI,CAAC,SAAS,QAAQ,IAAI,CAAC,YAAY,EAAE;aAC3D,CAAC,IAAI,CAAC,IAAI,CAAC;SACb,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,CAAC;QAC1B,OAAO,EAAE,IAAI,EAAE,+BAA+B,EAAE,CAAC;IACnD,CAAC;IAED,MAAM,OAAO,GAAG,QAAQ,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IAE/C,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACzB,OAAO,EAAE,IAAI,EAAE,uBAAuB,KAAK,IAAI,EAAE,CAAC;IACpD,CAAC;IAED,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CACjC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,CAAC,IAAI,CAAC,UAAU,OAAO,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAC5E,CAAC;IAEF,OAAO;QACL,IAAI,EAAE,6BAA6B,KAAK,QAAQ,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;KACnE,CAAC;AACJ,CAAC;AAUD,MAAM,UAAU,gBAAgB,CAAC,KAAc,EAAE,EAAY;IAC3D,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ;QAAE,OAAO;IAEhD,MAAM,IAAI,GAAG,KAAgC,CAAC;IAE9C,wBAAwB;IACxB,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,IAAI,OAAO,IAAI,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC7D,MAAM,GAAG,GAAG,IAAoC,CAAC;QAEjD,wDAAwD;QACxD,IAAI,GAAG,CAAC,IAAI,CAAC,MAAM,GAAG,EAAE;YAAE,OAAO;QAEjC,EAAE,CAAC,GAAG,CACJ;kGAC4F,EAC5F;YACE,MAAM,CAAC,UAAU,EAAE;YACnB,sBAAsB,GAAG,CAAC,IAAI,OAAO,GAAG,CAAC,OAAO,EAAE;YAClD,GAAG,CAAC,IAAI;YACR,mBAAmB,GAAG,CAAC,OAAO,YAAY,GAAG,CAAC,EAAE,EAAE;SACnD,CACF,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,31 @@
1
+ import type { Database } from '../storage/database.js';
2
+ import type { UsageStats } from '../types/index.js';
3
+ export declare class UsageTracker {
4
+ private db;
5
+ constructor(db: Database);
6
+ trackToolCall(opts: {
7
+ teamId?: string;
8
+ userId?: string;
9
+ serverName: string;
10
+ toolName: string;
11
+ latencyMs: number;
12
+ success: boolean;
13
+ error?: string;
14
+ }): void;
15
+ trackEvent(opts: {
16
+ teamId?: string;
17
+ userId?: string;
18
+ eventType: string;
19
+ serverName?: string;
20
+ toolName?: string;
21
+ metadata?: Record<string, unknown>;
22
+ }): void;
23
+ getOverview(period: '24h' | '7d' | '30d', teamId?: string): UsageStats;
24
+ getUserStats(userId: string, period: '24h' | '7d' | '30d'): UsageStats;
25
+ getServerStats(serverName: string, period: '24h' | '7d' | '30d'): {
26
+ calls: number;
27
+ errors: number;
28
+ avgLatency: number;
29
+ };
30
+ }
31
+ //# sourceMappingURL=tracker.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tracker.d.ts","sourceRoot":"","sources":["../../src/monitoring/tracker.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAUpD,qBAAa,YAAY;IACX,OAAO,CAAC,EAAE;gBAAF,EAAE,EAAE,QAAQ;IAEhC,aAAa,CAAC,IAAI,EAAE;QAClB,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;QACnB,QAAQ,EAAE,MAAM,CAAC;QACjB,SAAS,EAAE,MAAM,CAAC;QAClB,OAAO,EAAE,OAAO,CAAC;QACjB,KAAK,CAAC,EAAE,MAAM,CAAC;KAChB,GAAG,IAAI;IAgBR,UAAU,CAAC,IAAI,EAAE;QACf,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,SAAS,EAAE,MAAM,CAAC;QAClB,UAAU,CAAC,EAAE,MAAM,CAAC;QACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;KACpC,GAAG,IAAI;IAeR,WAAW,CAAC,MAAM,EAAE,KAAK,GAAG,IAAI,GAAG,KAAK,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,UAAU;IAmCtE,YAAY,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,GAAG,IAAI,GAAG,KAAK,GAAG,UAAU;IA2BtE,cAAc,CACZ,UAAU,EAAE,MAAM,EAClB,MAAM,EAAE,KAAK,GAAG,IAAI,GAAG,KAAK,GAC3B;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE;CAkBzD"}
@@ -0,0 +1,86 @@
1
+ function periodToSql(period) {
2
+ switch (period) {
3
+ case '24h': return "datetime('now', '-1 day')";
4
+ case '7d': return "datetime('now', '-7 days')";
5
+ case '30d': return "datetime('now', '-30 days')";
6
+ }
7
+ }
8
+ export class UsageTracker {
9
+ db;
10
+ constructor(db) {
11
+ this.db = db;
12
+ }
13
+ trackToolCall(opts) {
14
+ this.db.run(`INSERT INTO usage_events (team_id, user_id, event_type, server_name, tool_name, latency_ms, success, error)
15
+ VALUES (?, ?, 'tool_call', ?, ?, ?, ?, ?)`, [
16
+ opts.teamId ?? null,
17
+ opts.userId ?? null,
18
+ opts.serverName,
19
+ opts.toolName,
20
+ opts.latencyMs,
21
+ opts.success ? 1 : 0,
22
+ opts.error ?? null,
23
+ ]);
24
+ }
25
+ trackEvent(opts) {
26
+ this.db.run(`INSERT INTO usage_events (team_id, user_id, event_type, server_name, tool_name, success, metadata)
27
+ VALUES (?, ?, ?, ?, ?, 1, ?)`, [
28
+ opts.teamId ?? null,
29
+ opts.userId ?? null,
30
+ opts.eventType,
31
+ opts.serverName ?? null,
32
+ opts.toolName ?? null,
33
+ opts.metadata ? JSON.stringify(opts.metadata) : null,
34
+ ]);
35
+ }
36
+ getOverview(period, teamId) {
37
+ const since = periodToSql(period);
38
+ const teamFilter = teamId ? ' AND team_id = ?' : '';
39
+ const params = teamId ? [teamId] : [];
40
+ const totals = this.db.get(`SELECT COUNT(*) as total, SUM(CASE WHEN success = 0 THEN 1 ELSE 0 END) as errors
41
+ FROM usage_events WHERE created_at >= ${since}${teamFilter}`, params);
42
+ const activeUsers = this.db.get(`SELECT COUNT(DISTINCT user_id) as count
43
+ FROM usage_events WHERE user_id IS NOT NULL AND created_at >= ${since}${teamFilter}`, params);
44
+ const topTools = this.db.all(`SELECT tool_name as name, COUNT(*) as count
45
+ FROM usage_events WHERE tool_name IS NOT NULL AND created_at >= ${since}${teamFilter}
46
+ GROUP BY tool_name ORDER BY count DESC LIMIT 10`, params);
47
+ const total = totals?.total ?? 0;
48
+ const errors = totals?.errors ?? 0;
49
+ return {
50
+ totalCalls: total,
51
+ activeUsers: activeUsers?.count ?? 0,
52
+ topTools,
53
+ errorRate: total > 0 ? errors / total : 0,
54
+ };
55
+ }
56
+ getUserStats(userId, period) {
57
+ const since = periodToSql(period);
58
+ const totals = this.db.get(`SELECT COUNT(*) as total, SUM(CASE WHEN success = 0 THEN 1 ELSE 0 END) as errors
59
+ FROM usage_events WHERE user_id = ? AND created_at >= ${since}`, [userId]);
60
+ const topTools = this.db.all(`SELECT tool_name as name, COUNT(*) as count
61
+ FROM usage_events WHERE user_id = ? AND tool_name IS NOT NULL AND created_at >= ${since}
62
+ GROUP BY tool_name ORDER BY count DESC LIMIT 10`, [userId]);
63
+ const total = totals?.total ?? 0;
64
+ const errors = totals?.errors ?? 0;
65
+ return {
66
+ totalCalls: total,
67
+ activeUsers: 1,
68
+ topTools,
69
+ errorRate: total > 0 ? errors / total : 0,
70
+ };
71
+ }
72
+ getServerStats(serverName, period) {
73
+ const since = periodToSql(period);
74
+ const row = this.db.get(`SELECT
75
+ COUNT(*) as calls,
76
+ SUM(CASE WHEN success = 0 THEN 1 ELSE 0 END) as errors,
77
+ COALESCE(AVG(latency_ms), 0) as avg_latency
78
+ FROM usage_events WHERE server_name = ? AND created_at >= ${since}`, [serverName]);
79
+ return {
80
+ calls: row?.calls ?? 0,
81
+ errors: row?.errors ?? 0,
82
+ avgLatency: row?.avg_latency ?? 0,
83
+ };
84
+ }
85
+ }
86
+ //# sourceMappingURL=tracker.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tracker.js","sourceRoot":"","sources":["../../src/monitoring/tracker.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/server/app.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAM9B,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAI7D,wBAAsB,YAAY,CAAC,QAAQ,EAAE,iBAAiB,sSAuB7D;AAED,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,iBAAiB,EAC3B,OAAO,CAAC,EAAE;IAAE,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,sSAW3C"}
1
+ {"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../../src/server/app.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAgB9B,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AAI7D,wBAAsB,YAAY,CAAC,QAAQ,EAAE,iBAAiB,sSA8D7D;AAED,wBAAsB,WAAW,CAC/B,QAAQ,EAAE,iBAAiB,EAC3B,OAAO,CAAC,EAAE;IAAE,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAA;CAAE,sSAW3C"}
@@ -1,8 +1,18 @@
1
1
  import Fastify from 'fastify';
2
2
  import cors from '@fastify/cors';
3
+ import fastifyStatic from '@fastify/static';
4
+ import { join, dirname } from 'node:path';
5
+ import { fileURLToPath } from 'node:url';
6
+ import { existsSync } from 'node:fs';
3
7
  import { registerHealthRoutes } from './routes/health.js';
4
8
  import { registerServerRoutes } from './routes/servers.js';
5
9
  import { registerToolRoutes } from './routes/tools.js';
10
+ import { registerAuthRoutes } from './routes/auth.js';
11
+ import { registerTeamServerRoutes } from './routes/team-servers.js';
12
+ import { registerContextRoutes } from './routes/context.js';
13
+ import { registerStatsRoutes } from './routes/stats.js';
14
+ import { registerWebhookRoutes } from './routes/webhooks.js';
15
+ import { registerAuthDecorator, createAuthMiddleware } from '../auth/middleware.js';
6
16
  import { createLogger } from '../utils/logger.js';
7
17
  const log = createLogger('http');
8
18
  export async function createServer(registry) {
@@ -12,16 +22,52 @@ export async function createServer(registry) {
12
22
  await app.register(cors, {
13
23
  origin: ['http://localhost:*', 'http://127.0.0.1:*'],
14
24
  });
25
+ // Register the user decorator so request.user is available
26
+ registerAuthDecorator(app);
15
27
  app.addHook('onRequest', async (request) => {
16
28
  log.debug({ method: request.method, url: request.url }, 'request');
17
29
  });
18
30
  app.addHook('onResponse', async (request, reply) => {
19
31
  log.debug({ method: request.method, url: request.url, status: reply.statusCode }, 'response');
20
32
  });
33
+ // In team-server mode, add auth middleware globally for protected routes
34
+ if (registry.isTeamServer()) {
35
+ const authHook = createAuthMiddleware(registry.getDatabase());
36
+ app.addHook('onRequest', async (request, reply) => {
37
+ // Skip auth for public endpoints
38
+ const publicPaths = [
39
+ '/api/v1/health',
40
+ '/api/v1/auth/register',
41
+ '/api/v1/auth/login',
42
+ '/api/v1/webhooks/',
43
+ '/api/v1/slack/',
44
+ '/dashboard',
45
+ ];
46
+ const isPublic = publicPaths.some(p => request.url.startsWith(p));
47
+ if (!isPublic && request.url.startsWith('/api/')) {
48
+ await authHook(request, reply);
49
+ }
50
+ });
51
+ }
21
52
  // Register all routes
22
53
  registerHealthRoutes(app, registry);
23
54
  registerServerRoutes(app, registry);
24
55
  registerToolRoutes(app, registry);
56
+ registerAuthRoutes(app, registry);
57
+ registerTeamServerRoutes(app, registry);
58
+ registerContextRoutes(app, registry);
59
+ registerStatsRoutes(app, registry);
60
+ registerWebhookRoutes(app, registry);
61
+ // Serve dashboard static files if available
62
+ const __dirname = dirname(fileURLToPath(import.meta.url));
63
+ const dashboardPath = join(__dirname, '..', 'dashboard');
64
+ if (existsSync(dashboardPath)) {
65
+ await app.register(fastifyStatic, {
66
+ root: dashboardPath,
67
+ prefix: '/dashboard/',
68
+ decorateReply: false,
69
+ });
70
+ }
25
71
  return app;
26
72
  }
27
73
  export async function startServer(registry, options) {
@@ -1 +1 @@
1
- {"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/server/app.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,IAAI,MAAM,eAAe,CAAC;AACjC,OAAO,EAAE,oBAAoB,EAAE,MAAM,oBAAoB,CAAC;AAC1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,qBAAqB,CAAC;AAC3D,OAAO,EAAE,kBAAkB,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAGlD,MAAM,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC;AAEjC,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,QAA2B;IAC5D,MAAM,GAAG,GAAG,OAAO,CAAC;QAClB,MAAM,EAAE,KAAK,EAAG,6BAA6B;KAC9C,CAAC,CAAC;IAEH,MAAM,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE;QACvB,MAAM,EAAE,CAAC,oBAAoB,EAAE,oBAAoB,CAAC;KACrD,CAAC,CAAC;IAEH,GAAG,CAAC,OAAO,CAAC,WAAW,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE;QACzC,GAAG,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,EAAE,SAAS,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,OAAO,CAAC,YAAY,EAAE,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,EAAE;QACjD,GAAG,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,CAAC,UAAU,EAAE,EAAE,UAAU,CAAC,CAAC;IAChG,CAAC,CAAC,CAAC;IAEH,sBAAsB;IACtB,oBAAoB,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IACpC,oBAAoB,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IACpC,kBAAkB,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;IAElC,OAAO,GAAG,CAAC;AACb,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,QAA2B,EAC3B,OAA0C;IAE1C,MAAM,IAAI,GAAG,OAAO,EAAE,IAAI,IAAI,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC;IAC3D,MAAM,IAAI,GAAG,OAAO,EAAE,IAAI,IAAI,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC;IAE3D,MAAM,GAAG,GAAG,MAAM,YAAY,CAAC,QAAQ,CAAC,CAAC;IAEzC,MAAM,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;IACjC,GAAG,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,oBAAoB,CAAC,CAAC;IAE/C,OAAO,GAAG,CAAC;AACb,CAAC"}
1
+ {"version":3,"file":"app.js","sourceRoot":"","sources":["../../src/server/app.ts"],"names":[],"mappings":"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"}
@@ -8,7 +8,7 @@ export class McpGateway {
8
8
  mcpServer;
9
9
  constructor(registry) {
10
10
  this.registry = registry;
11
- this.mcpServer = new McpServer({ name: 'opalserve', version: '2.0.0' }, { capabilities: { tools: {} } });
11
+ this.mcpServer = new McpServer({ name: 'opalserve', version: '3.0.0' }, { capabilities: { tools: {} } });
12
12
  this.registerMetaTools();
13
13
  this.registerProxyTools();
14
14
  }
@@ -0,0 +1,4 @@
1
+ import type { FastifyInstance } from 'fastify';
2
+ import type { OpalServeRegistry } from '../../core/registry.js';
3
+ export declare function registerAuthRoutes(app: FastifyInstance, registry: OpalServeRegistry): void;
4
+ //# sourceMappingURL=auth.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.d.ts","sourceRoot":"","sources":["../../../src/server/routes/auth.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAE/C,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC;AAKhE,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,eAAe,EAAE,QAAQ,EAAE,iBAAiB,GAAG,IAAI,CA+J1F"}