mirror of
https://github.com/superseriousbusiness/gotosocial.git
synced 2024-11-25 21:26:40 +00:00
Require confirmed email when checking oauth token (#332)
* move token checker to security package * update tests with new security package * add oauth token checking to security package * check if user email confirmed when parsing token
This commit is contained in:
parent
5ed03480e7
commit
ce22e03f9d
|
@ -81,7 +81,5 @@ func (m *Module) Route(s router.Router) error {
|
||||||
s.AttachHandler(http.MethodPost, OauthAuthorizePath, m.AuthorizePOSTHandler)
|
s.AttachHandler(http.MethodPost, OauthAuthorizePath, m.AuthorizePOSTHandler)
|
||||||
|
|
||||||
s.AttachHandler(http.MethodGet, CallbackPath, m.CallbackGETHandler)
|
s.AttachHandler(http.MethodGet, CallbackPath, m.CallbackGETHandler)
|
||||||
|
|
||||||
s.AttachMiddleware(m.OauthTokenMiddleware)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,7 @@
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/email"
|
"github.com/superseriousbusiness/gotosocial/internal/email"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/federation"
|
"github.com/superseriousbusiness/gotosocial/internal/federation"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/processing"
|
"github.com/superseriousbusiness/gotosocial/internal/processing"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/typeutils"
|
"github.com/superseriousbusiness/gotosocial/internal/typeutils"
|
||||||
"github.com/superseriousbusiness/gotosocial/testrig"
|
"github.com/superseriousbusiness/gotosocial/testrig"
|
||||||
|
@ -43,6 +44,7 @@ type UserStandardTestSuite struct {
|
||||||
emailSender email.Sender
|
emailSender email.Sender
|
||||||
processor processing.Processor
|
processor processing.Processor
|
||||||
storage *kv.KVStore
|
storage *kv.KVStore
|
||||||
|
oauthServer oauth.Server
|
||||||
securityModule *security.Module
|
securityModule *security.Module
|
||||||
|
|
||||||
// standard suite models
|
// standard suite models
|
||||||
|
@ -80,7 +82,8 @@ func (suite *UserStandardTestSuite) SetupTest() {
|
||||||
suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil)
|
suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil)
|
||||||
suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender)
|
suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender)
|
||||||
suite.userModule = user.New(suite.config, suite.processor).(*user.Module)
|
suite.userModule = user.New(suite.config, suite.processor).(*user.Module)
|
||||||
suite.securityModule = security.New(suite.config, suite.db).(*security.Module)
|
suite.oauthServer = testrig.NewTestOauthServer(suite.db)
|
||||||
|
suite.securityModule = security.New(suite.config, suite.db, suite.oauthServer).(*security.Module)
|
||||||
testrig.StandardDBSetup(suite.db, suite.testAccounts)
|
testrig.StandardDBSetup(suite.db, suite.testAccounts)
|
||||||
testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media")
|
testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media")
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/email"
|
"github.com/superseriousbusiness/gotosocial/internal/email"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/federation"
|
"github.com/superseriousbusiness/gotosocial/internal/federation"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/processing"
|
"github.com/superseriousbusiness/gotosocial/internal/processing"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/typeutils"
|
"github.com/superseriousbusiness/gotosocial/internal/typeutils"
|
||||||
"github.com/superseriousbusiness/gotosocial/testrig"
|
"github.com/superseriousbusiness/gotosocial/testrig"
|
||||||
|
@ -48,6 +49,7 @@ type WebfingerStandardTestSuite struct {
|
||||||
emailSender email.Sender
|
emailSender email.Sender
|
||||||
processor processing.Processor
|
processor processing.Processor
|
||||||
storage *kv.KVStore
|
storage *kv.KVStore
|
||||||
|
oauthServer oauth.Server
|
||||||
securityModule *security.Module
|
securityModule *security.Module
|
||||||
|
|
||||||
// standard suite models
|
// standard suite models
|
||||||
|
@ -83,7 +85,8 @@ func (suite *WebfingerStandardTestSuite) SetupTest() {
|
||||||
suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil)
|
suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil)
|
||||||
suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender)
|
suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender)
|
||||||
suite.webfingerModule = webfinger.New(suite.config, suite.processor).(*webfinger.Module)
|
suite.webfingerModule = webfinger.New(suite.config, suite.processor).(*webfinger.Module)
|
||||||
suite.securityModule = security.New(suite.config, suite.db).(*security.Module)
|
suite.oauthServer = testrig.NewTestOauthServer(suite.db)
|
||||||
|
suite.securityModule = security.New(suite.config, suite.db, suite.oauthServer).(*security.Module)
|
||||||
testrig.StandardDBSetup(suite.db, suite.testAccounts)
|
testrig.StandardDBSetup(suite.db, suite.testAccounts)
|
||||||
testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media")
|
testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media")
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/api"
|
"github.com/superseriousbusiness/gotosocial/internal/api"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/config"
|
"github.com/superseriousbusiness/gotosocial/internal/config"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/db"
|
"github.com/superseriousbusiness/gotosocial/internal/db"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/router"
|
"github.com/superseriousbusiness/gotosocial/internal/router"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -33,13 +34,15 @@
|
||||||
type Module struct {
|
type Module struct {
|
||||||
config *config.Config
|
config *config.Config
|
||||||
db db.DB
|
db db.DB
|
||||||
|
server oauth.Server
|
||||||
}
|
}
|
||||||
|
|
||||||
// New returns a new security module
|
// New returns a new security module
|
||||||
func New(config *config.Config, db db.DB) api.ClientModule {
|
func New(config *config.Config, db db.DB, server oauth.Server) api.ClientModule {
|
||||||
return &Module{
|
return &Module{
|
||||||
config: config,
|
config: config,
|
||||||
db: db,
|
db: db,
|
||||||
|
server: server,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -49,6 +52,7 @@ func (m *Module) Route(s router.Router) error {
|
||||||
s.AttachMiddleware(m.FlocBlock)
|
s.AttachMiddleware(m.FlocBlock)
|
||||||
s.AttachMiddleware(m.ExtraHeaders)
|
s.AttachMiddleware(m.ExtraHeaders)
|
||||||
s.AttachMiddleware(m.UserAgentBlock)
|
s.AttachMiddleware(m.UserAgentBlock)
|
||||||
|
s.AttachMiddleware(m.TokenCheck)
|
||||||
s.AttachHandler(http.MethodGet, robotsPath, m.RobotsGETHandler)
|
s.AttachHandler(http.MethodGet, robotsPath, m.RobotsGETHandler)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package auth
|
package security
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
@ -26,55 +26,71 @@
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
)
|
)
|
||||||
|
|
||||||
// OauthTokenMiddleware checks if the client has presented a valid oauth Bearer token.
|
// TokenCheck checks if the client has presented a valid oauth Bearer token.
|
||||||
// If so, it will check the User that the token belongs to, and set that in the context of
|
// If so, it will check the User that the token belongs to, and set that in the context of
|
||||||
// the request. Then, it will look up the account for that user, and set that in the request too.
|
// the request. Then, it will look up the account for that user, and set that in the request too.
|
||||||
// If user or account can't be found, then the handler won't *fail*, in case the server wants to allow
|
// If user or account can't be found, then the handler won't *fail*, in case the server wants to allow
|
||||||
// public requests that don't have a Bearer token set (eg., for public instance information and so on).
|
// public requests that don't have a Bearer token set (eg., for public instance information and so on).
|
||||||
func (m *Module) OauthTokenMiddleware(c *gin.Context) {
|
func (m *Module) TokenCheck(c *gin.Context) {
|
||||||
l := logrus.WithField("func", "OauthTokenMiddleware")
|
l := logrus.WithField("func", "OauthTokenMiddleware")
|
||||||
l.Trace("entering OauthTokenMiddleware")
|
ctx := c.Request.Context()
|
||||||
|
defer c.Next()
|
||||||
|
|
||||||
|
if c.Request.Header.Get("Authorization") == "" {
|
||||||
|
// no token set in the header, we can just bail
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
ti, err := m.server.ValidationBearerToken(c.Copy().Request)
|
ti, err := m.server.ValidationBearerToken(c.Copy().Request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
l.Tracef("could not validate token: %s", err)
|
l.Infof("token was passed in Authorization header but we could not validate it: %s", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
l.Trace("continuing with unauthenticated request")
|
|
||||||
c.Set(oauth.SessionAuthorizedToken, ti)
|
c.Set(oauth.SessionAuthorizedToken, ti)
|
||||||
l.Tracef("set gin context %s to %+v", oauth.SessionAuthorizedToken, ti)
|
|
||||||
|
|
||||||
// check for user-level token
|
// check for user-level token
|
||||||
if uid := ti.GetUserID(); uid != "" {
|
if userID := ti.GetUserID(); userID != "" {
|
||||||
l.Tracef("authenticated user %s with bearer token, scope is %s", uid, ti.GetScope())
|
l.Tracef("authenticated user %s with bearer token, scope is %s", userID, ti.GetScope())
|
||||||
|
|
||||||
// fetch user's and account for this user id
|
// fetch user for this token
|
||||||
user := >smodel.User{}
|
user := >smodel.User{}
|
||||||
if err := m.db.GetByID(c.Request.Context(), uid, user); err != nil || user == nil {
|
if err := m.db.GetByID(ctx, userID, user); err != nil {
|
||||||
l.Warnf("no user found for validated uid %s", uid)
|
if err != db.ErrNoEntries {
|
||||||
|
l.Errorf("database error looking for user with id %s: %s", userID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
l.Warnf("no user found for userID %s", userID)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.Set(oauth.SessionAuthorizedUser, user)
|
c.Set(oauth.SessionAuthorizedUser, user)
|
||||||
l.Tracef("set gin context %s to %+v", oauth.SessionAuthorizedUser, user)
|
|
||||||
|
|
||||||
acct, err := m.db.GetAccountByID(c.Request.Context(), user.AccountID)
|
// fetch account for this token
|
||||||
if err != nil || acct == nil {
|
acct, err := m.db.GetAccountByID(ctx, user.AccountID)
|
||||||
l.Warnf("no account found for validated user %s", uid)
|
if err != nil {
|
||||||
|
if err != db.ErrNoEntries {
|
||||||
|
l.Errorf("database error looking for account with id %s: %s", user.AccountID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
l.Warnf("no account found for userID %s", userID)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.Set(oauth.SessionAuthorizedAccount, acct)
|
c.Set(oauth.SessionAuthorizedAccount, acct)
|
||||||
l.Tracef("set gin context %s to %+v", oauth.SessionAuthorizedAccount, acct)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// check for application token
|
// check for application token
|
||||||
if cid := ti.GetClientID(); cid != "" {
|
if clientID := ti.GetClientID(); clientID != "" {
|
||||||
l.Tracef("authenticated client %s with bearer token, scope is %s", cid, ti.GetScope())
|
l.Tracef("authenticated client %s with bearer token, scope is %s", clientID, ti.GetScope())
|
||||||
|
|
||||||
|
// fetch app for this token
|
||||||
app := >smodel.Application{}
|
app := >smodel.Application{}
|
||||||
if err := m.db.GetWhere(c.Request.Context(), []db.Where{{Key: "client_id", Value: cid}}, app); err != nil {
|
if err := m.db.GetWhere(ctx, []db.Where{{Key: "client_id", Value: clientID}}, app); err != nil {
|
||||||
l.Tracef("no app found for client %s", cid)
|
if err != db.ErrNoEntries {
|
||||||
|
l.Errorf("database error looking for application with clientID %s: %s", clientID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
l.Warnf("no app found for client %s", clientID)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
c.Set(oauth.SessionAuthorizedApplication, app)
|
c.Set(oauth.SessionAuthorizedApplication, app)
|
||||||
l.Tracef("set gin context %s to %+v", oauth.SessionAuthorizedApplication, app)
|
|
||||||
}
|
}
|
||||||
c.Next()
|
|
||||||
}
|
}
|
|
@ -137,7 +137,7 @@
|
||||||
fileServerModule := fileserver.New(c, processor)
|
fileServerModule := fileserver.New(c, processor)
|
||||||
adminModule := admin.New(c, processor)
|
adminModule := admin.New(c, processor)
|
||||||
statusModule := status.New(c, processor)
|
statusModule := status.New(c, processor)
|
||||||
securityModule := security.New(c, dbService)
|
securityModule := security.New(c, dbService, oauthServer)
|
||||||
streamingModule := streaming.New(c, processor)
|
streamingModule := streaming.New(c, processor)
|
||||||
favouritesModule := favourites.New(c, processor)
|
favouritesModule := favourites.New(c, processor)
|
||||||
blocksModule := blocks.New(c, processor)
|
blocksModule := blocks.New(c, processor)
|
||||||
|
|
|
@ -97,7 +97,7 @@
|
||||||
fileServerModule := fileserver.New(c, processor)
|
fileServerModule := fileserver.New(c, processor)
|
||||||
adminModule := admin.New(c, processor)
|
adminModule := admin.New(c, processor)
|
||||||
statusModule := status.New(c, processor)
|
statusModule := status.New(c, processor)
|
||||||
securityModule := security.New(c, dbService)
|
securityModule := security.New(c, dbService, oauthServer)
|
||||||
streamingModule := streaming.New(c, processor)
|
streamingModule := streaming.New(c, processor)
|
||||||
favouritesModule := favourites.New(c, processor)
|
favouritesModule := favourites.New(c, processor)
|
||||||
blocksModule := blocks.New(c, processor)
|
blocksModule := blocks.New(c, processor)
|
||||||
|
|
|
@ -85,6 +85,9 @@ func Authed(c *gin.Context, requireToken bool, requireApp bool, requireUser bool
|
||||||
if a.User.Disabled || !a.User.Approved {
|
if a.User.Disabled || !a.User.Approved {
|
||||||
return nil, errors.New("user disabled or not approved")
|
return nil, errors.New("user disabled or not approved")
|
||||||
}
|
}
|
||||||
|
if a.User.Email == "" {
|
||||||
|
return nil, errors.New("user has no confirmed email address")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if requireAccount {
|
if requireAccount {
|
||||||
|
|
Loading…
Reference in a new issue