anonpenguin23 f41242538e feat(serverless): add raw http response mode and secrets encryption
- 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
2026-06-09 13:01:02 +03:00

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
}