package proxy import ( _ "embed" "fmt" "github.com/caddyserver/caddy/v2" "github.com/caddyserver/caddy/v2/caddyconfig/caddyfile" "github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile" "github.com/caddyserver/caddy/v2/modules/caddyhttp/fileserver" "github.com/openziti/zrok/endpoints" "github.com/openziti/zrok/sdk" "github.com/sirupsen/logrus" "os" "strings" "text/template" ) //go:embed browse.html var browseHtml string func init() { fileserver.BrowseTemplate = browseHtml } type CaddyfileBackendConfig struct { CaddyfilePath string Shr *sdk.Share Requests chan *endpoints.Request } type CaddyfileBackend struct { cfg []byte } func NewCaddyfileBackend(cfg *CaddyfileBackendConfig) (*CaddyfileBackend, error) { cdyf, err := preprocessCaddyfile(cfg.CaddyfilePath, cfg.Shr) if err != nil { return nil, err } var adapter caddyfile.Adapter adapter.ServerType = httpcaddyfile.ServerType{} caddyCfg, warnings, err := adapter.Adapt([]byte(cdyf), map[string]interface{}{"filename": cfg.CaddyfilePath}) if err != nil { return nil, err } for _, warning := range warnings { logrus.Warnf("%v [%d] (%v): %v", cfg.CaddyfilePath, warning.Line, warning.Directive, warning.Message) } return &CaddyfileBackend{cfg: caddyCfg}, nil } func (b *CaddyfileBackend) Run() error { if err := caddy.Load(b.cfg, true); err != nil { return err } return nil } func preprocessCaddyfile(inF string, shr *sdk.Share) (string, error) { input, err := os.ReadFile(inF) if err != nil { return "", err } tmpl, err := template.New(inF).Parse(string(input)) if err != nil { return "", err } output := new(strings.Builder) if err := tmpl.Execute(output, &caddyfileData{ZrokBindAddress: fmt.Sprintf("zrok/%s", shr.Token)}); err != nil { return "", err } return output.String(), nil } type caddyfileData struct { ZrokBindAddress string }