Upload files to "/"

This commit is contained in:
Ryan Malloy 2025-02-10 09:05:35 +00:00
commit e1e41070af

172
gitea.go Normal file
View File

@ -0,0 +1,172 @@
package gitea
import (
"fmt"
"io"
"net/http"
"strings"
"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"
"git.supported.systems/rsp2k/caddy-gitea"
)
func init() {
// Register the module and the Caddyfile handler directive.
caddy.RegisterModule(Middleware{})
httpcaddyfile.RegisterHandlerDirective("gitea", parseCaddyfile)
}
// parseCaddyfile creates and configures the module from Caddyfile input.
func parseCaddyfile(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, error) {
var m Middleware
if err := m.UnmarshalCaddyfile(h.Dispenser); err != nil {
return nil, err
}
return &m, nil
}
// Middleware implements the Gitea plugin.
type Middleware struct {
// The Gitea client is created during Provision.
Client *gitea.Client `json:"-"`
// Required/optional configuration fields:
Server string `json:"server,omitempty"` // e.g. "https://gitea.example.com"
Token string `json:"token,omitempty"` // API token if needed
GiteaPages string `json:"gitea_pages,omitempty"` // e.g. URL for Gitea pages
GiteaPagesAllowAll string `json:"gitea_pages_allowall,omitempty"` // configuration for allow-all pages mode
Domain string `json:"domain,omitempty"` // if set, used to split subdomain parts
}
// CaddyModule returns the Caddy module information.
func (Middleware) CaddyModule() caddy.ModuleInfo {
return caddy.ModuleInfo{
ID: "http.handlers.gitea",
New: func() caddy.Module {
return new(Middleware)
},
}
}
// Provision creates the Gitea client.
func (m *Middleware) Provision(ctx caddy.Context) error {
// Validate required configuration.
if m.Server == "" {
return fmt.Errorf("gitea: server must be specified")
}
var err error
m.Client, err = gitea.NewClient(m.Server, m.Token, m.GiteaPages, m.GiteaPagesAllowAll)
return err
}
// Validate performs additional configuration validation.
func (m *Middleware) Validate() error {
if m.Server == "" {
return fmt.Errorf("gitea: server is required")
}
return nil
}
// UnmarshalCaddyfile sets up the configuration from Caddyfile tokens.
func (m *Middleware) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
// Each block of tokens (if more than one block is given) will be processed.
for d.Next() {
// Process all subdirectives in the block.
for n := d.Nesting(); d.NextBlock(n); {
switch d.Val() {
case "server":
if !d.NextArg() {
return d.ArgErr()
}
m.Server = d.Val()
case "token":
if !d.NextArg() {
return d.ArgErr()
}
m.Token = d.Val()
case "gitea_pages":
if !d.NextArg() {
return d.ArgErr()
}
m.GiteaPages = d.Val()
case "gitea_pages_allowall":
if !d.NextArg() {
return d.ArgErr()
}
m.GiteaPagesAllowAll = d.Val()
case "domain":
if !d.NextArg() {
return d.ArgErr()
}
m.Domain = d.Val()
default:
return d.Errf("gitea: unrecognized subdirective '%s'", d.Val())
}
}
}
return nil
}
// ServeHTTP implements the HTTP handler. It uses the Gitea client to
// retrieve content based on the requests host and path.
func (m Middleware) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyhttp.Handler) error {
// Derive the file path from the request's host and URL.
//
// If m.Domain is set then we expect requests in one of two forms:
// - repo.username.<domain> (len=2 after splitting by '.')
// - branch.repo.username.<domain> (len=3)
//
// Otherwise, we just use the first segment of the host.
host := r.Host
if m.Domain != "" {
// Remove the configured domain (and a trailing dot, if any)
host = strings.TrimSuffix(host, m.Domain)
host = strings.TrimRight(host, ".")
}
parts := strings.Split(host, ".")
var fp string
// Default: use first part plus the URL path.
if len(parts) > 0 {
fp = parts[0] + r.URL.Path
}
// Adjust based on subdomain count when a domain is configured.
ref := r.URL.Query().Get("ref")
if m.Domain != "" {
switch len(parts) {
case 2:
// Format: repo.username.<domain>
fp = parts[1] + "/" + parts[0] + r.URL.Path
case 3:
// Format: branch.repo.username.<domain>
fp = parts[2] + "/" + parts[1] + r.URL.Path
ref = parts[0]
}
}
// Open the file from Gitea.
f, err := m.Client.Open(fp, ref)
if err != nil {
// Return a 404 error if the file isnt found.
return caddyhttp.Error(http.StatusNotFound, err)
}
// If the returned file supports closing, ensure it gets closed.
if closer, ok := f.(io.Closer); ok {
defer closer.Close()
}
// Copy the file's contents to the response.
_, err = io.Copy(w, f)
return err
}
// Interface guards
var (
_ caddy.Provisioner = (*Middleware)(nil)
_ caddy.Validator = (*Middleware)(nil)
_ caddyhttp.MiddlewareHandler = (*Middleware)(nil)
_ caddyfile.Unmarshaler = (*Middleware)(nil)
)