mirror of
https://github.com/DeBrosOfficial/orama.git
synced 2026-06-16 23:14:13 +00:00
Migration 028: namespace_push_credentials
- Per-(namespace, provider) AES-256-GCM encrypted credential blob.
- Generic schema — apns/ntfy/expo/future plug in with zero migration.
- Separated from migration 026's namespace_push_config (preferences vs
credentials, different access patterns).
pkg/push/credentials
- Manager + Registry + RQLite store; HKDF purpose "namespace-push-credentials"
via pkg/secrets. Provider Validator interface for per-provider schema.
pkg/push/providers/apns
- Apple Push Notification service direct provider (no Expo proxy).
- Validator + dispatcher; credentials are p8 signing key + key_id + team_id.
pkg/push/providers/ntfy/credentials.go
- ntfy credential schema (auth_token + default topic). Used both with
the public ntfy.sh and our self-hosted instance.
pkg/environments/production/installers/ntfy.go
- Self-hosted ntfy server installer. Binary, system user, hardened
/etc/ntfy/server.yml, systemd unit. Listens on 127.0.0.1:NtfyListenPort
only — Caddy is the only public path.
pkg/environments/production/installers/caddy.go
- Emit reverse_proxy block for push.<dnsZone> -> 127.0.0.1:NtfyListenPort
when operator enables ntfy on a node.
CLI: install/upgrade orchestrators learn a new "ntfy" install/preserve
phase; flag gating in install/flags.go + upgrade/flags.go.
Gateway handlers/push/credentials_handler.go
- GET/PUT/DELETE /v1/namespace/push-credentials/{provider}.
- PUT validates against provider Validator before encrypting and storing.
- GET returns a redacted view (booleans + non-secret fields only).
Push manager: provider resolution now also consults
namespace_push_credentials before falling back to YAML defaults.
Docs: core/docs/PUSH_NOTIFICATIONS.md walks through end-to-end setup.
VERSION bumped to 0.122.14.
306 lines
13 KiB
Go
306 lines
13 KiB
Go
package gateway
|
|
|
|
import (
|
|
"net/http"
|
|
|
|
"github.com/DeBrosOfficial/network/pkg/gateway/ctxkeys"
|
|
)
|
|
|
|
// Routes returns the http.Handler with all routes and middleware configured
|
|
func (g *Gateway) Routes() http.Handler {
|
|
mux := http.NewServeMux()
|
|
|
|
// root and v1 health/status
|
|
mux.HandleFunc("/health", g.healthHandler)
|
|
mux.HandleFunc("/status", g.statusHandler)
|
|
mux.HandleFunc("/v1/health", g.healthHandler)
|
|
mux.HandleFunc("/v1/version", g.versionHandler)
|
|
mux.HandleFunc("/v1/status", g.statusHandler)
|
|
// Schema-version contract (bug #214 audit follow-up): tenants can
|
|
// self-check whether their gateway's required schema is applied.
|
|
mux.HandleFunc("/v1/schema-status", g.handleSchemaStatus)
|
|
|
|
// Internal ping for peer-to-peer health monitoring
|
|
mux.HandleFunc("/v1/internal/ping", g.pingHandler)
|
|
|
|
// TLS check endpoint for Caddy on-demand TLS
|
|
mux.HandleFunc("/v1/internal/tls/check", g.tlsCheckHandler)
|
|
|
|
// ACME DNS-01 challenge endpoints (for Caddy httpreq DNS provider)
|
|
mux.HandleFunc("/v1/internal/acme/present", g.acmePresentHandler)
|
|
mux.HandleFunc("/v1/internal/acme/cleanup", g.acmeCleanupHandler)
|
|
|
|
// WireGuard peer exchange (internal, cluster-secret auth)
|
|
if g.wireguardHandler != nil {
|
|
mux.HandleFunc("/v1/internal/wg/peer", g.wireguardHandler.HandleRegisterPeer)
|
|
mux.HandleFunc("/v1/internal/wg/peers", g.wireguardHandler.HandleListPeers)
|
|
mux.HandleFunc("/v1/internal/wg/peer/remove", g.wireguardHandler.HandleRemovePeer)
|
|
}
|
|
|
|
// Node join endpoint (token-authenticated, no middleware auth needed)
|
|
if g.joinHandler != nil {
|
|
mux.HandleFunc("/v1/internal/join", g.joinHandler.HandleJoin)
|
|
}
|
|
|
|
// OramaOS node management (handler does its own auth)
|
|
if g.enrollHandler != nil {
|
|
mux.HandleFunc("/v1/node/enroll", g.enrollHandler.HandleEnroll)
|
|
mux.HandleFunc("/v1/node/status", g.enrollHandler.HandleNodeStatus)
|
|
mux.HandleFunc("/v1/node/command", g.enrollHandler.HandleNodeCommand)
|
|
mux.HandleFunc("/v1/node/logs", g.enrollHandler.HandleNodeLogs)
|
|
mux.HandleFunc("/v1/node/leave", g.enrollHandler.HandleNodeLeave)
|
|
}
|
|
|
|
// Namespace instance spawn/stop (internal, handler does its own auth)
|
|
if g.spawnHandler != nil {
|
|
mux.Handle("/v1/internal/namespace/spawn", g.spawnHandler)
|
|
}
|
|
|
|
// Namespace cluster repair (internal, handler does its own auth)
|
|
mux.HandleFunc("/v1/internal/namespace/repair", g.namespaceClusterRepairHandler)
|
|
|
|
// Namespace WebRTC enable/disable/status (internal, handler does its own auth)
|
|
mux.HandleFunc("/v1/internal/namespace/webrtc/enable", g.namespaceWebRTCEnableHandler)
|
|
mux.HandleFunc("/v1/internal/namespace/webrtc/disable", g.namespaceWebRTCDisableHandler)
|
|
mux.HandleFunc("/v1/internal/namespace/webrtc/status", g.namespaceWebRTCStatusHandler)
|
|
|
|
// Namespace WebRTC enable/disable/status (public, JWT/API key auth via middleware)
|
|
mux.HandleFunc("/v1/namespace/webrtc/enable", g.namespaceWebRTCEnablePublicHandler)
|
|
mux.HandleFunc("/v1/namespace/webrtc/disable", g.namespaceWebRTCDisablePublicHandler)
|
|
mux.HandleFunc("/v1/namespace/webrtc/status", g.namespaceWebRTCStatusPublicHandler)
|
|
|
|
// auth endpoints
|
|
mux.HandleFunc("/v1/auth/jwks", g.authService.JWKSHandler)
|
|
mux.HandleFunc("/.well-known/jwks.json", g.authService.JWKSHandler)
|
|
if g.authHandlers != nil {
|
|
mux.HandleFunc("/v1/auth/challenge", g.authHandlers.ChallengeHandler)
|
|
mux.HandleFunc("/v1/auth/verify", g.authHandlers.VerifyHandler)
|
|
// Issue JWT from API key; create or return API key for a wallet after verification
|
|
mux.HandleFunc("/v1/auth/token", g.authHandlers.APIKeyToJWTHandler)
|
|
mux.HandleFunc("/v1/auth/api-key", g.authHandlers.IssueAPIKeyHandler)
|
|
mux.HandleFunc("/v1/auth/simple-key", g.authHandlers.SimpleAPIKeyHandler)
|
|
mux.HandleFunc("/v1/auth/register", g.authHandlers.RegisterHandler)
|
|
mux.HandleFunc("/v1/auth/refresh", g.authHandlers.RefreshHandler)
|
|
mux.HandleFunc("/v1/auth/logout", g.authHandlers.LogoutHandler)
|
|
mux.HandleFunc("/v1/auth/whoami", g.authHandlers.WhoamiHandler)
|
|
// Phantom Solana auth (QR code + deep link)
|
|
mux.HandleFunc("/v1/auth/phantom/session", g.authHandlers.PhantomSessionHandler)
|
|
mux.HandleFunc("/v1/auth/phantom/session/", g.authHandlers.PhantomSessionStatusHandler)
|
|
mux.HandleFunc("/v1/auth/phantom/complete", g.authHandlers.PhantomCompleteHandler)
|
|
}
|
|
|
|
// RQLite native backup/restore proxy (namespace auth via /v1/rqlite/ prefix)
|
|
mux.HandleFunc("/v1/rqlite/export", g.rqliteExportHandler)
|
|
mux.HandleFunc("/v1/rqlite/import", g.rqliteImportHandler)
|
|
|
|
// rqlite ORM HTTP gateway (mounts /v1/rqlite/* endpoints)
|
|
if g.ormHTTP != nil {
|
|
g.ormHTTP.BasePath = "/v1/rqlite"
|
|
g.ormHTTP.RegisterRoutes(mux)
|
|
}
|
|
|
|
// namespace cluster status (public endpoint for polling during provisioning)
|
|
mux.HandleFunc("/v1/namespace/status", g.namespaceClusterStatusHandler)
|
|
|
|
// namespace delete (authenticated — goes through auth middleware)
|
|
if g.namespaceDeleteHandler != nil {
|
|
mux.Handle("/v1/namespace/delete", g.namespaceDeleteHandler)
|
|
}
|
|
|
|
// namespace list (authenticated — lists namespaces owned by the current wallet)
|
|
if g.namespaceListHandler != nil {
|
|
mux.Handle("/v1/namespace/list", g.namespaceListHandler)
|
|
}
|
|
|
|
// network
|
|
mux.HandleFunc("/v1/network/status", g.networkStatusHandler)
|
|
mux.HandleFunc("/v1/network/peers", g.networkPeersHandler)
|
|
mux.HandleFunc("/v1/network/connect", g.networkConnectHandler)
|
|
mux.HandleFunc("/v1/network/disconnect", g.networkDisconnectHandler)
|
|
|
|
// pubsub
|
|
if g.pubsubHandlers != nil {
|
|
mux.HandleFunc("/v1/pubsub/ws", g.pubsubHandlers.WebsocketHandler)
|
|
mux.HandleFunc("/v1/pubsub/publish", g.pubsubHandlers.PublishHandler)
|
|
mux.HandleFunc("/v1/pubsub/publish-batch", g.pubsubHandlers.PublishBatchHandler)
|
|
mux.HandleFunc("/v1/pubsub/topics", g.pubsubHandlers.TopicsHandler)
|
|
mux.HandleFunc("/v1/pubsub/presence", g.pubsubHandlers.PresenceHandler)
|
|
}
|
|
|
|
// push notifications
|
|
//
|
|
// Routes are ALWAYS registered (bug #220). When no provider is
|
|
// configured, the handler returns a canonical 503 envelope explaining
|
|
// that push isn't enabled — far better UX than a bare 404 that sends
|
|
// operators down "is the gateway broken?" rabbit holes.
|
|
mux.HandleFunc("/v1/push/devices", g.pushDevicesHandler)
|
|
// DELETE /v1/push/devices/{id} — uses path-prefix routing because
|
|
// net/http mux doesn't extract path params; the handler parses {id}.
|
|
mux.HandleFunc("/v1/push/devices/", g.pushDevicesByIDHandler)
|
|
mux.HandleFunc("/v1/push/send", g.pushSendHandler)
|
|
|
|
// Per-namespace push provider configuration (bug #220 follow-up):
|
|
// GET / PUT / DELETE — tenants self-serve their ntfy/expo credentials
|
|
// instead of filing an ops ticket. Method dispatched in the handler.
|
|
mux.HandleFunc("/v1/push/config", g.pushConfigHandler)
|
|
|
|
// Per-namespace, per-provider push credentials (feature #72 —
|
|
// full-privacy push with APNs-direct + self-hosted ntfy). Generic by
|
|
// design: any provider with a registered Validator plugs in here
|
|
// without changes. Method + provider segment dispatched in the handler.
|
|
//
|
|
// Summary endpoint (no provider segment) returns "what's configured"
|
|
// + "what's supported" in one round trip.
|
|
mux.HandleFunc("/v1/namespace/push-credentials", g.pushCredentialsSummaryHandler)
|
|
mux.HandleFunc("/v1/namespace/push-credentials/", g.pushCredentialsByProviderHandler)
|
|
|
|
// Per-namespace rate-limit configuration (feature #69).
|
|
// GET / PUT / DELETE — tenants self-serve their gateway-level rate
|
|
// limit override (requests_per_minute, burst) up to an operator-set
|
|
// ceiling. Falls back to gateway YAML defaults when no override is set.
|
|
if g.rateLimitHandlers != nil {
|
|
mux.HandleFunc("/v1/namespace/rate-limit", g.rateLimitConfigDispatcher)
|
|
}
|
|
|
|
// operator node management (wallet JWT auth via middleware)
|
|
if g.operatorHandler != nil {
|
|
mux.HandleFunc("/v1/operator/invite", g.operatorHandler.HandleInvite)
|
|
mux.HandleFunc("/v1/operator/nodes", g.operatorHandler.HandleListNodes)
|
|
mux.HandleFunc("/v1/operator/node/register", g.operatorHandler.HandleRegister)
|
|
}
|
|
|
|
// vault proxy (public, rate-limited per identity within handler)
|
|
if g.vaultHandlers != nil {
|
|
mux.HandleFunc("/v1/vault/push", g.vaultHandlers.HandlePush)
|
|
mux.HandleFunc("/v1/vault/pull", g.vaultHandlers.HandlePull)
|
|
mux.HandleFunc("/v1/vault/health", g.vaultHandlers.HandleHealth)
|
|
mux.HandleFunc("/v1/vault/status", g.vaultHandlers.HandleStatus)
|
|
}
|
|
|
|
// webrtc
|
|
if g.webrtcHandlers != nil {
|
|
mux.HandleFunc("/v1/webrtc/turn/credentials", g.webrtcHandlers.CredentialsHandler)
|
|
mux.HandleFunc("/v1/webrtc/signal", g.webrtcHandlers.SignalHandler)
|
|
mux.HandleFunc("/v1/webrtc/rooms", g.webrtcHandlers.RoomsHandler)
|
|
}
|
|
|
|
// anon proxy (authenticated users only)
|
|
mux.HandleFunc("/v1/proxy/anon", g.anonProxyHandler)
|
|
|
|
// cache endpoints (Olric) - always register, check handler dynamically
|
|
// This allows cache routes to work after background Olric reconnection
|
|
mux.HandleFunc("/v1/cache/health", g.cacheHealthHandler)
|
|
mux.HandleFunc("/v1/cache/get", g.cacheGetHandler)
|
|
mux.HandleFunc("/v1/cache/mget", g.cacheMGetHandler)
|
|
mux.HandleFunc("/v1/cache/put", g.cachePutHandler)
|
|
mux.HandleFunc("/v1/cache/delete", g.cacheDeleteHandler)
|
|
mux.HandleFunc("/v1/cache/scan", g.cacheScanHandler)
|
|
|
|
// storage endpoints (IPFS)
|
|
if g.storageHandlers != nil {
|
|
mux.HandleFunc("/v1/storage/upload", g.storageHandlers.UploadHandler)
|
|
mux.HandleFunc("/v1/storage/pin", g.storageHandlers.PinHandler)
|
|
mux.HandleFunc("/v1/storage/status/", g.storageHandlers.StatusHandler)
|
|
mux.HandleFunc("/v1/storage/get/", g.storageHandlers.DownloadHandler)
|
|
mux.HandleFunc("/v1/storage/unpin/", g.storageHandlers.UnpinHandler)
|
|
}
|
|
|
|
// serverless functions (if enabled)
|
|
if g.serverlessHandlers != nil {
|
|
g.serverlessHandlers.RegisterRoutes(mux)
|
|
}
|
|
|
|
// deployment endpoints
|
|
if g.deploymentService != nil {
|
|
// Static deployments
|
|
mux.HandleFunc("/v1/deployments/static/upload", g.staticHandler.HandleUpload)
|
|
mux.HandleFunc("/v1/deployments/static/update", g.withHomeNodeProxy(g.updateHandler.HandleUpdate))
|
|
|
|
// Next.js deployments
|
|
mux.HandleFunc("/v1/deployments/nextjs/upload", g.nextjsHandler.HandleUpload)
|
|
mux.HandleFunc("/v1/deployments/nextjs/update", g.withHomeNodeProxy(g.updateHandler.HandleUpdate))
|
|
|
|
// Go backend deployments
|
|
if g.goHandler != nil {
|
|
mux.HandleFunc("/v1/deployments/go/upload", g.goHandler.HandleUpload)
|
|
mux.HandleFunc("/v1/deployments/go/update", g.withHomeNodeProxy(g.updateHandler.HandleUpdate))
|
|
}
|
|
|
|
// Node.js backend deployments
|
|
if g.nodejsHandler != nil {
|
|
mux.HandleFunc("/v1/deployments/nodejs/upload", g.nodejsHandler.HandleUpload)
|
|
mux.HandleFunc("/v1/deployments/nodejs/update", g.withHomeNodeProxy(g.updateHandler.HandleUpdate))
|
|
}
|
|
|
|
// Deployment management
|
|
mux.HandleFunc("/v1/deployments/list", g.listHandler.HandleList)
|
|
mux.HandleFunc("/v1/deployments/get", g.listHandler.HandleGet)
|
|
mux.HandleFunc("/v1/deployments/delete", g.withHomeNodeProxy(g.listHandler.HandleDelete))
|
|
mux.HandleFunc("/v1/deployments/rollback", g.withHomeNodeProxy(g.rollbackHandler.HandleRollback))
|
|
mux.HandleFunc("/v1/deployments/versions", g.rollbackHandler.HandleListVersions)
|
|
mux.HandleFunc("/v1/deployments/logs", g.withHomeNodeProxy(g.logsHandler.HandleLogs))
|
|
mux.HandleFunc("/v1/deployments/stats", g.withHomeNodeProxy(g.statsHandler.HandleStats))
|
|
mux.HandleFunc("/v1/deployments/events", g.logsHandler.HandleGetEvents)
|
|
|
|
// Internal replica coordination endpoints
|
|
if g.replicaHandler != nil {
|
|
mux.HandleFunc("/v1/internal/deployments/replica/setup", g.replicaHandler.HandleSetup)
|
|
mux.HandleFunc("/v1/internal/deployments/replica/update", g.replicaHandler.HandleUpdate)
|
|
mux.HandleFunc("/v1/internal/deployments/replica/rollback", g.replicaHandler.HandleRollback)
|
|
mux.HandleFunc("/v1/internal/deployments/replica/teardown", g.replicaHandler.HandleTeardown)
|
|
}
|
|
|
|
// Custom domains
|
|
mux.HandleFunc("/v1/deployments/domains/add", g.domainHandler.HandleAddDomain)
|
|
mux.HandleFunc("/v1/deployments/domains/verify", g.domainHandler.HandleVerifyDomain)
|
|
mux.HandleFunc("/v1/deployments/domains/list", g.domainHandler.HandleListDomains)
|
|
mux.HandleFunc("/v1/deployments/domains/remove", g.domainHandler.HandleRemoveDomain)
|
|
}
|
|
|
|
// SQLite database endpoints
|
|
if g.sqliteHandler != nil {
|
|
mux.HandleFunc("/v1/db/sqlite/create", g.sqliteHandler.CreateDatabase)
|
|
mux.HandleFunc("/v1/db/sqlite/query", g.sqliteHandler.QueryDatabase)
|
|
mux.HandleFunc("/v1/db/sqlite/list", g.sqliteHandler.ListDatabases)
|
|
mux.HandleFunc("/v1/db/sqlite/backup", g.sqliteBackupHandler.BackupDatabase)
|
|
mux.HandleFunc("/v1/db/sqlite/backups", g.sqliteBackupHandler.ListBackups)
|
|
}
|
|
|
|
return g.withMiddleware(mux)
|
|
}
|
|
|
|
// withHomeNodeProxy wraps a deployment handler to proxy requests to the home node
|
|
// if the current node is not the home node for the deployment.
|
|
func (g *Gateway) withHomeNodeProxy(handler http.HandlerFunc) http.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
// Already proxied — prevent loops
|
|
if r.Header.Get("X-Orama-Proxy-Node") != "" {
|
|
handler(w, r)
|
|
return
|
|
}
|
|
name := r.URL.Query().Get("name")
|
|
if name == "" {
|
|
handler(w, r)
|
|
return
|
|
}
|
|
ctx := r.Context()
|
|
namespace, _ := ctx.Value(ctxkeys.NamespaceOverride).(string)
|
|
if namespace == "" {
|
|
handler(w, r)
|
|
return
|
|
}
|
|
deployment, err := g.deploymentService.GetDeployment(ctx, namespace, name)
|
|
if err != nil {
|
|
handler(w, r) // let handler return proper error
|
|
return
|
|
}
|
|
if g.nodePeerID != "" && deployment.HomeNodeID != "" &&
|
|
deployment.HomeNodeID != g.nodePeerID {
|
|
if g.proxyCrossNode(w, r, deployment) {
|
|
return
|
|
}
|
|
}
|
|
handler(w, r)
|
|
}
|
|
}
|