Merge pull request 'Add authmw for other services to use as middleware' (#1) from authmw into master

Reviewed-on: #1
This commit is contained in:
Torjus Håkestad 2023-10-22 19:36:35 +00:00
commit abbc102060
3 changed files with 96 additions and 3 deletions

92
authmw/token.go Normal file
View File

@ -0,0 +1,92 @@
package authmw
import (
"context"
"crypto/x509"
"encoding/json"
"fmt"
"net/http"
"slices"
"strings"
"git.t-juice.club/microfilm/auth"
"github.com/golang-jwt/jwt/v5"
)
func VerifyToken(authURL string, permittedRoles []string) func(http.Handler) http.Handler {
// Fetch current pubkey
url := fmt.Sprintf("%s/key", authURL)
req, err := http.NewRequest(http.MethodGet, url, nil)
if err != nil {
panic(err)
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
panic(err)
}
defer resp.Body.Close()
var authResponse auth.PubkeyResponse
decoder := json.NewDecoder(resp.Body)
if err := decoder.Decode(&authResponse); err != nil {
panic(err)
}
// Parse pubkey
pub, err := x509.ParsePKIXPublicKey(authResponse.PubKey)
if err != nil {
panic(err)
}
fn := func(next http.Handler) http.Handler {
fn := func(w http.ResponseWriter, r *http.Request) {
authHeader := r.Header.Get("Authorization")
if !strings.Contains(authHeader, "Bearer ") {
// No token, pass if unathorized in permitted
// else reject
if slices.Contains[[]string, string](permittedRoles, auth.RoleUnauthorized) {
next.ServeHTTP(w, r)
return
}
// Reject and write error response
w.WriteHeader(http.StatusUnauthorized)
var errResp auth.ErrorResponse
errResp.Message = fmt.Sprintf("Authorization required: %s", strings.Join(permittedRoles, ","))
errResp.Status = http.StatusUnauthorized
encoder := json.NewEncoder(w)
_ = encoder.Encode(&errResp)
return
}
// Validate token
tokenString := strings.Split(authHeader, " ")[1]
token, err := jwt.ParseWithClaims(tokenString, &auth.MicrofilmClaims{}, func(t *jwt.Token) (interface{}, error) { return pub, nil })
if err != nil {
// Reject and write error response
w.WriteHeader(http.StatusUnauthorized)
var errResp auth.ErrorResponse
errResp.Message = fmt.Sprintf("Token verification failed: %s", err)
errResp.Status = http.StatusUnauthorized
encoder := json.NewEncoder(w)
_ = encoder.Encode(&errResp)
return
}
// Add claims to request context
if claims, ok := token.Claims.(*auth.MicrofilmClaims); ok && token.Valid {
ctx := context.WithValue(r.Context(), "claims", claims)
next.ServeHTTP(w, r.WithContext(ctx))
return
}
next.ServeHTTP(w, r)
}
return http.HandlerFunc(fn)
}
return fn
}

View File

@ -3,6 +3,7 @@ package auth
import "github.com/golang-jwt/jwt/v5" import "github.com/golang-jwt/jwt/v5"
const ( const (
RoleUnauthorized = ""
RoleUser = "user" RoleUser = "user"
RoleAdmin = "admin" RoleAdmin = "admin"
) )

View File

@ -1,3 +1,3 @@
package auth package auth
const Version = "v0.1.0" const Version = "v0.1.1"