2025-01-05 20:37:12 +00:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"github.com/1f349/bluebell/database"
|
|
|
|
"github.com/1f349/bluebell/upload"
|
2025-01-08 00:59:27 +00:00
|
|
|
"github.com/1f349/bluebell/validation"
|
2025-01-05 20:37:12 +00:00
|
|
|
"github.com/1f349/mjwt"
|
|
|
|
"github.com/1f349/mjwt/auth"
|
|
|
|
"github.com/julienschmidt/httprouter"
|
|
|
|
"golang.org/x/net/publicsuffix"
|
|
|
|
"net/http"
|
|
|
|
)
|
|
|
|
|
|
|
|
type apiDB interface {
|
|
|
|
SetDomainBranchEnabled(ctx context.Context, arg database.SetDomainBranchEnabledParams) error
|
|
|
|
}
|
|
|
|
|
|
|
|
func New(upload *upload.Handler, keyStore *mjwt.KeyStore, db apiDB) *httprouter.Router {
|
|
|
|
router := httprouter.New()
|
|
|
|
router.POST("/u/:site/:branch", upload.Handle)
|
|
|
|
router.PUT("/sites/:host/:branch/enable", checkAuth(keyStore, func(rw http.ResponseWriter, req *http.Request, params httprouter.Params, b AuthClaims) {
|
|
|
|
setEnabled(rw, req, params, b, db, true)
|
|
|
|
}))
|
|
|
|
router.DELETE("/sites/:host/:branch/enable", checkAuth(keyStore, func(rw http.ResponseWriter, req *http.Request, params httprouter.Params, b AuthClaims) {
|
|
|
|
setEnabled(rw, req, params, b, db, false)
|
|
|
|
}))
|
|
|
|
return router
|
|
|
|
}
|
|
|
|
|
|
|
|
func setEnabled(rw http.ResponseWriter, req *http.Request, params httprouter.Params, b AuthClaims, db apiDB, enable bool) {
|
|
|
|
host := params.ByName("host")
|
|
|
|
branch := params.ByName("branch")
|
|
|
|
|
2025-01-08 00:59:27 +00:00
|
|
|
if !validation.IsValidSite(host) {
|
|
|
|
http.Error(rw, "Invalid site", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !validation.IsValidBranch(branch) {
|
|
|
|
http.Error(rw, "Invalid branch", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2025-01-05 20:37:12 +00:00
|
|
|
if !validateDomainOwnershipClaims(host, b.Claims.Perms) {
|
|
|
|
http.Error(rw, "Forbidden", http.StatusForbidden)
|
2025-01-08 00:32:31 +00:00
|
|
|
return
|
2025-01-05 20:37:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err := db.SetDomainBranchEnabled(req.Context(), database.SetDomainBranchEnabledParams{
|
|
|
|
Domain: host,
|
|
|
|
Branch: branch,
|
|
|
|
Enable: enable,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
http.Error(rw, "Failed to update branch state", http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
rw.WriteHeader(http.StatusAccepted)
|
|
|
|
}
|
|
|
|
|
|
|
|
// apiError outputs a generic JSON error message
|
|
|
|
func apiError(rw http.ResponseWriter, code int, m string) {
|
|
|
|
rw.WriteHeader(code)
|
|
|
|
_ = json.NewEncoder(rw).Encode(map[string]string{
|
|
|
|
"error": m,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// validateDomainOwnershipClaims validates if the claims contain the
|
|
|
|
// `domain:owns=<fqdn>` field with the matching top level domain
|
|
|
|
func validateDomainOwnershipClaims(a string, perms *auth.PermStorage) bool {
|
|
|
|
if fqdn, err := publicsuffix.EffectiveTLDPlusOne(a); err == nil {
|
|
|
|
if perms.Has("domain:owns=" + fqdn) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|