This commit is contained in:
Thomas Miceli 2025-01-06 14:18:31 +01:00
parent 68c6f26385
commit e589d3d727
13 changed files with 227 additions and 234 deletions

View file

@ -13,7 +13,7 @@ import (
"sync" "sync"
) )
type OGContext struct { type Context struct {
echo.Context echo.Context
data echo.Map data echo.Map
@ -23,33 +23,33 @@ type OGContext struct {
User *db.User User *db.User
} }
func NewContext(c echo.Context, sessionPath string) *OGContext { func NewContext(c echo.Context, sessionPath string) *Context {
return &OGContext{ return &Context{
Context: c, Context: c,
data: make(echo.Map), data: make(echo.Map),
store: NewStore(sessionPath), store: NewStore(sessionPath),
} }
} }
func (ctx *OGContext) SetData(key string, value any) { func (ctx *Context) SetData(key string, value any) {
ctx.lock.Lock() ctx.lock.Lock()
defer ctx.lock.Unlock() defer ctx.lock.Unlock()
ctx.data[key] = value ctx.data[key] = value
} }
func (ctx *OGContext) GetData(key string) any { func (ctx *Context) GetData(key string) any {
ctx.lock.RLock() ctx.lock.RLock()
defer ctx.lock.RUnlock() defer ctx.lock.RUnlock()
return ctx.data[key] return ctx.data[key]
} }
func (ctx *OGContext) DataMap() echo.Map { func (ctx *Context) DataMap() echo.Map {
return ctx.data return ctx.data
} }
func (ctx *OGContext) ErrorRes(code int, message string, err error) error { func (ctx *Context) ErrorRes(code int, message string, err error) error {
if code >= 500 { if code >= 500 {
var skipLogger = log.With().CallerWithSkipFrameCount(3).Logger() var skipLogger = log.With().CallerWithSkipFrameCount(3).Logger()
skipLogger.Error().Err(err).Msg(message) skipLogger.Error().Err(err).Msg(message)
@ -60,36 +60,36 @@ func (ctx *OGContext) ErrorRes(code int, message string, err error) error {
return &echo.HTTPError{Code: code, Message: message, Internal: err} return &echo.HTTPError{Code: code, Message: message, Internal: err}
} }
func (ctx *OGContext) RedirectTo(location string) error { func (ctx *Context) RedirectTo(location string) error {
return ctx.Context.Redirect(302, config.C.ExternalUrl+location) return ctx.Context.Redirect(302, config.C.ExternalUrl+location)
} }
func (ctx *OGContext) HTML_(template string) error { func (ctx *Context) HTML_(template string) error {
return ctx.HtmlWithCode(200, template) return ctx.HtmlWithCode(200, template)
} }
func (ctx *OGContext) HtmlWithCode(code int, template string) error { func (ctx *Context) HtmlWithCode(code int, template string) error {
ctx.setErrorFlashes() ctx.setErrorFlashes()
return ctx.Render(code, template, ctx.DataMap()) return ctx.Render(code, template, ctx.DataMap())
} }
func (ctx *OGContext) JSON_(data any) error { func (ctx *Context) JSON_(data any) error {
return ctx.JsonWithCode(200, data) return ctx.JsonWithCode(200, data)
} }
func (ctx *OGContext) JsonWithCode(code int, data any) error { func (ctx *Context) JsonWithCode(code int, data any) error {
return ctx.JSON(code, data) return ctx.JSON(code, data)
} }
func (ctx *OGContext) PlainText(code int, message string) error { func (ctx *Context) PlainText(code int, message string) error {
return ctx.String(code, message) return ctx.String(code, message)
} }
func (ctx *OGContext) NotFound(message string) error { func (ctx *Context) NotFound(message string) error {
return ctx.ErrorRes(404, message, nil) return ctx.ErrorRes(404, message, nil)
} }
func (ctx *OGContext) setErrorFlashes() { func (ctx *Context) setErrorFlashes() {
sess, _ := ctx.store.flashStore.Get(ctx.Request(), "flash") sess, _ := ctx.store.flashStore.Get(ctx.Request(), "flash")
ctx.SetData("flashErrors", sess.Flashes("error")) ctx.SetData("flashErrors", sess.Flashes("error"))
@ -99,28 +99,28 @@ func (ctx *OGContext) setErrorFlashes() {
_ = sess.Save(ctx.Request(), ctx.Response()) _ = sess.Save(ctx.Request(), ctx.Response())
} }
func (ctx *OGContext) GetSession() *sessions.Session { func (ctx *Context) GetSession() *sessions.Session {
sess, _ := ctx.store.UserStore.Get(ctx.Request(), "session") sess, _ := ctx.store.UserStore.Get(ctx.Request(), "session")
return sess return sess
} }
func (ctx *OGContext) SaveSession(sess *sessions.Session) { func (ctx *Context) SaveSession(sess *sessions.Session) {
_ = sess.Save(ctx.Request(), ctx.Response()) _ = sess.Save(ctx.Request(), ctx.Response())
} }
func (ctx *OGContext) DeleteSession() { func (ctx *Context) DeleteSession() {
sess := ctx.GetSession() sess := ctx.GetSession()
sess.Options.MaxAge = -1 sess.Options.MaxAge = -1
ctx.SaveSession(sess) ctx.SaveSession(sess)
} }
func (ctx *OGContext) AddFlash(flashMessage string, flashType string) { func (ctx *Context) AddFlash(flashMessage string, flashType string) {
sess, _ := ctx.store.flashStore.Get(ctx.Request(), "flash") sess, _ := ctx.store.flashStore.Get(ctx.Request(), "flash")
sess.AddFlash(flashMessage, flashType) sess.AddFlash(flashMessage, flashType)
_ = sess.Save(ctx.Request(), ctx.Response()) _ = sess.Save(ctx.Request(), ctx.Response())
} }
func (ctx *OGContext) getUserLogged() *db.User { func (ctx *Context) getUserLogged() *db.User {
user := ctx.GetData("userLogged") user := ctx.GetData("userLogged")
if user != nil { if user != nil {
return user.(*db.User) return user.(*db.User)
@ -128,16 +128,16 @@ func (ctx *OGContext) getUserLogged() *db.User {
return nil return nil
} }
func (ctx *OGContext) DeleteCsrfCookie() { func (ctx *Context) DeleteCsrfCookie() {
ctx.SetCookie(&http.Cookie{Name: "_csrf", Path: "/", MaxAge: -1}) ctx.SetCookie(&http.Cookie{Name: "_csrf", Path: "/", MaxAge: -1})
} }
func (ctx *OGContext) TrH(key string, args ...any) template.HTML { func (ctx *Context) TrH(key string, args ...any) template.HTML {
l := ctx.GetData("locale").(*i18n.Locale) l := ctx.GetData("locale").(*i18n.Locale)
return l.Tr(key, args...) return l.Tr(key, args...)
} }
func (ctx *OGContext) Tr(key string, args ...any) string { func (ctx *Context) Tr(key string, args ...any) string {
l := ctx.GetData("locale").(*i18n.Locale) l := ctx.GetData("locale").(*i18n.Locale)
return l.String(key, args...) return l.String(key, args...)
} }

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"github.com/thomiceli/opengist/internal/actions" "github.com/thomiceli/opengist/internal/actions"
@ -11,7 +11,7 @@ import (
"time" "time"
) )
func AdminIndex(ctx *context.OGContext) error { func AdminIndex(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("admin.admin_panel")) ctx.SetData("htmlTitle", ctx.TrH("admin.admin_panel"))
ctx.SetData("adminHeaderPage", "index") ctx.SetData("adminHeaderPage", "index")
@ -50,7 +50,7 @@ func AdminIndex(ctx *context.OGContext) error {
return ctx.HTML_("admin_index.html") return ctx.HTML_("admin_index.html")
} }
func AdminUsers(ctx *context.OGContext) error { func AdminUsers(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("admin.users")+" - "+ctx.TrH("admin.admin_panel")) ctx.SetData("htmlTitle", ctx.TrH("admin.users")+" - "+ctx.TrH("admin.admin_panel"))
ctx.SetData("adminHeaderPage", "users") ctx.SetData("adminHeaderPage", "users")
ctx.SetData("loadStartTime", time.Now()) ctx.SetData("loadStartTime", time.Now())
@ -70,7 +70,7 @@ func AdminUsers(ctx *context.OGContext) error {
return ctx.HTML_("admin_users.html") return ctx.HTML_("admin_users.html")
} }
func AdminGists(ctx *context.OGContext) error { func AdminGists(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("admin.gists")+" - "+ctx.TrH("admin.admin_panel")) ctx.SetData("htmlTitle", ctx.TrH("admin.gists")+" - "+ctx.TrH("admin.admin_panel"))
ctx.SetData("adminHeaderPage", "gists") ctx.SetData("adminHeaderPage", "gists")
pageInt := getPage(ctx) pageInt := getPage(ctx)
@ -88,7 +88,7 @@ func AdminGists(ctx *context.OGContext) error {
return ctx.HTML_("admin_gists.html") return ctx.HTML_("admin_gists.html")
} }
func AdminUserDelete(ctx *context.OGContext) error { func AdminUserDelete(ctx *context.Context) error {
userId, _ := strconv.ParseUint(ctx.Param("user"), 10, 64) userId, _ := strconv.ParseUint(ctx.Param("user"), 10, 64)
user, err := db.GetUserById(uint(userId)) user, err := db.GetUserById(uint(userId))
if err != nil { if err != nil {
@ -103,7 +103,7 @@ func AdminUserDelete(ctx *context.OGContext) error {
return ctx.RedirectTo("/admin-panel/users") return ctx.RedirectTo("/admin-panel/users")
} }
func AdminGistDelete(ctx *context.OGContext) error { func AdminGistDelete(ctx *context.Context) error {
gist, err := db.GetGistByID(ctx.Param("gist")) gist, err := db.GetGistByID(ctx.Param("gist"))
if err != nil { if err != nil {
return ctx.ErrorRes(500, "Cannot retrieve gist", err) return ctx.ErrorRes(500, "Cannot retrieve gist", err)
@ -123,43 +123,43 @@ func AdminGistDelete(ctx *context.OGContext) error {
return ctx.RedirectTo("/admin-panel/gists") return ctx.RedirectTo("/admin-panel/gists")
} }
func AdminSyncReposFromFS(ctx *context.OGContext) error { func AdminSyncReposFromFS(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.sync-fs"), "success") ctx.AddFlash(ctx.Tr("flash.admin.sync-fs"), "success")
go actions.Run(actions.SyncReposFromFS) go actions.Run(actions.SyncReposFromFS)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminSyncReposFromDB(ctx *context.OGContext) error { func AdminSyncReposFromDB(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.sync-db"), "success") ctx.AddFlash(ctx.Tr("flash.admin.sync-db"), "success")
go actions.Run(actions.SyncReposFromDB) go actions.Run(actions.SyncReposFromDB)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminGcRepos(ctx *context.OGContext) error { func AdminGcRepos(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.git-gc"), "success") ctx.AddFlash(ctx.Tr("flash.admin.git-gc"), "success")
go actions.Run(actions.GitGcRepos) go actions.Run(actions.GitGcRepos)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminSyncGistPreviews(ctx *context.OGContext) error { func AdminSyncGistPreviews(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.sync-previews"), "success") ctx.AddFlash(ctx.Tr("flash.admin.sync-previews"), "success")
go actions.Run(actions.SyncGistPreviews) go actions.Run(actions.SyncGistPreviews)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminResetHooks(ctx *context.OGContext) error { func AdminResetHooks(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.reset-hooks"), "success") ctx.AddFlash(ctx.Tr("flash.admin.reset-hooks"), "success")
go actions.Run(actions.ResetHooks) go actions.Run(actions.ResetHooks)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminIndexGists(ctx *context.OGContext) error { func AdminIndexGists(ctx *context.Context) error {
ctx.AddFlash(ctx.Tr("flash.admin.index-gists"), "success") ctx.AddFlash(ctx.Tr("flash.admin.index-gists"), "success")
go actions.Run(actions.IndexGists) go actions.Run(actions.IndexGists)
return ctx.RedirectTo("/admin-panel") return ctx.RedirectTo("/admin-panel")
} }
func AdminConfig(ctx *context.OGContext) error { func AdminConfig(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("admin.configuration")+" - "+ctx.TrH("admin.admin_panel")) ctx.SetData("htmlTitle", ctx.TrH("admin.configuration")+" - "+ctx.TrH("admin.admin_panel"))
ctx.SetData("adminHeaderPage", "config") ctx.SetData("adminHeaderPage", "config")
@ -169,7 +169,7 @@ func AdminConfig(ctx *context.OGContext) error {
return ctx.HTML_("admin_config.html") return ctx.HTML_("admin_config.html")
} }
func AdminSetConfig(ctx *context.OGContext) error { func AdminSetConfig(ctx *context.Context) error {
key := ctx.FormValue("key") key := ctx.FormValue("key")
value := ctx.FormValue("value") value := ctx.FormValue("value")
@ -182,7 +182,7 @@ func AdminSetConfig(ctx *context.OGContext) error {
}) })
} }
func AdminInvitations(ctx *context.OGContext) error { func AdminInvitations(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("admin.invitations")+" - "+ctx.TrH("admin.admin_panel")) ctx.SetData("htmlTitle", ctx.TrH("admin.invitations")+" - "+ctx.TrH("admin.admin_panel"))
ctx.SetData("adminHeaderPage", "invitations") ctx.SetData("adminHeaderPage", "invitations")
@ -196,7 +196,7 @@ func AdminInvitations(ctx *context.OGContext) error {
return ctx.HTML_("admin_invitations.html") return ctx.HTML_("admin_invitations.html")
} }
func AdminInvitationsCreate(ctx *context.OGContext) error { func AdminInvitationsCreate(ctx *context.Context) error {
code := ctx.FormValue("code") code := ctx.FormValue("code")
nbMax, err := strconv.ParseUint(ctx.FormValue("nbMax"), 10, 64) nbMax, err := strconv.ParseUint(ctx.FormValue("nbMax"), 10, 64)
if err != nil { if err != nil {
@ -222,7 +222,7 @@ func AdminInvitationsCreate(ctx *context.OGContext) error {
return ctx.RedirectTo("/admin-panel/invitations") return ctx.RedirectTo("/admin-panel/invitations")
} }
func AdminInvitationsDelete(ctx *context.OGContext) error { func AdminInvitationsDelete(ctx *context.Context) error {
id, _ := strconv.ParseUint(ctx.Param("id"), 10, 64) id, _ := strconv.ParseUint(ctx.Param("id"), 10, 64)
invitation, err := db.GetInvitationByID(uint(id)) invitation, err := db.GetInvitationByID(uint(id))
if err != nil { if err != nil {

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"bytes" "bytes"
@ -37,7 +37,7 @@ const (
OpenIDConnect = "openid-connect" OpenIDConnect = "openid-connect"
) )
func Register(ctx *context.OGContext) error { func Register(ctx *context.Context) error {
disableSignup := ctx.GetData("DisableSignup") disableSignup := ctx.GetData("DisableSignup")
disableForm := ctx.GetData("DisableLoginForm") disableForm := ctx.GetData("DisableLoginForm")
@ -58,7 +58,7 @@ func Register(ctx *context.OGContext) error {
return ctx.HTML_("auth_form.html") return ctx.HTML_("auth_form.html")
} }
func ProcessRegister(ctx *context.OGContext) error { func ProcessRegister(ctx *context.Context) error {
disableSignup := ctx.GetData("DisableSignup") disableSignup := ctx.GetData("DisableSignup")
code := ctx.QueryParam("code") code := ctx.QueryParam("code")
@ -127,7 +127,7 @@ func ProcessRegister(ctx *context.OGContext) error {
return ctx.RedirectTo("/") return ctx.RedirectTo("/")
} }
func Login(ctx *context.OGContext) error { func Login(ctx *context.Context) error {
ctx.SetData("title", ctx.TrH("auth.login")) ctx.SetData("title", ctx.TrH("auth.login"))
ctx.SetData("htmlTitle", ctx.TrH("auth.login")) ctx.SetData("htmlTitle", ctx.TrH("auth.login"))
ctx.SetData("disableForm", ctx.GetData("DisableLoginForm")) ctx.SetData("disableForm", ctx.GetData("DisableLoginForm"))
@ -135,7 +135,7 @@ func Login(ctx *context.OGContext) error {
return ctx.HTML_("auth_form.html") return ctx.HTML_("auth_form.html")
} }
func ProcessLogin(ctx *context.OGContext) error { func ProcessLogin(ctx *context.Context) error {
if ctx.GetData("DisableLoginForm") == true { if ctx.GetData("DisableLoginForm") == true {
return ctx.ErrorRes(403, ctx.Tr("error.login-disabled-form"), nil) return ctx.ErrorRes(403, ctx.Tr("error.login-disabled-form"), nil)
} }
@ -189,7 +189,7 @@ func ProcessLogin(ctx *context.OGContext) error {
return ctx.RedirectTo("/") return ctx.RedirectTo("/")
} }
func Mfa(ctx *context.OGContext) error { func Mfa(ctx *context.Context) error {
var err error var err error
user := db.User{ID: ctx.GetSession().Values["mfaID"].(uint)} user := db.User{ID: ctx.GetSession().Values["mfaID"].(uint)}
@ -205,7 +205,7 @@ func Mfa(ctx *context.OGContext) error {
return ctx.HTML_("mfa.html") return ctx.HTML_("mfa.html")
} }
func OauthCallback(ctx *context.OGContext) error { func OauthCallback(ctx *context.Context) error {
user, err := gothic.CompleteUserAuth(ctx.Response(), ctx.Request()) user, err := gothic.CompleteUserAuth(ctx.Response(), ctx.Request())
if err != nil { if err != nil {
return ctx.ErrorRes(400, ctx.Tr("error.complete-oauth-login", err.Error()), err) return ctx.ErrorRes(400, ctx.Tr("error.complete-oauth-login", err.Error()), err)
@ -311,7 +311,7 @@ func OauthCallback(ctx *context.OGContext) error {
return ctx.RedirectTo("/") return ctx.RedirectTo("/")
} }
func Oauth(ctx *context.OGContext) error { func Oauth(ctx *context.Context) error {
provider := ctx.Param("provider") provider := ctx.Param("provider")
httpProtocol := "http" httpProtocol := "http"
@ -401,7 +401,7 @@ func Oauth(ctx *context.OGContext) error {
return nil return nil
} }
func OauthUnlink(ctx *context.OGContext) error { func OauthUnlink(ctx *context.Context) error {
provider := ctx.Param("provider") provider := ctx.Param("provider")
currUser := ctx.User currUser := ctx.User
@ -425,7 +425,7 @@ func OauthUnlink(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func BeginWebAuthnBinding(ctx *context.OGContext) error { func BeginWebAuthnBinding(ctx *context.Context) error {
credsCreation, jsonWaSession, err := webauthn.BeginBinding(ctx.User) credsCreation, jsonWaSession, err := webauthn.BeginBinding(ctx.User)
if err != nil { if err != nil {
return ctx.ErrorRes(500, "Cannot begin WebAuthn registration", err) return ctx.ErrorRes(500, "Cannot begin WebAuthn registration", err)
@ -439,7 +439,7 @@ func BeginWebAuthnBinding(ctx *context.OGContext) error {
return ctx.JSON(200, credsCreation) return ctx.JSON(200, credsCreation)
} }
func FinishWebAuthnBinding(ctx *context.OGContext) error { func FinishWebAuthnBinding(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
jsonWaSession, ok := sess.Values["webauthn_registration_session"].([]byte) jsonWaSession, ok := sess.Values["webauthn_registration_session"].([]byte)
if !ok { if !ok {
@ -483,7 +483,7 @@ func FinishWebAuthnBinding(ctx *context.OGContext) error {
return ctx.JSON_([]string{"OK"}) return ctx.JSON_([]string{"OK"})
} }
func BeginWebAuthnLogin(ctx *context.OGContext) error { func BeginWebAuthnLogin(ctx *context.Context) error {
credsCreation, jsonWaSession, err := webauthn.BeginDiscoverableLogin() credsCreation, jsonWaSession, err := webauthn.BeginDiscoverableLogin()
if err != nil { if err != nil {
return ctx.ErrorRes(401, "Cannot begin WebAuthn login", err) return ctx.ErrorRes(401, "Cannot begin WebAuthn login", err)
@ -497,7 +497,7 @@ func BeginWebAuthnLogin(ctx *context.OGContext) error {
return ctx.JSON_(credsCreation) return ctx.JSON_(credsCreation)
} }
func FinishWebAuthnLogin(ctx *context.OGContext) error { func FinishWebAuthnLogin(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
sessionData, ok := sess.Values["webauthn_login_session"].([]byte) sessionData, ok := sess.Values["webauthn_login_session"].([]byte)
if !ok { if !ok {
@ -518,7 +518,7 @@ func FinishWebAuthnLogin(ctx *context.OGContext) error {
return ctx.JSON_([]string{"OK"}) return ctx.JSON_([]string{"OK"})
} }
func BeginWebAuthnAssertion(ctx *context.OGContext) error { func BeginWebAuthnAssertion(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
ogUser, err := db.GetUserById(sess.Values["mfaID"].(uint)) ogUser, err := db.GetUserById(sess.Values["mfaID"].(uint))
@ -538,7 +538,7 @@ func BeginWebAuthnAssertion(ctx *context.OGContext) error {
return ctx.JSON_(credsCreation) return ctx.JSON_(credsCreation)
} }
func FinishWebAuthnAssertion(ctx *context.OGContext) error { func FinishWebAuthnAssertion(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
sessionData, ok := sess.Values["webauthn_assertion_session"].([]byte) sessionData, ok := sess.Values["webauthn_assertion_session"].([]byte)
if !ok { if !ok {
@ -566,7 +566,7 @@ func FinishWebAuthnAssertion(ctx *context.OGContext) error {
return ctx.JSON_([]string{"OK"}) return ctx.JSON_([]string{"OK"})
} }
func BeginTotp(ctx *context.OGContext) error { func BeginTotp(ctx *context.Context) error {
user := ctx.User user := ctx.User
if _, hasTotp, err := user.HasMFA(); err != nil { if _, hasTotp, err := user.HasMFA(); err != nil {
@ -599,7 +599,7 @@ func BeginTotp(ctx *context.OGContext) error {
} }
func FinishTotp(ctx *context.OGContext) error { func FinishTotp(ctx *context.Context) error {
user := ctx.User user := ctx.User
if _, hasTotp, err := user.HasMFA(); err != nil { if _, hasTotp, err := user.HasMFA(); err != nil {
@ -656,7 +656,7 @@ func FinishTotp(ctx *context.OGContext) error {
return ctx.HTML_("totp.html") return ctx.HTML_("totp.html")
} }
func AssertTotp(ctx *context.OGContext) error { func AssertTotp(ctx *context.Context) error {
var err error var err error
dto := &db.TOTPDTO{} dto := &db.TOTPDTO{}
if err := ctx.Bind(dto); err != nil { if err := ctx.Bind(dto); err != nil {
@ -704,7 +704,7 @@ func AssertTotp(ctx *context.OGContext) error {
return ctx.RedirectTo(redirectUrl) return ctx.RedirectTo(redirectUrl)
} }
func DisableTotp(ctx *context.OGContext) error { func DisableTotp(ctx *context.Context) error {
user := ctx.User user := ctx.User
userTotp, err := db.GetTOTPByUserID(user.ID) userTotp, err := db.GetTOTPByUserID(user.ID)
if err != nil { if err != nil {
@ -719,7 +719,7 @@ func DisableTotp(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func RegenerateTotpRecoveryCodes(ctx *context.OGContext) error { func RegenerateTotpRecoveryCodes(ctx *context.Context) error {
user := ctx.User user := ctx.User
userTotp, err := db.GetTOTPByUserID(user.ID) userTotp, err := db.GetTOTPByUserID(user.ID)
if err != nil { if err != nil {
@ -735,7 +735,7 @@ func RegenerateTotpRecoveryCodes(ctx *context.OGContext) error {
return ctx.HTML_("totp.html") return ctx.HTML_("totp.html")
} }
func Logout(ctx *context.OGContext) error { func Logout(ctx *context.Context) error {
ctx.DeleteSession() ctx.DeleteSession()
ctx.DeleteCsrfCookie() ctx.DeleteCsrfCookie()
return ctx.RedirectTo("/all") return ctx.RedirectTo("/all")
@ -803,7 +803,7 @@ func getAvatarUrlFromProvider(provider string, identifier string) string {
} }
type ContextAuthInfo struct { type ContextAuthInfo struct {
Context *context.OGContext Context *context.Context
} }
func (auth ContextAuthInfo) RequireLogin() (bool, error) { func (auth ContextAuthInfo) RequireLogin() (bool, error) {

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"archive/zip" "archive/zip"
@ -27,7 +27,7 @@ import (
"gorm.io/gorm" "gorm.io/gorm"
) )
func AllGists(ctx *context.OGContext) error { func AllGists(ctx *context.Context) error {
var err error var err error
var urlPage string var urlPage string
@ -158,7 +158,7 @@ func AllGists(ctx *context.OGContext) error {
return ctx.HTML_("all.html") return ctx.HTML_("all.html")
} }
func Search(ctx *context.OGContext) error { func Search(ctx *context.Context) error {
var err error var err error
content, meta := ParseSearchQueryStr(ctx.QueryParam("q")) content, meta := ParseSearchQueryStr(ctx.QueryParam("q"))
@ -221,7 +221,7 @@ func Search(ctx *context.OGContext) error {
return ctx.HTML_("search.html") return ctx.HTML_("search.html")
} }
func GistIndex(ctx *context.OGContext) error { func GistIndex(ctx *context.Context) error {
if ctx.GetData("gistpage") == "js" { if ctx.GetData("gistpage") == "js" {
return GistJs(ctx) return GistJs(ctx)
} else if ctx.GetData("gistpage") == "json" { } else if ctx.GetData("gistpage") == "json" {
@ -252,7 +252,7 @@ func GistIndex(ctx *context.OGContext) error {
return ctx.HTML_("gist.html") return ctx.HTML_("gist.html")
} }
func GistJson(ctx *context.OGContext) error { func GistJson(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
files, err := gist.Files("HEAD", true) files, err := gist.Files("HEAD", true)
if err != nil { if err != nil {
@ -297,7 +297,7 @@ func GistJson(ctx *context.OGContext) error {
}) })
} }
func GistJs(ctx *context.OGContext) error { func GistJs(ctx *context.Context) error {
if _, exists := ctx.QueryParams()["dark"]; exists { if _, exists := ctx.QueryParams()["dark"]; exists {
ctx.SetData("dark", "dark") ctx.SetData("dark", "dark")
} }
@ -331,7 +331,7 @@ func GistJs(ctx *context.OGContext) error {
return ctx.PlainText(200, js) return ctx.PlainText(200, js)
} }
func Revisions(ctx *context.OGContext) error { func Revisions(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
userName := gist.User.Username userName := gist.User.Username
gistName := gist.Identifier() gistName := gist.Identifier()
@ -368,12 +368,12 @@ func Revisions(ctx *context.OGContext) error {
return ctx.HTML_("revisions.html") return ctx.HTML_("revisions.html")
} }
func Create(ctx *context.OGContext) error { func Create(ctx *context.Context) error {
ctx.SetData("htmlTitle", ctx.TrH("gist.new.create-a-new-gist")) ctx.SetData("htmlTitle", ctx.TrH("gist.new.create-a-new-gist"))
return ctx.HTML_("create.html") return ctx.HTML_("create.html")
} }
func ProcessCreate(ctx *context.OGContext) error { func ProcessCreate(ctx *context.Context) error {
isCreate := false isCreate := false
if ctx.Request().URL.Path == "/" { if ctx.Request().URL.Path == "/" {
isCreate = true isCreate = true
@ -497,7 +497,7 @@ func ProcessCreate(ctx *context.OGContext) error {
return ctx.RedirectTo("/" + user.Username + "/" + gist.Identifier()) return ctx.RedirectTo("/" + user.Username + "/" + gist.Identifier())
} }
func EditVisibility(ctx *context.OGContext) error { func EditVisibility(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
dto := new(db.VisibilityDTO) dto := new(db.VisibilityDTO)
@ -514,7 +514,7 @@ func EditVisibility(ctx *context.OGContext) error {
return ctx.RedirectTo("/" + gist.User.Username + "/" + gist.Identifier()) return ctx.RedirectTo("/" + gist.User.Username + "/" + gist.Identifier())
} }
func DeleteGist(ctx *context.OGContext) error { func DeleteGist(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
if err := gist.Delete(); err != nil { if err := gist.Delete(); err != nil {
@ -526,7 +526,7 @@ func DeleteGist(ctx *context.OGContext) error {
return ctx.RedirectTo("/") return ctx.RedirectTo("/")
} }
func Like(ctx *context.OGContext) error { func Like(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
currentUser := ctx.User currentUser := ctx.User
@ -552,7 +552,7 @@ func Like(ctx *context.OGContext) error {
return ctx.RedirectTo(redirectTo) return ctx.RedirectTo(redirectTo)
} }
func Fork(ctx *context.OGContext) error { func Fork(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
currentUser := ctx.User currentUser := ctx.User
@ -603,7 +603,7 @@ func Fork(ctx *context.OGContext) error {
return ctx.RedirectTo("/" + currentUser.Username + "/" + newGist.Identifier()) return ctx.RedirectTo("/" + currentUser.Username + "/" + newGist.Identifier())
} }
func RawFile(ctx *context.OGContext) error { func RawFile(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
file, err := gist.File(ctx.Param("revision"), ctx.Param("file"), false) file, err := gist.File(ctx.Param("revision"), ctx.Param("file"), false)
if err != nil { if err != nil {
@ -617,7 +617,7 @@ func RawFile(ctx *context.OGContext) error {
return ctx.PlainText(200, file.Content) return ctx.PlainText(200, file.Content)
} }
func DownloadFile(ctx *context.OGContext) error { func DownloadFile(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
file, err := gist.File(ctx.Param("revision"), ctx.Param("file"), false) file, err := gist.File(ctx.Param("revision"), ctx.Param("file"), false)
if err != nil { if err != nil {
@ -639,7 +639,7 @@ func DownloadFile(ctx *context.OGContext) error {
return nil return nil
} }
func Edit(ctx *context.OGContext) error { func Edit(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
files, err := gist.Files("HEAD", false) files, err := gist.Files("HEAD", false)
@ -653,7 +653,7 @@ func Edit(ctx *context.OGContext) error {
return ctx.HTML_("edit.html") return ctx.HTML_("edit.html")
} }
func DownloadZip(ctx *context.OGContext) error { func DownloadZip(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
revision := ctx.Param("revision") revision := ctx.Param("revision")
@ -698,7 +698,7 @@ func DownloadZip(ctx *context.OGContext) error {
return nil return nil
} }
func Likes(ctx *context.OGContext) error { func Likes(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
pageInt := getPage(ctx) pageInt := getPage(ctx)
@ -717,7 +717,7 @@ func Likes(ctx *context.OGContext) error {
return ctx.HTML_("likes.html") return ctx.HTML_("likes.html")
} }
func Forks(ctx *context.OGContext) error { func Forks(ctx *context.Context) error {
gist := ctx.GetData("gist").(*db.Gist) gist := ctx.GetData("gist").(*db.Gist)
pageInt := getPage(ctx) pageInt := getPage(ctx)
@ -741,7 +741,7 @@ func Forks(ctx *context.OGContext) error {
return ctx.HTML_("forks.html") return ctx.HTML_("forks.html")
} }
func Checkbox(ctx *context.OGContext) error { func Checkbox(ctx *context.Context) error {
filename := ctx.FormValue("file") filename := ctx.FormValue("file")
checkboxNb := ctx.FormValue("checkbox") checkboxNb := ctx.FormValue("checkbox")
@ -777,7 +777,7 @@ func Checkbox(ctx *context.OGContext) error {
return ctx.PlainText(200, "ok") return ctx.PlainText(200, "ok")
} }
func Preview(ctx *context.OGContext) error { func Preview(ctx *context.Context) error {
content := ctx.FormValue("content") content := ctx.FormValue("content")
previewStr, err := render.MarkdownString(content) previewStr, err := render.MarkdownString(content)

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"bytes" "bytes"
@ -29,7 +29,7 @@ import (
var routes = []struct { var routes = []struct {
gitUrl string gitUrl string
method string method string
handler func(ctx *context.OGContext) error handler func(ctx *context.Context) error
}{ }{
{"(.*?)/git-upload-pack$", "POST", uploadPack}, {"(.*?)/git-upload-pack$", "POST", uploadPack},
{"(.*?)/git-receive-pack$", "POST", receivePack}, {"(.*?)/git-receive-pack$", "POST", receivePack},
@ -44,7 +44,7 @@ var routes = []struct {
{"(.*?)/objects/pack/pack-[0-9a-f]{40}\\.idx$", "GET", idxFile}, {"(.*?)/objects/pack/pack-[0-9a-f]{40}\\.idx$", "GET", idxFile},
} }
func GitHttp(ctx *context.OGContext) error { func GitHttp(ctx *context.Context) error {
for _, route := range routes { for _, route := range routes {
matched, _ := regexp.MatchString(route.gitUrl, ctx.Request().URL.Path) matched, _ := regexp.MatchString(route.gitUrl, ctx.Request().URL.Path)
if ctx.Request().Method == route.method && matched { if ctx.Request().Method == route.method && matched {
@ -179,15 +179,15 @@ func GitHttp(ctx *context.OGContext) error {
return ctx.NotFound("Gist not found") return ctx.NotFound("Gist not found")
} }
func uploadPack(ctx *context.OGContext) error { func uploadPack(ctx *context.Context) error {
return pack(ctx, "upload-pack") return pack(ctx, "upload-pack")
} }
func receivePack(ctx *context.OGContext) error { func receivePack(ctx *context.Context) error {
return pack(ctx, "receive-pack") return pack(ctx, "receive-pack")
} }
func pack(ctx *context.OGContext, serviceType string) error { func pack(ctx *context.Context, serviceType string) error {
noCacheHeaders(ctx) noCacheHeaders(ctx)
defer ctx.Request().Body.Close() defer ctx.Request().Body.Close()
@ -226,7 +226,7 @@ func pack(ctx *context.OGContext, serviceType string) error {
return nil return nil
} }
func infoRefs(ctx *context.OGContext) error { func infoRefs(ctx *context.Context) error {
noCacheHeaders(ctx) noCacheHeaders(ctx)
var service string var service string
@ -258,38 +258,38 @@ func infoRefs(ctx *context.OGContext) error {
return nil return nil
} }
func textFile(ctx *context.OGContext) error { func textFile(ctx *context.Context) error {
noCacheHeaders(ctx) noCacheHeaders(ctx)
return sendFile(ctx, "text/plain") return sendFile(ctx, "text/plain")
} }
func infoPacks(ctx *context.OGContext) error { func infoPacks(ctx *context.Context) error {
cacheHeadersForever(ctx) cacheHeadersForever(ctx)
return sendFile(ctx, "text/plain; charset=utf-8") return sendFile(ctx, "text/plain; charset=utf-8")
} }
func looseObject(ctx *context.OGContext) error { func looseObject(ctx *context.Context) error {
cacheHeadersForever(ctx) cacheHeadersForever(ctx)
return sendFile(ctx, "application/x-git-loose-object") return sendFile(ctx, "application/x-git-loose-object")
} }
func packFile(ctx *context.OGContext) error { func packFile(ctx *context.Context) error {
cacheHeadersForever(ctx) cacheHeadersForever(ctx)
return sendFile(ctx, "application/x-git-packed-objects") return sendFile(ctx, "application/x-git-packed-objects")
} }
func idxFile(ctx *context.OGContext) error { func idxFile(ctx *context.Context) error {
cacheHeadersForever(ctx) cacheHeadersForever(ctx)
return sendFile(ctx, "application/x-git-packed-objects-toc") return sendFile(ctx, "application/x-git-packed-objects-toc")
} }
func noCacheHeaders(ctx *context.OGContext) { func noCacheHeaders(ctx *context.Context) {
ctx.Response().Header().Set("Expires", "Thu, 01 Jan 1970 00:00:00 UTC") ctx.Response().Header().Set("Expires", "Thu, 01 Jan 1970 00:00:00 UTC")
ctx.Response().Header().Set("Pragma", "no-cache") ctx.Response().Header().Set("Pragma", "no-cache")
ctx.Response().Header().Set("Cache-Control", "no-cache, max-age=0, must-revalidate") ctx.Response().Header().Set("Cache-Control", "no-cache, max-age=0, must-revalidate")
} }
func cacheHeadersForever(ctx *context.OGContext) { func cacheHeadersForever(ctx *context.Context) {
now := time.Now().Unix() now := time.Now().Unix()
expires := now + 31536000 expires := now + 31536000
ctx.Response().Header().Set("Date", fmt.Sprintf("%d", now)) ctx.Response().Header().Set("Date", fmt.Sprintf("%d", now))
@ -297,7 +297,7 @@ func cacheHeadersForever(ctx *context.OGContext) {
ctx.Response().Header().Set("Cache-Control", "public, max-age=31536000") ctx.Response().Header().Set("Cache-Control", "public, max-age=31536000")
} }
func basicAuth(ctx *context.OGContext) error { func basicAuth(ctx *context.Context) error {
ctx.Response().Header().Set("WWW-Authenticate", `Basic realm="."`) ctx.Response().Header().Set("WWW-Authenticate", `Basic realm="."`)
return ctx.PlainText(401, "Requires authentication") return ctx.PlainText(401, "Requires authentication")
} }
@ -312,7 +312,7 @@ func basicAuthDecode(encoded string) (string, string, error) {
return auth[0], auth[1], nil return auth[0], auth[1], nil
} }
func sendFile(ctx *context.OGContext, contentType string) error { func sendFile(ctx *context.Context, contentType string) error {
gitFile := "/" + strings.Join(strings.Split(ctx.Request().URL.Path, "/")[3:], "/") gitFile := "/" + strings.Join(strings.Split(ctx.Request().URL.Path, "/")[3:], "/")
gitFile = path.Join(ctx.GetData("repositoryPath").(string), gitFile) gitFile = path.Join(ctx.GetData("repositoryPath").(string), gitFile)
fi, err := os.Stat(gitFile) fi, err := os.Stat(gitFile)

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"github.com/thomiceli/opengist/internal/db" "github.com/thomiceli/opengist/internal/db"
@ -6,7 +6,7 @@ import (
"time" "time"
) )
func Healthcheck(ctx *context.OGContext) error { func Healthcheck(ctx *context.Context) error {
// Check database connection // Check database connection
dbOk := "ok" dbOk := "ok"
httpStatus := 200 httpStatus := 200
@ -26,6 +26,6 @@ func Healthcheck(ctx *context.OGContext) error {
// Metrics is a dummy handler to satisfy the /metrics endpoint (for Prometheus, Openmetrics, etc.) // Metrics is a dummy handler to satisfy the /metrics endpoint (for Prometheus, Openmetrics, etc.)
// until we have a proper metrics endpoint // until we have a proper metrics endpoint
func Metrics(ctx *context.OGContext) error { func Metrics(ctx *context.Context) error {
return ctx.String(200, "") return ctx.String(200, "")
} }

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"crypto/md5" "crypto/md5"
@ -18,7 +18,7 @@ import (
"golang.org/x/crypto/ssh" "golang.org/x/crypto/ssh"
) )
func UserSettings(ctx *context.OGContext) error { func UserSettings(ctx *context.Context) error {
user := ctx.User user := ctx.User
keys, err := db.GetSSHKeysByUserID(user.ID) keys, err := db.GetSSHKeysByUserID(user.ID)
@ -46,7 +46,7 @@ func UserSettings(ctx *context.OGContext) error {
return ctx.HTML_("settings.html") return ctx.HTML_("settings.html")
} }
func EmailProcess(ctx *context.OGContext) error { func EmailProcess(ctx *context.Context) error {
user := ctx.User user := ctx.User
email := ctx.FormValue("email") email := ctx.FormValue("email")
var hash string var hash string
@ -69,7 +69,7 @@ func EmailProcess(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func AccountDeleteProcess(ctx *context.OGContext) error { func AccountDeleteProcess(ctx *context.Context) error {
user := ctx.User user := ctx.User
if err := user.Delete(); err != nil { if err := user.Delete(); err != nil {
@ -79,7 +79,7 @@ func AccountDeleteProcess(ctx *context.OGContext) error {
return ctx.RedirectTo("/all") return ctx.RedirectTo("/all")
} }
func SshKeysProcess(ctx *context.OGContext) error { func SshKeysProcess(ctx *context.Context) error {
user := ctx.User user := ctx.User
dto := new(db.SSHKeyDTO) dto := new(db.SSHKeyDTO)
@ -118,7 +118,7 @@ func SshKeysProcess(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func SshKeysDelete(ctx *context.OGContext) error { func SshKeysDelete(ctx *context.Context) error {
user := ctx.User user := ctx.User
keyId, err := strconv.Atoi(ctx.Param("id")) keyId, err := strconv.Atoi(ctx.Param("id"))
if err != nil { if err != nil {
@ -139,7 +139,7 @@ func SshKeysDelete(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func PasskeyDelete(ctx *context.OGContext) error { func PasskeyDelete(ctx *context.Context) error {
user := ctx.User user := ctx.User
keyId, err := strconv.Atoi(ctx.Param("id")) keyId, err := strconv.Atoi(ctx.Param("id"))
if err != nil { if err != nil {
@ -159,7 +159,7 @@ func PasskeyDelete(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func PasswordProcess(ctx *context.OGContext) error { func PasswordProcess(ctx *context.Context) error {
user := ctx.User user := ctx.User
dto := new(db.UserDTO) dto := new(db.UserDTO)
@ -187,7 +187,7 @@ func PasswordProcess(ctx *context.OGContext) error {
return ctx.RedirectTo("/settings") return ctx.RedirectTo("/settings")
} }
func UsernameProcess(ctx *context.OGContext) error { func UsernameProcess(ctx *context.Context) error {
user := ctx.User user := ctx.User
dto := new(db.UserDTO) dto := new(db.UserDTO)

View file

@ -1,4 +1,4 @@
package handler package handlers
import ( import (
"errors" "errors"
@ -8,7 +8,7 @@ import (
"strings" "strings"
) )
func getPage(ctx *context.OGContext) int { func getPage(ctx *context.Context) int {
page := ctx.QueryParam("page") page := ctx.QueryParam("page")
if page == "" { if page == "" {
page = "1" page = "1"
@ -22,7 +22,7 @@ func getPage(ctx *context.OGContext) int {
return pageInt return pageInt
} }
func paginate[T any](ctx *context.OGContext, data []*T, pageInt int, perPage int, templateDataName string, urlPage string, labels int, urlParams ...string) error { func paginate[T any](ctx *context.Context, data []*T, pageInt int, perPage int, templateDataName string, urlPage string, labels int, urlParams ...string) error {
lenData := len(data) lenData := len(data)
if lenData == 0 && pageInt != 1 { if lenData == 0 && pageInt != 1 {
return errors.New("page not found") return errors.New("page not found")

View file

@ -5,30 +5,30 @@ import (
"github.com/thomiceli/opengist/internal/web/context" "github.com/thomiceli/opengist/internal/web/context"
) )
type Handler func(ctx *context.OGContext) error type Handler func(ctx *context.Context) error
type Middleware func(next Handler) Handler type Middleware func(next Handler) Handler
func (h Handler) ToEcho() echo.HandlerFunc { func (h Handler) toEcho() echo.HandlerFunc {
return func(c echo.Context) error { return func(c echo.Context) error {
return h(c.(*context.OGContext)) return h(c.(*context.Context))
} }
} }
func (m Middleware) ToEcho() echo.MiddlewareFunc { func (m Middleware) toEcho() echo.MiddlewareFunc {
return func(next echo.HandlerFunc) echo.HandlerFunc { return func(next echo.HandlerFunc) echo.HandlerFunc {
return m(func(c *context.OGContext) error { return m(func(c *context.Context) error {
return next(c) return next(c)
}).ToEcho() }).toEcho()
} }
} }
func (h Handler) toEchoHandler() echo.HandlerFunc { func (h Handler) toEchoHandler() echo.HandlerFunc {
return func(c echo.Context) error { return func(c echo.Context) error {
if ogc, ok := c.(*context.OGContext); ok { if ogc, ok := c.(*context.Context); ok {
return h(ogc) return h(ogc)
} }
// Could also add error handling for incorrect context type // Could also add error handling for incorrect context type
return h(c.(*context.OGContext)) return h(c.(*context.Context))
} }
} }

View file

@ -11,7 +11,7 @@ import (
"github.com/thomiceli/opengist/internal/db" "github.com/thomiceli/opengist/internal/db"
"github.com/thomiceli/opengist/internal/i18n" "github.com/thomiceli/opengist/internal/i18n"
"github.com/thomiceli/opengist/internal/web/context" "github.com/thomiceli/opengist/internal/web/context"
"github.com/thomiceli/opengist/internal/web/handler" "github.com/thomiceli/opengist/internal/web/handlers"
"golang.org/x/text/cases" "golang.org/x/text/cases"
"golang.org/x/text/language" "golang.org/x/text/language"
"html/template" "html/template"
@ -31,8 +31,8 @@ func (s *Server) useCustomContext() {
} }
func (s *Server) registerMiddlewares() { func (s *Server) registerMiddlewares() {
s.echo.Use(Middleware(dataInit).ToEcho()) s.echo.Use(Middleware(dataInit).toEcho())
s.echo.Use(Middleware(locale).ToEcho()) s.echo.Use(Middleware(locale).toEcho())
s.echo.Pre(middleware.MethodOverrideWithConfig(middleware.MethodOverrideConfig{ s.echo.Pre(middleware.MethodOverrideWithConfig(middleware.MethodOverrideConfig{
Getter: middleware.MethodFromForm("_method"), Getter: middleware.MethodFromForm("_method"),
@ -50,8 +50,7 @@ func (s *Server) registerMiddlewares() {
})) }))
s.echo.Use(middleware.Recover()) s.echo.Use(middleware.Recover())
s.echo.Use(middleware.Secure()) s.echo.Use(middleware.Secure())
s.echo.Use(Middleware(sessionInit).toEcho())
s.echo.Use(Middleware(sessionInit).ToEcho())
if !s.ignoreCsrf { if !s.ignoreCsrf {
s.echo.Use(middleware.CSRFWithConfig(middleware.CSRFConfig{ s.echo.Use(middleware.CSRFWithConfig(middleware.CSRFConfig{
@ -60,7 +59,7 @@ func (s *Server) registerMiddlewares() {
CookieHTTPOnly: true, CookieHTTPOnly: true,
CookieSameSite: http.SameSiteStrictMode, CookieSameSite: http.SameSiteStrictMode,
})) }))
s.echo.Use(Middleware(csrfInit).ToEcho()) s.echo.Use(Middleware(csrfInit).toEcho())
} }
} }
@ -87,7 +86,7 @@ func (s *Server) errorHandler(err error, ctx echo.Context) {
} }
func dataInit(next Handler) Handler { func dataInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
ctx.SetData("loadStartTime", time.Now()) ctx.SetData("loadStartTime", time.Now())
if err := loadSettings(ctx); err != nil { if err := loadSettings(ctx); err != nil {
@ -122,7 +121,7 @@ func dataInit(next Handler) Handler {
} }
func writePermission(next Handler) Handler { func writePermission(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
gist := ctx.GetData("gist") gist := ctx.GetData("gist")
user := ctx.User user := ctx.User
if !gist.(*db.Gist).CanWrite(user) { if !gist.(*db.Gist).CanWrite(user) {
@ -133,7 +132,7 @@ func writePermission(next Handler) Handler {
} }
func adminPermission(next Handler) Handler { func adminPermission(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
user := ctx.User user := ctx.User
if user == nil || !user.IsAdmin { if user == nil || !user.IsAdmin {
return ctx.NotFound("User not found") return ctx.NotFound("User not found")
@ -143,7 +142,7 @@ func adminPermission(next Handler) Handler {
} }
func logged(next Handler) Handler { func logged(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
user := ctx.User user := ctx.User
if user != nil { if user != nil {
return next(ctx) return next(ctx)
@ -153,7 +152,7 @@ func logged(next Handler) Handler {
} }
func inMFASession(next Handler) Handler { func inMFASession(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
_, ok := sess.Values["mfaID"].(uint) _, ok := sess.Values["mfaID"].(uint)
if !ok { if !ok {
@ -165,12 +164,12 @@ func inMFASession(next Handler) Handler {
func makeCheckRequireLogin(isSingleGistAccess bool) Middleware { func makeCheckRequireLogin(isSingleGistAccess bool) Middleware {
return func(next Handler) Handler { return func(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
if user := ctx.User; user != nil { if user := ctx.User; user != nil {
return next(ctx) return next(ctx)
} }
allow, err := auth.ShouldAllowUnauthenticatedGistAccess(handler.ContextAuthInfo{Context: ctx}, isSingleGistAccess) allow, err := auth.ShouldAllowUnauthenticatedGistAccess(handlers.ContextAuthInfo{Context: ctx}, isSingleGistAccess)
if err != nil { if err != nil {
log.Fatal().Err(err).Msg("Failed to check if unauthenticated access is allowed") log.Fatal().Err(err).Msg("Failed to check if unauthenticated access is allowed")
} }
@ -188,12 +187,12 @@ func checkRequireLogin(next Handler) Handler {
return makeCheckRequireLogin(false)(next) return makeCheckRequireLogin(false)(next)
} }
func noRouteFound(ctx *context.OGContext) error { func noRouteFound(ctx *context.Context) error {
return ctx.NotFound("Page not found") return ctx.NotFound("Page not found")
} }
func locale(next Handler) Handler { func locale(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
// Check URL arguments // Check URL arguments
lang := ctx.Request().URL.Query().Get("lang") lang := ctx.Request().URL.Query().Get("lang")
changeLang := lang != "" changeLang := lang != ""
@ -236,7 +235,7 @@ func locale(next Handler) Handler {
} }
func sessionInit(next Handler) Handler { func sessionInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
sess := ctx.GetSession() sess := ctx.GetSession()
if sess.Values["user"] != nil { if sess.Values["user"] != nil {
var err error var err error
@ -263,7 +262,7 @@ func sessionInit(next Handler) Handler {
} }
func csrfInit(next Handler) Handler { func csrfInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
var csrf string var csrf string
if csrfToken, ok := ctx.Get("csrf").(string); ok { if csrfToken, ok := ctx.Get("csrf").(string); ok {
csrf = csrfToken csrf = csrfToken
@ -275,7 +274,7 @@ func csrfInit(next Handler) Handler {
} }
} }
func loadSettings(ctx *context.OGContext) error { func loadSettings(ctx *context.Context) error {
settings, err := db.GetSettings() settings, err := db.GetSettings()
if err != nil { if err != nil {
return err return err
@ -289,8 +288,8 @@ func loadSettings(ctx *context.OGContext) error {
return nil return nil
} }
func GistInit(next Handler) Handler { func gistInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
currUser := ctx.User currUser := ctx.User
userName := ctx.Param("user") userName := ctx.Param("user")
@ -369,10 +368,10 @@ func GistInit(next Handler) Handler {
} }
} }
// GistSoftInit try to load a gist (same as gistInit) but does not return a 404 if the gist is not found // gistSoftInit try to load a gist (same as gistInit) but does not return a 404 if the gist is not found
// useful for git clients using HTTP to obfuscate the existence of a private gist // useful for git clients using HTTP to obfuscate the existence of a private gist
func GistSoftInit(next Handler) Handler { func gistSoftInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
userName := ctx.Param("user") userName := ctx.Param("user")
gistName := ctx.Param("gistname") gistName := ctx.Param("gistname")
@ -385,9 +384,9 @@ func GistSoftInit(next Handler) Handler {
} }
} }
// GistNewPushSoftInit has the same behavior as gistSoftInit but create a new gist empty instead // gistNewPushSoftInit has the same behavior as gistSoftInit but create a new gist empty instead
func GistNewPushSoftInit(next Handler) Handler { func gistNewPushSoftInit(next Handler) Handler {
return func(ctx *context.OGContext) error { return func(ctx *context.Context) error {
ctx.SetData("gist", new(db.Gist)) ctx.SetData("gist", new(db.Gist))
return next(ctx) return next(ctx)
} }

View file

@ -11,7 +11,7 @@ import (
"github.com/thomiceli/opengist/internal/git" "github.com/thomiceli/opengist/internal/git"
"github.com/thomiceli/opengist/internal/index" "github.com/thomiceli/opengist/internal/index"
"github.com/thomiceli/opengist/internal/web/context" "github.com/thomiceli/opengist/internal/web/context"
"github.com/thomiceli/opengist/internal/web/handler" "github.com/thomiceli/opengist/internal/web/handlers"
"github.com/thomiceli/opengist/public" "github.com/thomiceli/opengist/public"
"github.com/thomiceli/opengist/templates" "github.com/thomiceli/opengist/templates"
htmlpkg "html" htmlpkg "html"
@ -158,7 +158,7 @@ func (s *Server) setFuncMap() {
return dict, nil return dict, nil
}, },
"addMetadataToSearchQuery": func(input, key, value string) string { "addMetadataToSearchQuery": func(input, key, value string) string {
content, metadata := handler.ParseSearchQueryStr(input) content, metadata := handlers.ParseSearchQueryStr(input)
metadata[key] = value metadata[key] = value

View file

@ -5,7 +5,7 @@ import (
"github.com/thomiceli/opengist/internal/config" "github.com/thomiceli/opengist/internal/config"
"github.com/thomiceli/opengist/internal/index" "github.com/thomiceli/opengist/internal/index"
"github.com/thomiceli/opengist/internal/web/context" "github.com/thomiceli/opengist/internal/web/context"
"github.com/thomiceli/opengist/internal/web/handler" "github.com/thomiceli/opengist/internal/web/handlers"
"github.com/thomiceli/opengist/public" "github.com/thomiceli/opengist/public"
"net/http" "net/http"
"os" "os"
@ -19,107 +19,107 @@ func (s *Server) registerRoutes() {
r := NewRouter(s.echo.Group("")) r := NewRouter(s.echo.Group(""))
{ {
r.GET("/", handler.Create, logged) r.GET("/", handlers.Create, logged)
r.POST("/", handler.ProcessCreate, logged) r.POST("/", handlers.ProcessCreate, logged)
r.POST("/preview", handler.Preview, logged) r.POST("/preview", handlers.Preview, logged)
r.GET("/healthcheck", handler.Healthcheck) r.GET("/healthcheck", handlers.Healthcheck)
r.GET("/metrics", handler.Metrics) r.GET("/metrics", handlers.Metrics)
r.GET("/register", handler.Register) r.GET("/register", handlers.Register)
r.POST("/register", handler.ProcessRegister) r.POST("/register", handlers.ProcessRegister)
r.GET("/login", handler.Login) r.GET("/login", handlers.Login)
r.POST("/login", handler.ProcessLogin) r.POST("/login", handlers.ProcessLogin)
r.GET("/logout", handler.Logout) r.GET("/logout", handlers.Logout)
r.GET("/oauth/:provider", handler.Oauth) r.GET("/oauth/:provider", handlers.Oauth)
r.GET("/oauth/:provider/callback", handler.OauthCallback) r.GET("/oauth/:provider/callback", handlers.OauthCallback)
r.GET("/oauth/:provider/unlink", handler.OauthUnlink, logged) r.GET("/oauth/:provider/unlink", handlers.OauthUnlink, logged)
r.POST("/webauthn/bind", handler.BeginWebAuthnBinding, logged) r.POST("/webauthn/bind", handlers.BeginWebAuthnBinding, logged)
r.POST("/webauthn/bind/finish", handler.FinishWebAuthnBinding, logged) r.POST("/webauthn/bind/finish", handlers.FinishWebAuthnBinding, logged)
r.POST("/webauthn/login", handler.BeginWebAuthnLogin) r.POST("/webauthn/login", handlers.BeginWebAuthnLogin)
r.POST("/webauthn/login/finish", handler.FinishWebAuthnLogin) r.POST("/webauthn/login/finish", handlers.FinishWebAuthnLogin)
r.POST("/webauthn/assertion", handler.BeginWebAuthnAssertion, inMFASession) r.POST("/webauthn/assertion", handlers.BeginWebAuthnAssertion, inMFASession)
r.POST("/webauthn/assertion/finish", handler.FinishWebAuthnAssertion, inMFASession) r.POST("/webauthn/assertion/finish", handlers.FinishWebAuthnAssertion, inMFASession)
r.GET("/mfa", handler.Mfa, inMFASession) r.GET("/mfa", handlers.Mfa, inMFASession)
r.POST("/mfa/totp/assertion", handler.AssertTotp, inMFASession) r.POST("/mfa/totp/assertion", handlers.AssertTotp, inMFASession)
sA := r.SubGroup("/settings") sA := r.SubGroup("/settings")
{ {
sA.Use(logged) sA.Use(logged)
sA.GET("", handler.UserSettings) sA.GET("", handlers.UserSettings)
sA.POST("/email", handler.EmailProcess) sA.POST("/email", handlers.EmailProcess)
sA.DELETE("/account", handler.AccountDeleteProcess) sA.DELETE("/account", handlers.AccountDeleteProcess)
sA.POST("/ssh-keys", handler.SshKeysProcess) sA.POST("/ssh-keys", handlers.SshKeysProcess)
sA.DELETE("/ssh-keys/:id", handler.SshKeysDelete) sA.DELETE("/ssh-keys/:id", handlers.SshKeysDelete)
sA.DELETE("/passkeys/:id", handler.PasskeyDelete) sA.DELETE("/passkeys/:id", handlers.PasskeyDelete)
sA.PUT("/password", handler.PasswordProcess) sA.PUT("/password", handlers.PasswordProcess)
sA.PUT("/username", handler.UsernameProcess) sA.PUT("/username", handlers.UsernameProcess)
sA.GET("/totp/generate", handler.BeginTotp) sA.GET("/totp/generate", handlers.BeginTotp)
sA.POST("/totp/generate", handler.FinishTotp) sA.POST("/totp/generate", handlers.FinishTotp)
sA.DELETE("/totp", handler.DisableTotp) sA.DELETE("/totp", handlers.DisableTotp)
sA.POST("/totp/regenerate", handler.RegenerateTotpRecoveryCodes) sA.POST("/totp/regenerate", handlers.RegenerateTotpRecoveryCodes)
} }
sB := r.SubGroup("/admin-panel") sB := r.SubGroup("/admin-panel")
{ {
sB.Use(adminPermission) sB.Use(adminPermission)
sB.GET("", handler.AdminIndex) sB.GET("", handlers.AdminIndex)
sB.GET("/users", handler.AdminUsers) sB.GET("/users", handlers.AdminUsers)
sB.POST("/users/:user/delete", handler.AdminUserDelete) sB.POST("/users/:user/delete", handlers.AdminUserDelete)
sB.GET("/gists", handler.AdminGists) sB.GET("/gists", handlers.AdminGists)
sB.POST("/gists/:gist/delete", handler.AdminGistDelete) sB.POST("/gists/:gist/delete", handlers.AdminGistDelete)
sB.GET("/invitations", handler.AdminInvitations) sB.GET("/invitations", handlers.AdminInvitations)
sB.POST("/invitations", handler.AdminInvitationsCreate) sB.POST("/invitations", handlers.AdminInvitationsCreate)
sB.POST("/invitations/:id/delete", handler.AdminInvitationsDelete) sB.POST("/invitations/:id/delete", handlers.AdminInvitationsDelete)
sB.POST("/sync-fs", handler.AdminSyncReposFromFS) sB.POST("/sync-fs", handlers.AdminSyncReposFromFS)
sB.POST("/sync-db", handler.AdminSyncReposFromDB) sB.POST("/sync-db", handlers.AdminSyncReposFromDB)
sB.POST("/gc-repos", handler.AdminGcRepos) sB.POST("/gc-repos", handlers.AdminGcRepos)
sB.POST("/sync-previews", handler.AdminSyncGistPreviews) sB.POST("/sync-previews", handlers.AdminSyncGistPreviews)
sB.POST("/reset-hooks", handler.AdminResetHooks) sB.POST("/reset-hooks", handlers.AdminResetHooks)
sB.POST("/index-gists", handler.AdminIndexGists) sB.POST("/index-gists", handlers.AdminIndexGists)
sB.GET("/configuration", handler.AdminConfig) sB.GET("/configuration", handlers.AdminConfig)
sB.PUT("/set-config", handler.AdminSetConfig) sB.PUT("/set-config", handlers.AdminSetConfig)
} }
if config.C.HttpGit { if config.C.HttpGit {
r.Any("/init/*", handler.GitHttp, GistNewPushSoftInit) r.Any("/init/*", handlers.GitHttp, gistNewPushSoftInit)
} }
r.GET("/all", handler.AllGists, checkRequireLogin) r.GET("/all", handlers.AllGists, checkRequireLogin)
if index.Enabled() { if index.Enabled() {
r.GET("/search", handler.Search, checkRequireLogin) r.GET("/search", handlers.Search, checkRequireLogin)
} else { } else {
r.GET("/search", handler.AllGists, checkRequireLogin) r.GET("/search", handlers.AllGists, checkRequireLogin)
} }
r.GET("/:user", handler.AllGists, checkRequireLogin) r.GET("/:user", handlers.AllGists, checkRequireLogin)
r.GET("/:user/liked", handler.AllGists, checkRequireLogin) r.GET("/:user/liked", handlers.AllGists, checkRequireLogin)
r.GET("/:user/forked", handler.AllGists, checkRequireLogin) r.GET("/:user/forked", handlers.AllGists, checkRequireLogin)
sC := r.SubGroup("/:user/:gistname") sC := r.SubGroup("/:user/:gistname")
{ {
sC.Use(makeCheckRequireLogin(true), GistInit) sC.Use(makeCheckRequireLogin(true), gistInit)
sC.GET("", handler.GistIndex) sC.GET("", handlers.GistIndex)
sC.GET("/rev/:revision", handler.GistIndex) sC.GET("/rev/:revision", handlers.GistIndex)
sC.GET("/revisions", handler.Revisions) sC.GET("/revisions", handlers.Revisions)
sC.GET("/archive/:revision", handler.DownloadZip) sC.GET("/archive/:revision", handlers.DownloadZip)
sC.POST("/visibility", handler.EditVisibility, logged, writePermission) sC.POST("/visibility", handlers.EditVisibility, logged, writePermission)
sC.POST("/delete", handler.DeleteGist, logged, writePermission) sC.POST("/delete", handlers.DeleteGist, logged, writePermission)
sC.GET("/raw/:revision/:file", handler.RawFile) sC.GET("/raw/:revision/:file", handlers.RawFile)
sC.GET("/download/:revision/:file", handler.DownloadFile) sC.GET("/download/:revision/:file", handlers.DownloadFile)
sC.GET("/edit", handler.Edit, logged, writePermission) sC.GET("/edit", handlers.Edit, logged, writePermission)
sC.POST("/edit", handler.ProcessCreate, logged, writePermission) sC.POST("/edit", handlers.ProcessCreate, logged, writePermission)
sC.POST("/like", handler.Like, logged) sC.POST("/like", handlers.Like, logged)
sC.GET("/likes", handler.Likes, checkRequireLogin) sC.GET("/likes", handlers.Likes, checkRequireLogin)
sC.POST("/fork", handler.Fork, logged) sC.POST("/fork", handlers.Fork, logged)
sC.GET("/forks", handler.Forks, checkRequireLogin) sC.GET("/forks", handlers.Forks, checkRequireLogin)
sC.PUT("/checkbox", handler.Checkbox, logged, writePermission) sC.PUT("/checkbox", handlers.Checkbox, logged, writePermission)
} }
} }
customFs := os.DirFS(filepath.Join(config.GetHomeDir(), "custom")) customFs := os.DirFS(filepath.Join(config.GetHomeDir(), "custom"))
r.GET("/assets/*", func(ctx *context.OGContext) error { r.GET("/assets/*", func(ctx *context.Context) error {
if _, err := public.Files.Open(path.Join("assets", ctx.Param("*"))); !s.dev && err == nil { if _, err := public.Files.Open(path.Join("assets", ctx.Param("*"))); !s.dev && err == nil {
ctx.Response().Header().Set("Cache-Control", "public, max-age=31536000") ctx.Response().Header().Set("Cache-Control", "public, max-age=31536000")
ctx.Response().Header().Set("Expires", time.Now().AddDate(1, 0, 0).Format(http.TimeFormat)) ctx.Response().Header().Set("Expires", time.Now().AddDate(1, 0, 0).Format(http.TimeFormat))
@ -140,7 +140,7 @@ func (s *Server) registerRoutes() {
// Git HTTP routes // Git HTTP routes
if config.C.HttpGit { if config.C.HttpGit {
r.Any("/:user/:gistname/*", handler.GitHttp, GistSoftInit) r.Any("/:user/:gistname/*", handlers.GitHttp, gistSoftInit)
} }
r.Any("/*", noRouteFound) r.Any("/*", noRouteFound)
@ -151,19 +151,16 @@ type Router struct {
*echo.Group *echo.Group
} }
// NewRouter creates a new Router instance
func NewRouter(g *echo.Group) *Router { func NewRouter(g *echo.Group) *Router {
return &Router{Group: g} return &Router{Group: g}
} }
// SubGroup returns a new Router group with the given prefix and middleware
func (r *Router) SubGroup(prefix string, m ...Middleware) *Router { func (r *Router) SubGroup(prefix string, m ...Middleware) *Router {
// Convert middleware only when creating group
echoMiddleware := make([]echo.MiddlewareFunc, len(m)) echoMiddleware := make([]echo.MiddlewareFunc, len(m))
for i, mw := range m { for i, mw := range m {
mw := mw // capture for closure mw := mw // capture for closure
echoMiddleware[i] = func(next echo.HandlerFunc) echo.HandlerFunc { echoMiddleware[i] = func(next echo.HandlerFunc) echo.HandlerFunc {
return chain(func(c *context.OGContext) error { return chain(func(c *context.Context) error {
return next(c) return next(c)
}, mw).toEchoHandler() }, mw).toEchoHandler()
} }
@ -199,7 +196,7 @@ func (r *Router) Use(middleware ...Middleware) {
for _, m := range middleware { for _, m := range middleware {
m := m // capture for closure m := m // capture for closure
r.Group.Use(func(next echo.HandlerFunc) echo.HandlerFunc { r.Group.Use(func(next echo.HandlerFunc) echo.HandlerFunc {
return chain(func(c *context.OGContext) error { return chain(func(c *context.Context) error {
return next(c) return next(c)
}, m).toEchoHandler() }, m).toEchoHandler()
}) })

View file

@ -1,10 +1,8 @@
package server package server
import ( import (
"errors"
"github.com/thomiceli/opengist/internal/utils" "github.com/thomiceli/opengist/internal/utils"
"net/http" "net/http"
"strings"
"github.com/gorilla/sessions" "github.com/gorilla/sessions"
"github.com/labstack/echo/v4" "github.com/labstack/echo/v4"
@ -27,6 +25,7 @@ func NewServer(isDev bool, sessionsPath string, ignoreCsrf bool) *Server {
e := echo.New() e := echo.New()
e.HideBanner = true e.HideBanner = true
e.HidePort = true e.HidePort = true
e.Validator = utils.NewValidator()
s := &Server{echo: e, dev: isDev, sessionsPath: sessionsPath, ignoreCsrf: ignoreCsrf} s := &Server{echo: e, dev: isDev, sessionsPath: sessionsPath, ignoreCsrf: ignoreCsrf}
@ -40,8 +39,6 @@ func NewServer(isDev bool, sessionsPath string, ignoreCsrf bool) *Server {
s.setFuncMap() s.setFuncMap()
s.echo.HTTPErrorHandler = s.errorHandler s.echo.HTTPErrorHandler = s.errorHandler
e.Validator = utils.NewValidator()
if !s.dev { if !s.dev {
s.parseManifestEntries() s.parseManifestEntries()
} }