Compare commits

..

6 commits

13 changed files with 111 additions and 198 deletions

View file

@ -29,7 +29,7 @@ COOKIE_SECRET=13d6b4dff8f84a10851021ec8608f814570d562c92fe6b5ec4c9f595bcb3234b
###### JWT ###### ###### JWT ######
####################################### #######################################
JWT_SECRET=dkfTgonmgaAdlgkw JWT_SECRET=dkfTgonmgaAdlgkw
JWT_LIFESPAN_MINUTES=4320 # expires after 3 days JWT_LIFESPAN_MINUTES=1440 # expires after 1 day
####################################### #######################################
###### DATABASE ###### ###### DATABASE ######

View file

@ -28,7 +28,7 @@ CDNEnable=false
###### JWT ###### ###### JWT ######
####################################### #######################################
JWT_SECRET=dkfTgonmgaAdlgkw JWT_SECRET=dkfTgonmgaAdlgkw
JWT_LIFESPAN_MINUTES=4320 # expires after 3 days JWT_LIFESPAN_MINUTES=1440 # expires after 1 day
####################################### #######################################
###### DATABASE ###### ###### DATABASE ######

View file

@ -16,5 +16,15 @@ func GetQueueConfig() core.QueueConfig {
// For enabling and disabling the queue system // For enabling and disabling the queue system
// set to true to enable it, set to false to disable // set to true to enable it, set to false to disable
EnableQueue: false, EnableQueue: false,
// Number of concurrent workers processing tasks
Concurrency: 10,
// Queue names with priority weights (higher number = higher priority)
Queues: map[string]int{
"critical": 6,
"default": 3,
"low": 1,
},
} }
} }

View file

@ -21,7 +21,7 @@ func AdminUsersList(c *core.Context) *core.Response {
// initiate authority // initiate authority
auth := new(utils.Authority) auth := new(utils.Authority)
var session = new(utils.SessionUser) session := c.GetSession()
// true if session is active // true if session is active
hassession := session.Init(c) hassession := session.Init(c)
@ -135,7 +135,7 @@ func AdminUsersAdd(c *core.Context) *core.Response {
// initiate authority // initiate authority
auth := new(utils.Authority) auth := new(utils.Authority)
var session = new(utils.SessionUser) session := c.GetSession()
// true if session is active // true if session is active
hassession := session.Init(c) hassession := session.Init(c)

View file

@ -1,5 +1,5 @@
// Copyright 2023 Harran Ali <harran.m@gmail.com>. All rights reserved. // Copyright 2023 Harran Ali <harran.m@gmail.com>. All rights reserved.
// Copyright (c) 2024 Zeni Kim <zenik@smarteching.com> // Copyright (c) 2026 Zeni Kim <zenik@smarteching.com>
// Use of this source code is governed by MIT-style // Use of this source code is governed by MIT-style
// license that can be found in the LICENSE file. // license that can be found in the LICENSE file.
@ -19,7 +19,6 @@ import (
"git.smarteching.com/goffee/core" "git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/events" "git.smarteching.com/goffee/cup/events"
"git.smarteching.com/goffee/cup/models" "git.smarteching.com/goffee/cup/models"
"git.smarteching.com/goffee/cup/utils"
"github.com/google/uuid" "github.com/google/uuid"
"gorm.io/gorm" "gorm.io/gorm"
) )
@ -101,7 +100,7 @@ func Signup(c *core.Context) *core.Response {
// cache the token // cache the token
userAgent := c.GetUserAgent() userAgent := c.GetUserAgent()
hashedCacheKey := utils.CreateAuthTokenHashedCacheKey(user.ID, userAgent) hashedCacheKey := core.CreateAuthTokenHashedCacheKey(user.ID, userAgent)
err = c.GetCache().Set(hashedCacheKey, token) err = c.GetCache().Set(hashedCacheKey, token)
if err != nil { if err != nil {
c.GetLogger().Error(err.Error()) c.GetLogger().Error(err.Error())
@ -223,12 +222,12 @@ func Signin(c *core.Context) *core.Response {
} }
// cache the token // cache the token
userAgent := c.GetUserAgent() userAgent := c.GetUserAgent()
hashedCacheKey := utils.CreateAuthTokenHashedCacheKey(user.ID, userAgent) hashedCacheKey := core.CreateAuthTokenHashedCacheKey(user.ID, userAgent)
err = c.GetCache().Set(hashedCacheKey, token) err = c.GetCache().Set(hashedCacheKey, token)
// delete data from old sessions // delete data from old sessions
sessionKey := fmt.Sprintf("sess_%v", userAgent) sessionKey := fmt.Sprintf("sess_%v", userAgent)
hashedSessionKey := utils.CreateAuthTokenHashedCacheKey(user.ID, sessionKey) hashedSessionKey := core.CreateAuthTokenHashedCacheKey(user.ID, sessionKey)
_ = c.GetCache().Delete(hashedSessionKey) _ = c.GetCache().Delete(hashedSessionKey)
if err != nil { if err != nil {
@ -463,7 +462,7 @@ func Signout(c *core.Context) *core.Response {
})) }))
} }
userAgent := c.GetUserAgent() userAgent := c.GetUserAgent()
hashedCacheKey := utils.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent) hashedCacheKey := core.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent)
err = c.GetCache().Delete(hashedCacheKey) err = c.GetCache().Delete(hashedCacheKey)
if err != nil { if err != nil {

View file

@ -47,8 +47,27 @@ func WelcomeHome(c *core.Context) *core.Response {
} }
// Show dashboard
func WelcomeToDashboard(c *core.Context) *core.Response { func WelcomeToDashboard(c *core.Context) *core.Response {
message := "{\"message\": \"Welcome to Dashboard\"}" message := "{\"message\": \"Welcome to Dashboard\"}"
return c.Response.Json(message) return c.Response.Json(message)
} }
// Show basic app login
func AppLogin(c *core.Context) *core.Response {
// first, include all compoments
// first, include all compoments
type templateData struct {
PageCard components.PageCard
}
// now fill data of the components
tmplData := templateData{
PageCard: components.PageCard{
CardTitle: "Card title",
CardBody: "Loerm ipsum at deim",
},
}
return c.Response.Template("login.html", tmplData)
}

View file

@ -0,0 +1,55 @@
// Copyright (c) 2025 Zeni Kim <zenik@smarteching.com>
// Use of this source code is governed by MIT-style
// license that can be found in the LICENSE file.
package controllers
import (
"encoding/json"
"fmt"
"time"
"git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/workers"
"github.com/hibiken/asynq"
)
// Make samples queues
func Queuesample(c *core.Context) *core.Response {
// Get client queue asynq
client := c.GetQueueClient()
// Create a task with typename and payload.
payload, err := json.Marshal(workers.EmailTaskPayload{UserID: 42})
if err != nil {
c.GetLogger().Error(err.Error())
return c.Response.SetStatusCode(500).Json(`{"message": "internal error"}`)
}
t1 := asynq.NewTask(workers.TypeWelcomeEmail, payload)
t2 := asynq.NewTask(workers.TypeReminderEmail, payload)
// Process the task immediately.
info, err := client.Enqueue(t1)
if err != nil {
c.GetLogger().Error(err.Error())
return c.Response.SetStatusCode(500).Json(`{"message": "internal error"}`)
}
c.GetLogger().Info(fmt.Sprintf(" [*] Successfully enqueued task: %+v", info))
// Process 2 task 1 min later.
for i := 1; i < 3; i++ {
info, err = client.Enqueue(t2, asynq.ProcessIn(1*time.Minute))
if err != nil {
c.GetLogger().Error(err.Error())
return c.Response.SetStatusCode(500).Json(`{"message": "internal error"}`)
}
c.GetLogger().Info(fmt.Sprintf(" [*] Successfully enqueued task: %+v", info))
}
message := "{\"message\": \"Task queued\"}"
return c.Response.Json(message)
}

View file

@ -9,7 +9,6 @@ import (
"git.smarteching.com/goffee/core" "git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/models" "git.smarteching.com/goffee/cup/models"
"git.smarteching.com/goffee/cup/utils"
"gorm.io/gorm" "gorm.io/gorm"
) )
@ -31,7 +30,7 @@ var CheckSessionCookie core.Hook = func(c *core.Context) {
} else { } else {
userAgent := c.GetUserAgent() userAgent := c.GetUserAgent()
hashedCacheKey := utils.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent) hashedCacheKey := core.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent)
cachedToken, err := c.GetCache().Get(hashedCacheKey) cachedToken, err := c.GetCache().Get(hashedCacheKey)
if err != nil { if err != nil {
@ -102,7 +101,7 @@ var AuthCheck core.Hook = func(c *core.Context) {
return return
} }
userAgent := c.GetUserAgent() userAgent := c.GetUserAgent()
hashedCacheKey := utils.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent) hashedCacheKey := core.CreateAuthTokenHashedCacheKey(uint(c.CastToInt(payload["userID"])), userAgent)
cachedToken, err := c.GetCache().Get(hashedCacheKey) cachedToken, err := c.GetCache().Get(hashedCacheKey)
if err != nil { if err != nil {

View file

@ -6,6 +6,7 @@ package main
import ( import (
"git.smarteching.com/goffee/core" "git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/config"
"git.smarteching.com/goffee/cup/workers" "git.smarteching.com/goffee/cup/workers"
) )
@ -26,5 +27,5 @@ func registerQueues() {
//######################################## //########################################
// Start queue server, DO NOT TOUCH // Start queue server, DO NOT TOUCH
//######################################## //########################################
go queque.RunQueueserver() go queque.RunQueueserver(config.GetQueueConfig())
} }

View file

@ -8,6 +8,7 @@ package main
import ( import (
"git.smarteching.com/goffee/core" "git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/controllers" "git.smarteching.com/goffee/cup/controllers"
"git.smarteching.com/goffee/cup/hooks"
) )
// Register the app controllers // Register the app controllers
@ -27,6 +28,9 @@ func registerRoutes() {
controller.Post("/reset-password", controllers.ResetPasswordRequest) controller.Post("/reset-password", controllers.ResetPasswordRequest)
controller.Post("/reset-password/code/:code", controllers.SetNewPassword) controller.Post("/reset-password/code/:code", controllers.SetNewPassword)
// queue sample route
controller.Get("/queuesample", controllers.Queuesample)
// Uncomment the lines below to enable user administration // Uncomment the lines below to enable user administration
controller.Get("/admin/users", controllers.AdminUsersList) controller.Get("/admin/users", controllers.AdminUsersList)
controller.Post("/admin/users", controllers.AdminUsersList) controller.Post("/admin/users", controllers.AdminUsersList)
@ -37,5 +41,8 @@ func registerRoutes() {
controller.Post("/admin/users/delete", controllers.AdminUsersDelete) controller.Post("/admin/users/delete", controllers.AdminUsersDelete)
controller.Post("/admin/users/deleteconfirm", controllers.AdminUsersDelConfirm) controller.Post("/admin/users/deleteconfirm", controllers.AdminUsersDelConfirm)
controller.Get("/dashboard", controllers.WelcomeToDashboard, hooks.AuthCheck)
controller.Get("/signout", controllers.Signout)
controller.Get("/applogin", controllers.AppLogin, hooks.CheckSessionCookie)
controller.Post("/applogin", controllers.AppLogin, hooks.CheckSessionCookie)
} }

View file

@ -6,8 +6,6 @@
package utils package utils
import ( import (
"crypto/md5"
"fmt"
"log" "log"
"time" "time"
@ -71,13 +69,7 @@ func CreateSeedData() {
} }
} }
// generate a hashed string to be used as key for caching auth jwt token
func CreateAuthTokenHashedCacheKey(userID uint, userAgent string) string {
cacheKey := fmt.Sprintf("userid:_%v_useragent:_%v_jwt_token", userID, userAgent)
hashedCacheKey := fmt.Sprintf("%v", fmt.Sprintf("%x", md5.Sum([]byte(cacheKey))))
return hashedCacheKey
}
func FormatUnix(value int64) string { func FormatUnix(value int64) string {
return time.Unix(value, 0).Format("2006-01-02 15:04:05") return time.Unix(value, 0).Format("2006-01-02 15:04:05")

View file

@ -1,170 +0,0 @@
// Copyright (c) 2024 Zeni Kim <zenik@smarteching.com>
// Use of this source code is governed by MIT-style
// license that can be found in the LICENSE file.
package utils
import (
"errors"
"fmt"
"sync"
"time"
"encoding/json"
"git.smarteching.com/goffee/core"
"git.smarteching.com/goffee/cup/models"
"gorm.io/gorm"
)
type SessionUser struct {
mu sync.RWMutex
context *core.Context
userID uint
hashedSessionKey string
authenticated bool
sessionStart time.Time
values map[string]interface{}
}
// start the struct
func (s *SessionUser) Init(c *core.Context) bool {
// check session cookie
pass := true
token := ""
s.context = c
payload := make(map[string]interface{})
// get cookie
usercookie, err := c.GetCookie()
if err != nil {
}
token = usercookie.Token
if token == "" {
pass = false
} else {
payload, err = c.GetJWT().DecodeToken(token)
if err != nil {
pass = false
} else {
userID := uint(c.CastToInt(payload["userID"]))
userAgent := c.GetUserAgent()
// get data from redis
hashedCacheKey := CreateAuthTokenHashedCacheKey(userID, userAgent)
cachedToken, err := c.GetCache().Get(hashedCacheKey)
if err != nil {
pass = false
} else if cachedToken != token {
pass = false
} else {
var user models.User
res := c.GetGorm().Where("id = ?", userID).First(&user)
if res.Error != nil && !errors.Is(res.Error, gorm.ErrRecordNotFound) {
pass = false
}
// if have session start the struct
if pass {
userAgent := c.GetUserAgent()
sessionKey := fmt.Sprintf("sess_%v", userAgent)
s.hashedSessionKey = CreateAuthTokenHashedCacheKey(userID, sessionKey)
s.values = make(map[string]interface{})
s.authenticated = true
s.userID = userID
value, _ := c.GetCache().Get(s.hashedSessionKey)
if len(value) > 0 {
_ = json.Unmarshal([]byte(value), &s.values)
}
return true
} else {
s.hashedSessionKey = ""
s.authenticated = false
s.userID = 0
return false
}
}
}
}
return false
}
func (s *SessionUser) Set(key string, value interface{}) error {
s.mu.Lock()
s.values[key] = value
s.mu.Unlock()
return s.Save()
}
func (s *SessionUser) Get(key string) (interface{}, bool) {
s.mu.RLock()
defer s.mu.RUnlock()
val, ok := s.values[key]
return val, ok
}
func (s *SessionUser) Delete(key string) interface{} {
s.mu.RLock()
v, ok := s.values[key]
s.mu.RUnlock()
if ok {
s.mu.Lock()
delete(s.values, key)
s.mu.Unlock()
}
s.Save()
return v
}
func (s *SessionUser) Flush() error {
s.mu.Lock()
s.context.GetCache().Delete(s.hashedSessionKey)
s.mu.Unlock()
return nil
}
func (s *SessionUser) Save() error {
var value string
s.mu.RLock()
if len(s.values) > 0 {
buf, err := json.Marshal(&s.values)
if err != nil {
s.mu.RUnlock()
return err
}
value = string(buf)
}
if len(value) > 0 {
s.context.GetCache().Set(s.hashedSessionKey, value)
} else {
s.context.GetCache().Delete(s.hashedSessionKey)
}
s.mu.RUnlock()
return nil
}
func (s *SessionUser) GetUserID() uint {
return s.userID
}

View file

@ -3,8 +3,9 @@ package workers
import ( import (
"context" "context"
"encoding/json" "encoding/json"
"log" "fmt"
"git.smarteching.com/goffee/core/logger"
"github.com/hibiken/asynq" "github.com/hibiken/asynq"
) )
@ -25,7 +26,7 @@ func HandleWelcomeEmailTask(ctx context.Context, t *asynq.Task) error {
if err := json.Unmarshal(t.Payload(), &p); err != nil { if err := json.Unmarshal(t.Payload(), &p); err != nil {
return err return err
} }
log.Printf(" [*] Send Welcome Email to User %d", p.UserID) logger.ResolveLogger().Info(fmt.Sprintf(" [*] Send Welcome Email to User %d", p.UserID))
return nil return nil
} }
@ -34,6 +35,6 @@ func HandleReminderEmailTask(ctx context.Context, t *asynq.Task) error {
if err := json.Unmarshal(t.Payload(), &p); err != nil { if err := json.Unmarshal(t.Payload(), &p); err != nil {
return err return err
} }
log.Printf(" [*] Send Reminder Email to User %d", p.UserID) logger.ResolveLogger().Info(fmt.Sprintf(" [*] Send Reminder Email to User %d", p.UserID))
return nil return nil
} }