mirror of
https://github.com/DeBrosOfficial/orama.git
synced 2026-06-16 22:54:12 +00:00
- Add `raw_http_response` configuration to functions to allow verbatim HTTP responses - Implement cluster-wide secrets encryption key generation and distribution for serverless functions - Update documentation with UnifiedPush support for ntfy on Android/GrapheneOS
121 lines
4.4 KiB
Go
121 lines
4.4 KiB
Go
package hostfunctions
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
|
|
"github.com/DeBrosOfficial/network/pkg/serverless"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
// HTTPFetch makes an outbound HTTP request directly from the gateway.
|
|
func (h *HostFunctions) HTTPFetch(ctx context.Context, method, url string, headers map[string]string, body []byte) ([]byte, error) {
|
|
return h.doFetch(ctx, "http_fetch", h.httpClient, method, url, headers, body)
|
|
}
|
|
|
|
// SetHTTPResponse records a verbatim HTTP response for a RawHTTPResponse
|
|
// function (bugboard #835). It delegates to the per-invocation collector
|
|
// attached on ctx by the engine; the HTTP invoke handler replays the result
|
|
// byte-for-byte. Validation (raw mode enabled, status range, header/body caps)
|
|
// lives in serverless.SetRawHTTPResponse.
|
|
func (h *HostFunctions) SetHTTPResponse(ctx context.Context, status int, headers map[string]string, body []byte) error {
|
|
if err := serverless.SetRawHTTPResponse(ctx, status, headers, body); err != nil {
|
|
return &serverless.HostFunctionError{Function: "set_http_response", Cause: err}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// AnyoneFetch makes an outbound HTTP request routed through the Anyone
|
|
// (ANyONe protocol) SOCKS5 proxy, so the third-party endpoint sees an
|
|
// Anyone exit IP instead of the gateway IP and the gateway can't
|
|
// correlate (function → external request) traffic by source IP.
|
|
// Feat-11 — server-side analog of anchat's client-side proxyClient.
|
|
//
|
|
// Privacy guarantee: there is NO silent fallback to direct. If Anyone
|
|
// routing isn't available on this gateway (operator disabled it via
|
|
// --disable-anonrc / ANYONE_DISABLE=1, so h.anyoneHTTPClient is nil),
|
|
// this returns a typed error rather than leaking the request over the
|
|
// direct path. If the Anyone daemon is configured-but-down, the SOCKS
|
|
// dial to localhost:9050 fails and surfaces as a transport error — also
|
|
// never a direct send. This is the explicit ask in feat-11: a privacy
|
|
// regression must fail loudly, not degrade silently.
|
|
func (h *HostFunctions) AnyoneFetch(ctx context.Context, method, url string, headers map[string]string, body []byte) ([]byte, error) {
|
|
if h.anyoneHTTPClient == nil {
|
|
// Anyone routing not enabled on this gateway. Return the typed
|
|
// error envelope (status 0) rather than dialing direct — the
|
|
// caller explicitly asked for anonymized egress and we must not
|
|
// silently downgrade it.
|
|
errorResp := map[string]interface{}{
|
|
"error": "anyone routing not available on this gateway (disabled by operator)",
|
|
"status": 0,
|
|
"proxy": "anyone",
|
|
}
|
|
return json.Marshal(errorResp)
|
|
}
|
|
return h.doFetch(ctx, "anyone_fetch", h.anyoneHTTPClient, method, url, headers, body)
|
|
}
|
|
|
|
// doFetch is the shared request/response machinery for HTTPFetch and
|
|
// AnyoneFetch — identical except for which *http.Client (direct vs
|
|
// SOCKS-routed) does the dialing and the function name used in logs +
|
|
// HostFunctionError.
|
|
func (h *HostFunctions) doFetch(ctx context.Context, fnName string, client *http.Client, method, url string, headers map[string]string, body []byte) ([]byte, error) {
|
|
var bodyReader io.Reader
|
|
if len(body) > 0 {
|
|
bodyReader = bytes.NewReader(body)
|
|
}
|
|
|
|
req, err := http.NewRequestWithContext(ctx, method, url, bodyReader)
|
|
if err != nil {
|
|
h.logger.Error(fnName+" request creation error", zap.Error(err), zap.String("url", url))
|
|
errorResp := map[string]interface{}{
|
|
"error": "failed to create request: " + err.Error(),
|
|
"status": 0,
|
|
}
|
|
return json.Marshal(errorResp)
|
|
}
|
|
|
|
for key, value := range headers {
|
|
req.Header.Set(key, value)
|
|
}
|
|
|
|
resp, err := client.Do(req)
|
|
if err != nil {
|
|
h.logger.Error(fnName+" transport error", zap.Error(err), zap.String("url", url))
|
|
errorResp := map[string]interface{}{
|
|
"error": err.Error(),
|
|
"status": 0, // Transport error
|
|
}
|
|
return json.Marshal(errorResp)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
respBody, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
h.logger.Error(fnName+" response read error", zap.Error(err), zap.String("url", url))
|
|
errorResp := map[string]interface{}{
|
|
"error": "failed to read response: " + err.Error(),
|
|
"status": resp.StatusCode,
|
|
}
|
|
return json.Marshal(errorResp)
|
|
}
|
|
|
|
// Encode response with status code
|
|
response := map[string]interface{}{
|
|
"status": resp.StatusCode,
|
|
"headers": resp.Header,
|
|
"body": string(respBody),
|
|
}
|
|
|
|
data, err := json.Marshal(response)
|
|
if err != nil {
|
|
return nil, &serverless.HostFunctionError{Function: fnName, Cause: fmt.Errorf("failed to marshal response: %w", err)}
|
|
}
|
|
|
|
return data, nil
|
|
}
|