mirror of
https://github.com/pocket-id/pocket-id.git
synced 2025-12-12 09:13:14 +03:00
Compare commits
7 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9e94a436cc | ||
|
|
f82020ccfb | ||
|
|
a4a90a16a9 | ||
|
|
365734ec5d | ||
|
|
d02d8931a0 | ||
|
|
24c948e6a6 | ||
|
|
7a54d3ae20 |
14
CHANGELOG.md
14
CHANGELOG.md
@@ -1,3 +1,17 @@
|
|||||||
|
## [](https://github.com/stonith404/pocket-id/compare/v0.5.3...v) (2024-10-02)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add copy to clipboard option for OIDC client information ([f82020c](https://github.com/stonith404/pocket-id/commit/f82020ccfb0d4fbaa1dd98182188149d8085252a))
|
||||||
|
* add gravatar profile picture integration ([365734e](https://github.com/stonith404/pocket-id/commit/365734ec5d8966c2ab877c60cfb176b9cdc36880))
|
||||||
|
* add user groups ([24c948e](https://github.com/stonith404/pocket-id/commit/24c948e6a66f283866f6c8369c16fa6cbcfa626c))
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* only return user groups if it is explicitly requested ([a4a90a1](https://github.com/stonith404/pocket-id/commit/a4a90a16a9726569a22e42560184319b25fd7ca6))
|
||||||
|
|
||||||
## [](https://github.com/stonith404/pocket-id/compare/v0.5.2...v) (2024-09-26)
|
## [](https://github.com/stonith404/pocket-id/compare/v0.5.2...v) (2024-09-26)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -95,6 +95,7 @@ You may need the following information:
|
|||||||
- **Certificate URL**: `https://<your-domain>/.well-known/jwks.json`
|
- **Certificate URL**: `https://<your-domain>/.well-known/jwks.json`
|
||||||
- **OIDC Discovery URL**: `https://<your-domain>/.well-known/openid-configuration`
|
- **OIDC Discovery URL**: `https://<your-domain>/.well-known/openid-configuration`
|
||||||
- **PKCE**: `false` as this is not supported yet.
|
- **PKCE**: `false` as this is not supported yet.
|
||||||
|
- **Scopes**: At least `openid email`. Optionally you can add `profile` and `groups`.
|
||||||
|
|
||||||
### Proxy Services with Pocket ID
|
### Proxy Services with Pocket ID
|
||||||
|
|
||||||
|
|||||||
@@ -41,6 +41,7 @@ func initRouter(db *gorm.DB, appConfigService *service.AppConfigService) {
|
|||||||
userService := service.NewUserService(db, jwtService)
|
userService := service.NewUserService(db, jwtService)
|
||||||
oidcService := service.NewOidcService(db, jwtService, appConfigService, auditLogService)
|
oidcService := service.NewOidcService(db, jwtService, appConfigService, auditLogService)
|
||||||
testService := service.NewTestService(db, appConfigService)
|
testService := service.NewTestService(db, appConfigService)
|
||||||
|
userGroupService := service.NewUserGroupService(db)
|
||||||
|
|
||||||
r.Use(middleware.NewCorsMiddleware().Add())
|
r.Use(middleware.NewCorsMiddleware().Add())
|
||||||
r.Use(middleware.NewRateLimitMiddleware().Add(rate.Every(time.Second), 60))
|
r.Use(middleware.NewRateLimitMiddleware().Add(rate.Every(time.Second), 60))
|
||||||
@@ -57,6 +58,7 @@ func initRouter(db *gorm.DB, appConfigService *service.AppConfigService) {
|
|||||||
controller.NewUserController(apiGroup, jwtAuthMiddleware, middleware.NewRateLimitMiddleware(), userService)
|
controller.NewUserController(apiGroup, jwtAuthMiddleware, middleware.NewRateLimitMiddleware(), userService)
|
||||||
controller.NewAppConfigController(apiGroup, jwtAuthMiddleware, appConfigService)
|
controller.NewAppConfigController(apiGroup, jwtAuthMiddleware, appConfigService)
|
||||||
controller.NewAuditLogController(apiGroup, auditLogService, jwtAuthMiddleware)
|
controller.NewAuditLogController(apiGroup, auditLogService, jwtAuthMiddleware)
|
||||||
|
controller.NewUserGroupController(apiGroup, jwtAuthMiddleware, userGroupService)
|
||||||
|
|
||||||
// Add test controller in non-production environments
|
// Add test controller in non-production environments
|
||||||
if common.EnvConfig.AppEnv != "production" {
|
if common.EnvConfig.AppEnv != "production" {
|
||||||
|
|||||||
@@ -15,4 +15,5 @@ var (
|
|||||||
ErrOidcInvalidCallbackURL = errors.New("invalid callback URL")
|
ErrOidcInvalidCallbackURL = errors.New("invalid callback URL")
|
||||||
ErrFileTypeNotSupported = errors.New("file type not supported")
|
ErrFileTypeNotSupported = errors.New("file type not supported")
|
||||||
ErrInvalidCredentials = errors.New("no user found with provided credentials")
|
ErrInvalidCredentials = errors.New("no user found with provided credentials")
|
||||||
|
ErrNameAlreadyInUse = errors.New("name is already in use")
|
||||||
)
|
)
|
||||||
|
|||||||
162
backend/internal/controller/user_group_controller.go
Normal file
162
backend/internal/controller/user_group_controller.go
Normal file
@@ -0,0 +1,162 @@
|
|||||||
|
package controller
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/common"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/dto"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/middleware"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/service"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NewUserGroupController(group *gin.RouterGroup, jwtAuthMiddleware *middleware.JwtAuthMiddleware, userGroupService *service.UserGroupService) {
|
||||||
|
ugc := UserGroupController{
|
||||||
|
UserGroupService: userGroupService,
|
||||||
|
}
|
||||||
|
|
||||||
|
group.GET("/user-groups", jwtAuthMiddleware.Add(true), ugc.list)
|
||||||
|
group.GET("/user-groups/:id", jwtAuthMiddleware.Add(true), ugc.get)
|
||||||
|
group.POST("/user-groups", jwtAuthMiddleware.Add(true), ugc.create)
|
||||||
|
group.PUT("/user-groups/:id", jwtAuthMiddleware.Add(true), ugc.update)
|
||||||
|
group.DELETE("/user-groups/:id", jwtAuthMiddleware.Add(true), ugc.delete)
|
||||||
|
group.PUT("/user-groups/:id/users", jwtAuthMiddleware.Add(true), ugc.updateUsers)
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserGroupController struct {
|
||||||
|
UserGroupService *service.UserGroupService
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) list(c *gin.Context) {
|
||||||
|
page, _ := strconv.Atoi(c.DefaultQuery("page", "1"))
|
||||||
|
pageSize, _ := strconv.Atoi(c.DefaultQuery("limit", "10"))
|
||||||
|
searchTerm := c.Query("search")
|
||||||
|
|
||||||
|
groups, pagination, err := ugc.UserGroupService.List(searchTerm, page, pageSize)
|
||||||
|
if err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groupsDto = make([]dto.UserGroupDtoWithUserCount, len(groups))
|
||||||
|
for i, group := range groups {
|
||||||
|
var groupDto dto.UserGroupDtoWithUserCount
|
||||||
|
if err := dto.MapStruct(group, &groupDto); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
groupDto.UserCount, err = ugc.UserGroupService.GetUserCountOfGroup(group.ID)
|
||||||
|
if err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
groupsDto[i] = groupDto
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, gin.H{
|
||||||
|
"data": groupsDto,
|
||||||
|
"pagination": pagination,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) get(c *gin.Context) {
|
||||||
|
group, err := ugc.UserGroupService.Get(c.Param("id"))
|
||||||
|
if err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groupDto dto.UserGroupDtoWithUsers
|
||||||
|
if err := dto.MapStruct(group, &groupDto); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, groupDto)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) create(c *gin.Context) {
|
||||||
|
var input dto.UserGroupCreateDto
|
||||||
|
if err := c.ShouldBindJSON(&input); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group, err := ugc.UserGroupService.Create(input)
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, common.ErrNameAlreadyInUse) {
|
||||||
|
utils.CustomControllerError(c, http.StatusConflict, err.Error())
|
||||||
|
} else {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groupDto dto.UserGroupDtoWithUsers
|
||||||
|
if err := dto.MapStruct(group, &groupDto); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusCreated, groupDto)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) update(c *gin.Context) {
|
||||||
|
var input dto.UserGroupCreateDto
|
||||||
|
if err := c.ShouldBindJSON(&input); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group, err := ugc.UserGroupService.Update(c.Param("id"), input)
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, common.ErrNameAlreadyInUse) {
|
||||||
|
utils.CustomControllerError(c, http.StatusConflict, err.Error())
|
||||||
|
} else {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groupDto dto.UserGroupDtoWithUsers
|
||||||
|
if err := dto.MapStruct(group, &groupDto); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, groupDto)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) delete(c *gin.Context) {
|
||||||
|
if err := ugc.UserGroupService.Delete(c.Param("id")); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.Status(http.StatusNoContent)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ugc *UserGroupController) updateUsers(c *gin.Context) {
|
||||||
|
var input dto.UserGroupUpdateUsersDto
|
||||||
|
if err := c.ShouldBindJSON(&input); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
group, err := ugc.UserGroupService.UpdateUsers(c.Param("id"), input)
|
||||||
|
if err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var groupDto dto.UserGroupDtoWithUsers
|
||||||
|
if err := dto.MapStruct(group, &groupDto); err != nil {
|
||||||
|
utils.ControllerError(c, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, groupDto)
|
||||||
|
}
|
||||||
@@ -57,15 +57,37 @@ func mapStructInternal(sourceVal reflect.Value, destVal reflect.Value) error {
|
|||||||
// Handle direct assignment for simple types
|
// Handle direct assignment for simple types
|
||||||
if sourceField.Type() == destField.Type() {
|
if sourceField.Type() == destField.Type() {
|
||||||
destField.Set(sourceField)
|
destField.Set(sourceField)
|
||||||
|
|
||||||
} else if sourceField.Kind() == reflect.Slice && destField.Kind() == reflect.Slice {
|
} else if sourceField.Kind() == reflect.Slice && destField.Kind() == reflect.Slice {
|
||||||
// Handle slices
|
// Handle slices
|
||||||
if sourceField.Type().Elem() == destField.Type().Elem() {
|
if sourceField.Type().Elem() == destField.Type().Elem() {
|
||||||
|
// Direct assignment for slices of primitive types or non-struct elements
|
||||||
newSlice := reflect.MakeSlice(destField.Type(), sourceField.Len(), sourceField.Cap())
|
newSlice := reflect.MakeSlice(destField.Type(), sourceField.Len(), sourceField.Cap())
|
||||||
|
|
||||||
for j := 0; j < sourceField.Len(); j++ {
|
for j := 0; j < sourceField.Len(); j++ {
|
||||||
newSlice.Index(j).Set(sourceField.Index(j))
|
newSlice.Index(j).Set(sourceField.Index(j))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
destField.Set(newSlice)
|
||||||
|
|
||||||
|
} else if sourceField.Type().Elem().Kind() == reflect.Struct && destField.Type().Elem().Kind() == reflect.Struct {
|
||||||
|
// Recursively map slices of structs
|
||||||
|
newSlice := reflect.MakeSlice(destField.Type(), sourceField.Len(), sourceField.Cap())
|
||||||
|
|
||||||
|
for j := 0; j < sourceField.Len(); j++ {
|
||||||
|
// Get the element from both source and destination slice
|
||||||
|
sourceElem := sourceField.Index(j)
|
||||||
|
destElem := reflect.New(destField.Type().Elem()).Elem()
|
||||||
|
|
||||||
|
// Recursively map the struct elements
|
||||||
|
if err := mapStructInternal(sourceElem, destElem); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the mapped element in the new slice
|
||||||
|
newSlice.Index(j).Set(destElem)
|
||||||
|
}
|
||||||
|
|
||||||
destField.Set(newSlice)
|
destField.Set(newSlice)
|
||||||
}
|
}
|
||||||
} else if sourceField.Kind() == reflect.Struct && destField.Kind() == reflect.Struct {
|
} else if sourceField.Kind() == reflect.Struct && destField.Kind() == reflect.Struct {
|
||||||
|
|||||||
32
backend/internal/dto/user_group_dto.go
Normal file
32
backend/internal/dto/user_group_dto.go
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
package dto
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
type UserGroupDtoWithUsers struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
FriendlyName string `json:"friendlyName"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Users []UserDto `json:"users"`
|
||||||
|
CreatedAt time.Time `json:"createdAt"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserGroupDtoWithUserCount struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
FriendlyName string `json:"friendlyName"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
UserCount int64 `json:"userCount"`
|
||||||
|
CreatedAt time.Time `json:"createdAt"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserGroupCreateDto struct {
|
||||||
|
FriendlyName string `json:"friendlyName" binding:"required,min=3,max=30"`
|
||||||
|
Name string `json:"name" binding:"required,min=3,max=30,userGroupName"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserGroupUpdateUsersDto struct {
|
||||||
|
UserIDs []string `json:"userIds" binding:"required"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AssignUserToGroupDto struct {
|
||||||
|
UserID string `json:"userId" binding:"required"`
|
||||||
|
}
|
||||||
@@ -28,6 +28,13 @@ var validateUsername validator.Func = func(fl validator.FieldLevel) bool {
|
|||||||
return matched
|
return matched
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var validateUserGroupName validator.Func = func(fl validator.FieldLevel) bool {
|
||||||
|
// [a-z0-9_] : The group name can only contain lowercase letters, numbers, and underscores
|
||||||
|
regex := "^[a-z0-9_]+$"
|
||||||
|
matched, _ := regexp.MatchString(regex, fl.Field().String())
|
||||||
|
return matched
|
||||||
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
|
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
|
||||||
if err := v.RegisterValidation("urlList", validateUrlList); err != nil {
|
if err := v.RegisterValidation("urlList", validateUrlList); err != nil {
|
||||||
@@ -39,4 +46,10 @@ func init() {
|
|||||||
log.Fatalf("Failed to register custom validation: %v", err)
|
log.Fatalf("Failed to register custom validation: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if v, ok := binding.Validator.Engine().(*validator.Validate); ok {
|
||||||
|
if err := v.RegisterValidation("userGroupName", validateUserGroupName); err != nil {
|
||||||
|
log.Fatalf("Failed to register custom validation: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ type User struct {
|
|||||||
LastName string
|
LastName string
|
||||||
IsAdmin bool
|
IsAdmin bool
|
||||||
|
|
||||||
|
UserGroups []UserGroup `gorm:"many2many:user_groups_users;"`
|
||||||
Credentials []WebauthnCredential
|
Credentials []WebauthnCredential
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
8
backend/internal/model/user_group.go
Normal file
8
backend/internal/model/user_group.go
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
package model
|
||||||
|
|
||||||
|
type UserGroup struct {
|
||||||
|
Base
|
||||||
|
FriendlyName string
|
||||||
|
Name string `gorm:"unique"`
|
||||||
|
Users []User `gorm:"many2many:user_groups_users;"`
|
||||||
|
}
|
||||||
@@ -301,7 +301,7 @@ func (s *OidcService) DeleteClientLogo(clientID string) error {
|
|||||||
|
|
||||||
func (s *OidcService) GetUserClaimsForClient(userID string, clientID string) (map[string]interface{}, error) {
|
func (s *OidcService) GetUserClaimsForClient(userID string, clientID string) (map[string]interface{}, error) {
|
||||||
var authorizedOidcClient model.UserAuthorizedOidcClient
|
var authorizedOidcClient model.UserAuthorizedOidcClient
|
||||||
if err := s.db.Preload("User").First(&authorizedOidcClient, "user_id = ? AND client_id = ?", userID, clientID).Error; err != nil {
|
if err := s.db.Preload("User.UserGroups").First(&authorizedOidcClient, "user_id = ? AND client_id = ?", userID, clientID).Error; err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -316,6 +316,14 @@ func (s *OidcService) GetUserClaimsForClient(userID string, clientID string) (ma
|
|||||||
claims["email"] = user.Email
|
claims["email"] = user.Email
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if strings.Contains(scope, "groups") {
|
||||||
|
userGroups := make([]string, len(user.UserGroups))
|
||||||
|
for i, group := range user.UserGroups {
|
||||||
|
userGroups[i] = group.Name
|
||||||
|
}
|
||||||
|
claims["groups"] = userGroups
|
||||||
|
}
|
||||||
|
|
||||||
profileClaims := map[string]interface{}{
|
profileClaims := map[string]interface{}{
|
||||||
"given_name": user.FirstName,
|
"given_name": user.FirstName,
|
||||||
"family_name": user.LastName,
|
"family_name": user.LastName,
|
||||||
|
|||||||
@@ -56,6 +56,30 @@ func (s *TestService) SeedDatabase() error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
userGroups := []model.UserGroup{
|
||||||
|
{
|
||||||
|
Base: model.Base{
|
||||||
|
ID: "4110f814-56f1-4b28-8998-752b69bc97c0e",
|
||||||
|
},
|
||||||
|
Name: "developers",
|
||||||
|
FriendlyName: "Developers",
|
||||||
|
Users: []model.User{users[0], users[1]},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Base: model.Base{
|
||||||
|
ID: "adab18bf-f89d-4087-9ee1-70ff15b48211",
|
||||||
|
},
|
||||||
|
Name: "designers",
|
||||||
|
FriendlyName: "Designers",
|
||||||
|
Users: []model.User{users[0]},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, group := range userGroups {
|
||||||
|
if err := tx.Create(&group).Error; err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
oidcClients := []model.OidcClient{
|
oidcClients := []model.OidcClient{
|
||||||
{
|
{
|
||||||
Base: model.Base{
|
Base: model.Base{
|
||||||
|
|||||||
111
backend/internal/service/user_group_service.go
Normal file
111
backend/internal/service/user_group_service.go
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/common"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/dto"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/model"
|
||||||
|
"github.com/stonith404/pocket-id/backend/internal/utils"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserGroupService struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUserGroupService(db *gorm.DB) *UserGroupService {
|
||||||
|
return &UserGroupService{db: db}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) List(name string, page int, pageSize int) (groups []model.UserGroup, response utils.PaginationResponse, err error) {
|
||||||
|
query := s.db.Model(&model.UserGroup{})
|
||||||
|
|
||||||
|
if name != "" {
|
||||||
|
query = query.Where("name LIKE ?", "%"+name+"%")
|
||||||
|
}
|
||||||
|
|
||||||
|
response, err = utils.Paginate(page, pageSize, query, &groups)
|
||||||
|
return groups, response, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) Get(id string) (group model.UserGroup, err error) {
|
||||||
|
err = s.db.Where("id = ?", id).Preload("Users").First(&group).Error
|
||||||
|
return group, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) Delete(id string) error {
|
||||||
|
var group model.UserGroup
|
||||||
|
if err := s.db.Where("id = ?", id).First(&group).Error; err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return s.db.Delete(&group).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) Create(input dto.UserGroupCreateDto) (group model.UserGroup, err error) {
|
||||||
|
group = model.UserGroup{
|
||||||
|
FriendlyName: input.FriendlyName,
|
||||||
|
Name: input.Name,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := s.db.Preload("Users").Create(&group).Error; err != nil {
|
||||||
|
if errors.Is(err, gorm.ErrDuplicatedKey) {
|
||||||
|
return model.UserGroup{}, common.ErrNameAlreadyInUse
|
||||||
|
}
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
return group, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) Update(id string, input dto.UserGroupCreateDto) (group model.UserGroup, err error) {
|
||||||
|
group, err = s.Get(id)
|
||||||
|
if err != nil {
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
group.Name = input.Name
|
||||||
|
group.FriendlyName = input.FriendlyName
|
||||||
|
|
||||||
|
if err := s.db.Preload("Users").Save(&group).Error; err != nil {
|
||||||
|
if errors.Is(err, gorm.ErrDuplicatedKey) {
|
||||||
|
return model.UserGroup{}, common.ErrNameAlreadyInUse
|
||||||
|
}
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
return group, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) UpdateUsers(id string, input dto.UserGroupUpdateUsersDto) (group model.UserGroup, err error) {
|
||||||
|
group, err = s.Get(id)
|
||||||
|
if err != nil {
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch the users based on UserIDs in input
|
||||||
|
var users []model.User
|
||||||
|
if len(input.UserIDs) > 0 {
|
||||||
|
if err := s.db.Where("id IN (?)", input.UserIDs).Find(&users).Error; err != nil {
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Replace the current users with the new set of users
|
||||||
|
if err := s.db.Model(&group).Association("Users").Replace(users); err != nil {
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save the updated group
|
||||||
|
if err := s.db.Save(&group).Error; err != nil {
|
||||||
|
return model.UserGroup{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return group, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserGroupService) GetUserCountOfGroup(id string) (int64, error) {
|
||||||
|
var group model.UserGroup
|
||||||
|
if err := s.db.Preload("Users").Where("id = ?", id).First(&group).Error; err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return s.db.Model(&group).Association("Users").Count(), nil
|
||||||
|
}
|
||||||
@@ -5,9 +5,10 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type PaginationResponse struct {
|
type PaginationResponse struct {
|
||||||
TotalPages int64 `json:"totalPages"`
|
TotalPages int64 `json:"totalPages"`
|
||||||
TotalItems int64 `json:"totalItems"`
|
TotalItems int64 `json:"totalItems"`
|
||||||
CurrentPage int `json:"currentPage"`
|
CurrentPage int `json:"currentPage"`
|
||||||
|
ItemsPerPage int `json:"itemsPerPage"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func Paginate(page int, pageSize int, db *gorm.DB, result interface{}) (PaginationResponse, error) {
|
func Paginate(page int, pageSize int, db *gorm.DB, result interface{}) (PaginationResponse, error) {
|
||||||
@@ -33,8 +34,9 @@ func Paginate(page int, pageSize int, db *gorm.DB, result interface{}) (Paginati
|
|||||||
}
|
}
|
||||||
|
|
||||||
return PaginationResponse{
|
return PaginationResponse{
|
||||||
TotalPages: (totalItems + int64(pageSize) - 1) / int64(pageSize),
|
TotalPages: (totalItems + int64(pageSize) - 1) / int64(pageSize),
|
||||||
TotalItems: totalItems,
|
TotalItems: totalItems,
|
||||||
CurrentPage: page,
|
CurrentPage: page,
|
||||||
|
ItemsPerPage: pageSize,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|||||||
2
backend/migrations/20240924202721_user_groups.down.sql
Normal file
2
backend/migrations/20240924202721_user_groups.down.sql
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
DROP TABLE user_groups;
|
||||||
|
DROP TABLE user_groups_users;
|
||||||
16
backend/migrations/20240924202721_user_groups.up.sql
Normal file
16
backend/migrations/20240924202721_user_groups.up.sql
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
CREATE TABLE user_groups
|
||||||
|
(
|
||||||
|
id TEXT NOT NULL PRIMARY KEY,
|
||||||
|
created_at DATETIME,
|
||||||
|
friendly_name TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL UNIQUE
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE user_groups_users
|
||||||
|
(
|
||||||
|
user_id TEXT NOT NULL,
|
||||||
|
user_group_id TEXT NOT NULL,
|
||||||
|
PRIMARY KEY (user_id, user_group_id),
|
||||||
|
FOREIGN KEY (user_id) REFERENCES users (id) ON DELETE CASCADE,
|
||||||
|
FOREIGN KEY (user_group_id) REFERENCES user_groups (id) ON DELETE CASCADE
|
||||||
|
);
|
||||||
8
frontend/package-lock.json
generated
8
frontend/package-lock.json
generated
@@ -10,7 +10,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@simplewebauthn/browser": "^10.0.0",
|
"@simplewebauthn/browser": "^10.0.0",
|
||||||
"axios": "^1.7.5",
|
"axios": "^1.7.5",
|
||||||
"bits-ui": "^0.21.13",
|
"bits-ui": "^0.21.15",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"crypto": "^1.0.1",
|
"crypto": "^1.0.1",
|
||||||
"formsnap": "^1.0.1",
|
"formsnap": "^1.0.1",
|
||||||
@@ -1806,9 +1806,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/bits-ui": {
|
"node_modules/bits-ui": {
|
||||||
"version": "0.21.13",
|
"version": "0.21.15",
|
||||||
"resolved": "https://registry.npmjs.org/bits-ui/-/bits-ui-0.21.13.tgz",
|
"resolved": "https://registry.npmjs.org/bits-ui/-/bits-ui-0.21.15.tgz",
|
||||||
"integrity": "sha512-7nmOh6Ig7ND4DXZHv1FhNsY9yUGrad0+mf3tc4YN//3MgnJT1LnHtk4HZAKgmxCOe7txSX7/39LtYHbkrXokAQ==",
|
"integrity": "sha512-+m5WSpJnFdCcNdXSTIVC1WYBozipO03qRh03GFWgrdxoHiolCfwW71EYG4LPCWYPG6KcTZV0Cj6iHSiZ7cdKdg==",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@internationalized/date": "^3.5.1",
|
"@internationalized/date": "^3.5.1",
|
||||||
"@melt-ui/svelte": "0.76.2",
|
"@melt-ui/svelte": "0.76.2",
|
||||||
|
|||||||
@@ -42,7 +42,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@simplewebauthn/browser": "^10.0.0",
|
"@simplewebauthn/browser": "^10.0.0",
|
||||||
"axios": "^1.7.5",
|
"axios": "^1.7.5",
|
||||||
"bits-ui": "^0.21.13",
|
"bits-ui": "^0.21.15",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"crypto": "^1.0.1",
|
"crypto": "^1.0.1",
|
||||||
"formsnap": "^1.0.1",
|
"formsnap": "^1.0.1",
|
||||||
|
|||||||
154
frontend/src/lib/components/advanced-table.svelte
Normal file
154
frontend/src/lib/components/advanced-table.svelte
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
<script lang="ts" generics="T extends {id:string}">
|
||||||
|
import Checkbox from '$lib/components/ui/checkbox/checkbox.svelte';
|
||||||
|
import { Input } from '$lib/components/ui/input/index.js';
|
||||||
|
import * as Pagination from '$lib/components/ui/pagination';
|
||||||
|
import * as Select from '$lib/components/ui/select';
|
||||||
|
import * as Table from '$lib/components/ui/table/index.js';
|
||||||
|
import type { Paginated } from '$lib/types/pagination.type';
|
||||||
|
import { debounced } from '$lib/utils/debounce-util';
|
||||||
|
import type { Snippet } from 'svelte';
|
||||||
|
|
||||||
|
let {
|
||||||
|
items,
|
||||||
|
selectedIds = $bindable(),
|
||||||
|
fetchItems,
|
||||||
|
columns,
|
||||||
|
rows
|
||||||
|
}: {
|
||||||
|
items: Paginated<T>;
|
||||||
|
selectedIds?: string[];
|
||||||
|
fetchItems: (search: string, page: number, limit: number) => Promise<Paginated<T>>;
|
||||||
|
columns: (string | { label: string; hidden?: boolean })[];
|
||||||
|
rows: Snippet<[{ item: T }]>;
|
||||||
|
} = $props();
|
||||||
|
|
||||||
|
let availablePageSizes: number[] = [10, 20, 50, 100];
|
||||||
|
|
||||||
|
let allChecked = $derived.by(() => {
|
||||||
|
if (!selectedIds || items.data.length === 0) return false;
|
||||||
|
for (const item of items.data) {
|
||||||
|
if (!selectedIds.includes(item.id)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
const onSearch = debounced(async (searchValue: string) => {
|
||||||
|
items = await fetchItems(searchValue, 1, items.pagination.itemsPerPage);
|
||||||
|
}, 300);
|
||||||
|
|
||||||
|
async function onAllCheck(checked: boolean) {
|
||||||
|
if (checked) {
|
||||||
|
selectedIds = items.data.map((item) => item.id);
|
||||||
|
} else {
|
||||||
|
selectedIds = [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onCheck(checked: boolean, id: string) {
|
||||||
|
if (!selectedIds) return;
|
||||||
|
if (checked) {
|
||||||
|
selectedIds = [...selectedIds, id];
|
||||||
|
} else {
|
||||||
|
selectedIds = selectedIds.filter((selectedId) => selectedId !== id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onPageChange(page: number) {
|
||||||
|
items = await fetchItems('', page, items.pagination.itemsPerPage);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onPageSizeChange(size: number) {
|
||||||
|
items = await fetchItems('', 1, size);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="w-full">
|
||||||
|
<Input
|
||||||
|
class="mb-4 max-w-sm"
|
||||||
|
placeholder={'Search...'}
|
||||||
|
type="text"
|
||||||
|
oninput={(e) => onSearch((e.target as HTMLInputElement).value)}
|
||||||
|
/>
|
||||||
|
<Table.Root>
|
||||||
|
<Table.Header>
|
||||||
|
<Table.Row>
|
||||||
|
{#if selectedIds}
|
||||||
|
<Table.Head>
|
||||||
|
<Checkbox checked={allChecked} onCheckedChange={(c) => onAllCheck(c as boolean)} />
|
||||||
|
</Table.Head>
|
||||||
|
{/if}
|
||||||
|
{#each columns as column}
|
||||||
|
{#if typeof column === 'string'}
|
||||||
|
<Table.Head>{column}</Table.Head>
|
||||||
|
{:else}
|
||||||
|
<Table.Head class={column.hidden ? 'sr-only' : ''}>{column.label}</Table.Head>
|
||||||
|
{/if}
|
||||||
|
{/each}
|
||||||
|
</Table.Row>
|
||||||
|
</Table.Header>
|
||||||
|
<Table.Body>
|
||||||
|
{#each items.data as item}
|
||||||
|
<Table.Row class={selectedIds?.includes(item.id) ? 'bg-muted/20' : ''}>
|
||||||
|
{#if selectedIds}
|
||||||
|
<Table.Cell>
|
||||||
|
<Checkbox
|
||||||
|
checked={selectedIds.includes(item.id)}
|
||||||
|
onCheckedChange={(c) => onCheck(c as boolean, item.id)}
|
||||||
|
/>
|
||||||
|
</Table.Cell>
|
||||||
|
{/if}
|
||||||
|
{@render rows({ item })}
|
||||||
|
</Table.Row>
|
||||||
|
{/each}
|
||||||
|
</Table.Body>
|
||||||
|
</Table.Root>
|
||||||
|
<div class="mt-5 flex items-center justify-between space-x-2">
|
||||||
|
<div class="flex items-center space-x-2">
|
||||||
|
<p class="text-sm font-medium">Items per page</p>
|
||||||
|
<Select.Root
|
||||||
|
selected={{
|
||||||
|
label: items.pagination.itemsPerPage.toString(),
|
||||||
|
value: items.pagination.itemsPerPage
|
||||||
|
}}
|
||||||
|
onSelectedChange={(v) => onPageSizeChange(v?.value as number)}
|
||||||
|
>
|
||||||
|
<Select.Trigger class="h-9 w-[80px]">
|
||||||
|
<Select.Value>{items.pagination.itemsPerPage}</Select.Value>
|
||||||
|
</Select.Trigger>
|
||||||
|
<Select.Content>
|
||||||
|
{#each availablePageSizes as size}
|
||||||
|
<Select.Item value={size}>{size}</Select.Item>
|
||||||
|
{/each}
|
||||||
|
</Select.Content>
|
||||||
|
</Select.Root>
|
||||||
|
</div>
|
||||||
|
<Pagination.Root
|
||||||
|
class="mx-0 w-auto"
|
||||||
|
count={items.pagination.totalItems}
|
||||||
|
perPage={items.pagination.itemsPerPage}
|
||||||
|
{onPageChange}
|
||||||
|
page={items.pagination.currentPage}
|
||||||
|
let:pages
|
||||||
|
>
|
||||||
|
<Pagination.Content class="flex justify-end">
|
||||||
|
<Pagination.Item>
|
||||||
|
<Pagination.PrevButton />
|
||||||
|
</Pagination.Item>
|
||||||
|
{#each pages as page (page.key)}
|
||||||
|
{#if page.type !== 'ellipsis'}
|
||||||
|
<Pagination.Item>
|
||||||
|
<Pagination.Link {page} isActive={items.pagination.currentPage === page.value}>
|
||||||
|
{page.value}
|
||||||
|
</Pagination.Link>
|
||||||
|
</Pagination.Item>
|
||||||
|
{/if}
|
||||||
|
{/each}
|
||||||
|
<Pagination.Item>
|
||||||
|
<Pagination.NextButton />
|
||||||
|
</Pagination.Item>
|
||||||
|
</Pagination.Content>
|
||||||
|
</Pagination.Root>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
41
frontend/src/lib/components/copy-to-clipboard.svelte
Normal file
41
frontend/src/lib/components/copy-to-clipboard.svelte
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import * as Tooltip from '$lib/components/ui/tooltip';
|
||||||
|
import { LucideCheck } from 'lucide-svelte';
|
||||||
|
import type { Snippet } from 'svelte';
|
||||||
|
|
||||||
|
let { value, children }: { value: string; children: Snippet } = $props();
|
||||||
|
|
||||||
|
let open = $state(false);
|
||||||
|
let copied = $state(false);
|
||||||
|
|
||||||
|
function onClick() {
|
||||||
|
open = true;
|
||||||
|
copyToClipboard();
|
||||||
|
}
|
||||||
|
|
||||||
|
function onOpenChange(state: boolean) {
|
||||||
|
open = state;
|
||||||
|
if (!state) {
|
||||||
|
copied = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyToClipboard() {
|
||||||
|
navigator.clipboard.writeText(value);
|
||||||
|
copied = true;
|
||||||
|
setTimeout(() => onOpenChange(false), 1000);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<button onclick={onClick}>
|
||||||
|
<Tooltip.Root closeOnPointerDown={false} {onOpenChange} {open}>
|
||||||
|
<Tooltip.Trigger>{@render children()}</Tooltip.Trigger>
|
||||||
|
<Tooltip.Content onclick={copyToClipboard}>
|
||||||
|
{#if copied}
|
||||||
|
<span class="flex items-center"><LucideCheck class="mr-1 h-4 w-4" /> Copied</span>
|
||||||
|
{:else}
|
||||||
|
<span>Click to copy</span>
|
||||||
|
{/if}
|
||||||
|
</Tooltip.Content>
|
||||||
|
</Tooltip.Root>
|
||||||
|
</button>
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
import type { FormInput } from '$lib/utils/form-util';
|
import type { FormInput } from '$lib/utils/form-util';
|
||||||
import type { Snippet } from 'svelte';
|
import type { Snippet } from 'svelte';
|
||||||
import type { HTMLAttributes } from 'svelte/elements';
|
import type { HTMLAttributes } from 'svelte/elements';
|
||||||
import { Input } from './ui/input';
|
import { Input, type FormInputEvent } from './ui/input';
|
||||||
|
|
||||||
let {
|
let {
|
||||||
input = $bindable(),
|
input = $bindable(),
|
||||||
@@ -12,6 +12,7 @@
|
|||||||
disabled = false,
|
disabled = false,
|
||||||
type = 'text',
|
type = 'text',
|
||||||
children,
|
children,
|
||||||
|
onInput,
|
||||||
...restProps
|
...restProps
|
||||||
}: HTMLAttributes<HTMLDivElement> & {
|
}: HTMLAttributes<HTMLDivElement> & {
|
||||||
input?: FormInput<string | boolean | number>;
|
input?: FormInput<string | boolean | number>;
|
||||||
@@ -19,6 +20,7 @@
|
|||||||
description?: string;
|
description?: string;
|
||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
type?: 'text' | 'password' | 'email' | 'number' | 'checkbox';
|
type?: 'text' | 'password' | 'email' | 'number' | 'checkbox';
|
||||||
|
onInput?: (e: FormInputEvent) => void;
|
||||||
children?: Snippet;
|
children?: Snippet;
|
||||||
} = $props();
|
} = $props();
|
||||||
|
|
||||||
@@ -34,7 +36,7 @@
|
|||||||
{#if children}
|
{#if children}
|
||||||
{@render children()}
|
{@render children()}
|
||||||
{:else if input}
|
{:else if input}
|
||||||
<Input {id} {type} bind:value={input.value} {disabled} />
|
<Input {id} {type} bind:value={input.value} {disabled} on:input={(e) => onInput?.(e)} />
|
||||||
{/if}
|
{/if}
|
||||||
{#if input?.error}
|
{#if input?.error}
|
||||||
<p class="mt-1 text-sm text-red-500">{input.error}</p>
|
<p class="mt-1 text-sm text-red-500">{input.error}</p>
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
import * as DropdownMenu from '$lib/components/ui/dropdown-menu';
|
import * as DropdownMenu from '$lib/components/ui/dropdown-menu';
|
||||||
import WebAuthnService from '$lib/services/webauthn-service';
|
import WebAuthnService from '$lib/services/webauthn-service';
|
||||||
import userStore from '$lib/stores/user-store';
|
import userStore from '$lib/stores/user-store';
|
||||||
|
import { createSHA256hash } from '$lib/utils/crypto-util';
|
||||||
import { LucideLogOut, LucideUser } from 'lucide-svelte';
|
import { LucideLogOut, LucideUser } from 'lucide-svelte';
|
||||||
|
|
||||||
const webauthnService = new WebAuthnService();
|
const webauthnService = new WebAuthnService();
|
||||||
@@ -11,6 +12,13 @@
|
|||||||
($userStore!.firstName.charAt(0) + $userStore!.lastName?.charAt(0)).toUpperCase()
|
($userStore!.firstName.charAt(0) + $userStore!.lastName?.charAt(0)).toUpperCase()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
let gravatarURL: string | undefined = $state();
|
||||||
|
if ($userStore) {
|
||||||
|
createSHA256hash($userStore.email).then((email) => {
|
||||||
|
gravatarURL = `https://www.gravatar.com/avatar/${email}`;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async function logout() {
|
async function logout() {
|
||||||
await webauthnService.logout();
|
await webauthnService.logout();
|
||||||
window.location.reload();
|
window.location.reload();
|
||||||
@@ -19,7 +27,8 @@
|
|||||||
|
|
||||||
<DropdownMenu.Root>
|
<DropdownMenu.Root>
|
||||||
<DropdownMenu.Trigger
|
<DropdownMenu.Trigger
|
||||||
><Avatar.Root>
|
><Avatar.Root class="h-9 w-9">
|
||||||
|
<Avatar.Image src={gravatarURL} />
|
||||||
<Avatar.Fallback>{initials}</Avatar.Fallback>
|
<Avatar.Fallback>{initials}</Avatar.Fallback>
|
||||||
</Avatar.Root></DropdownMenu.Trigger
|
</Avatar.Root></DropdownMenu.Trigger
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -12,7 +12,11 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class=" w-full {isAuthPage ? 'absolute top-0 z-10 mt-4' : 'border-b'}">
|
<div class=" w-full {isAuthPage ? 'absolute top-0 z-10 mt-4' : 'border-b'}">
|
||||||
<div class="mx-auto flex w-full max-w-[1640px] items-center justify-between px-4 md:px-10">
|
<div
|
||||||
|
class="{!isAuthPage
|
||||||
|
? 'max-w-[1640px]'
|
||||||
|
: ''} mx-auto flex w-full items-center justify-between px-4 md:px-10"
|
||||||
|
>
|
||||||
<div class="flex h-16 items-center">
|
<div class="flex h-16 items-center">
|
||||||
{#if !isAuthPage}
|
{#if !isAuthPage}
|
||||||
<Logo class="mr-3 h-10 w-10" />
|
<Logo class="mr-3 h-10 w-10" />
|
||||||
|
|||||||
34
frontend/src/lib/components/ui/select/index.ts
Normal file
34
frontend/src/lib/components/ui/select/index.ts
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
|
||||||
|
import Label from "./select-label.svelte";
|
||||||
|
import Item from "./select-item.svelte";
|
||||||
|
import Content from "./select-content.svelte";
|
||||||
|
import Trigger from "./select-trigger.svelte";
|
||||||
|
import Separator from "./select-separator.svelte";
|
||||||
|
|
||||||
|
const Root = SelectPrimitive.Root;
|
||||||
|
const Group = SelectPrimitive.Group;
|
||||||
|
const Input = SelectPrimitive.Input;
|
||||||
|
const Value = SelectPrimitive.Value;
|
||||||
|
|
||||||
|
export {
|
||||||
|
Root,
|
||||||
|
Group,
|
||||||
|
Input,
|
||||||
|
Label,
|
||||||
|
Item,
|
||||||
|
Value,
|
||||||
|
Content,
|
||||||
|
Trigger,
|
||||||
|
Separator,
|
||||||
|
//
|
||||||
|
Root as Select,
|
||||||
|
Group as SelectGroup,
|
||||||
|
Input as SelectInput,
|
||||||
|
Label as SelectLabel,
|
||||||
|
Item as SelectItem,
|
||||||
|
Value as SelectValue,
|
||||||
|
Content as SelectContent,
|
||||||
|
Trigger as SelectTrigger,
|
||||||
|
Separator as SelectSeparator,
|
||||||
|
};
|
||||||
39
frontend/src/lib/components/ui/select/select-content.svelte
Normal file
39
frontend/src/lib/components/ui/select/select-content.svelte
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
import { scale } from "svelte/transition";
|
||||||
|
import { cn, flyAndScale } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = SelectPrimitive.ContentProps;
|
||||||
|
type $$Events = SelectPrimitive.ContentEvents;
|
||||||
|
|
||||||
|
export let sideOffset: $$Props["sideOffset"] = 4;
|
||||||
|
export let inTransition: $$Props["inTransition"] = flyAndScale;
|
||||||
|
export let inTransitionConfig: $$Props["inTransitionConfig"] = undefined;
|
||||||
|
export let outTransition: $$Props["outTransition"] = scale;
|
||||||
|
export let outTransitionConfig: $$Props["outTransitionConfig"] = {
|
||||||
|
start: 0.95,
|
||||||
|
opacity: 0,
|
||||||
|
duration: 50,
|
||||||
|
};
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<SelectPrimitive.Content
|
||||||
|
{inTransition}
|
||||||
|
{inTransitionConfig}
|
||||||
|
{outTransition}
|
||||||
|
{outTransitionConfig}
|
||||||
|
{sideOffset}
|
||||||
|
class={cn(
|
||||||
|
"bg-popover text-popover-foreground relative z-50 min-w-[8rem] overflow-hidden rounded-md border shadow-md outline-none",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...$$restProps}
|
||||||
|
on:keydown
|
||||||
|
>
|
||||||
|
<div class="w-full p-1">
|
||||||
|
<slot />
|
||||||
|
</div>
|
||||||
|
</SelectPrimitive.Content>
|
||||||
40
frontend/src/lib/components/ui/select/select-item.svelte
Normal file
40
frontend/src/lib/components/ui/select/select-item.svelte
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import Check from "lucide-svelte/icons/check";
|
||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
import { cn } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = SelectPrimitive.ItemProps;
|
||||||
|
type $$Events = SelectPrimitive.ItemEvents;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export let value: $$Props["value"];
|
||||||
|
export let label: $$Props["label"] = undefined;
|
||||||
|
export let disabled: $$Props["disabled"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<SelectPrimitive.Item
|
||||||
|
{value}
|
||||||
|
{disabled}
|
||||||
|
{label}
|
||||||
|
class={cn(
|
||||||
|
"data-[highlighted]:bg-accent data-[highlighted]:text-accent-foreground relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...$$restProps}
|
||||||
|
on:click
|
||||||
|
on:keydown
|
||||||
|
on:focusin
|
||||||
|
on:focusout
|
||||||
|
on:pointerleave
|
||||||
|
on:pointermove
|
||||||
|
>
|
||||||
|
<span class="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
|
||||||
|
<SelectPrimitive.ItemIndicator>
|
||||||
|
<Check class="h-4 w-4" />
|
||||||
|
</SelectPrimitive.ItemIndicator>
|
||||||
|
</span>
|
||||||
|
<slot>
|
||||||
|
{label || value}
|
||||||
|
</slot>
|
||||||
|
</SelectPrimitive.Item>
|
||||||
16
frontend/src/lib/components/ui/select/select-label.svelte
Normal file
16
frontend/src/lib/components/ui/select/select-label.svelte
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
import { cn } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = SelectPrimitive.LabelProps;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<SelectPrimitive.Label
|
||||||
|
class={cn("py-1.5 pl-8 pr-2 text-sm font-semibold", className)}
|
||||||
|
{...$$restProps}
|
||||||
|
>
|
||||||
|
<slot />
|
||||||
|
</SelectPrimitive.Label>
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
import { cn } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = SelectPrimitive.SeparatorProps;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<SelectPrimitive.Separator class={cn("bg-muted -mx-1 my-1 h-px", className)} {...$$restProps} />
|
||||||
27
frontend/src/lib/components/ui/select/select-trigger.svelte
Normal file
27
frontend/src/lib/components/ui/select/select-trigger.svelte
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Select as SelectPrimitive } from "bits-ui";
|
||||||
|
import ChevronDown from "lucide-svelte/icons/chevron-down";
|
||||||
|
import { cn } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = SelectPrimitive.TriggerProps;
|
||||||
|
type $$Events = SelectPrimitive.TriggerEvents;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<SelectPrimitive.Trigger
|
||||||
|
class={cn(
|
||||||
|
"border-input bg-background ring-offset-background focus-visible:ring-ring aria-[invalid]:border-destructive data-[placeholder]:[&>span]:text-muted-foreground flex h-10 w-full items-center justify-between rounded-md border px-3 py-2 text-sm focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 [&>span]:line-clamp-1",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...$$restProps}
|
||||||
|
let:builder
|
||||||
|
on:click
|
||||||
|
on:keydown
|
||||||
|
>
|
||||||
|
<slot {builder} />
|
||||||
|
<div>
|
||||||
|
<ChevronDown class="h-4 w-4 opacity-50" />
|
||||||
|
</div>
|
||||||
|
</SelectPrimitive.Trigger>
|
||||||
15
frontend/src/lib/components/ui/tooltip/index.ts
Normal file
15
frontend/src/lib/components/ui/tooltip/index.ts
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { Tooltip as TooltipPrimitive } from "bits-ui";
|
||||||
|
import Content from "./tooltip-content.svelte";
|
||||||
|
|
||||||
|
const Root = TooltipPrimitive.Root;
|
||||||
|
const Trigger = TooltipPrimitive.Trigger;
|
||||||
|
|
||||||
|
export {
|
||||||
|
Root,
|
||||||
|
Trigger,
|
||||||
|
Content,
|
||||||
|
//
|
||||||
|
Root as Tooltip,
|
||||||
|
Content as TooltipContent,
|
||||||
|
Trigger as TooltipTrigger,
|
||||||
|
};
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Tooltip as TooltipPrimitive } from "bits-ui";
|
||||||
|
import { cn, flyAndScale } from "$lib/utils/style.js";
|
||||||
|
|
||||||
|
type $$Props = TooltipPrimitive.ContentProps;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export let sideOffset: $$Props["sideOffset"] = 4;
|
||||||
|
export let transition: $$Props["transition"] = flyAndScale;
|
||||||
|
export let transitionConfig: $$Props["transitionConfig"] = {
|
||||||
|
y: 8,
|
||||||
|
duration: 150,
|
||||||
|
};
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<TooltipPrimitive.Content
|
||||||
|
{transition}
|
||||||
|
{transitionConfig}
|
||||||
|
{sideOffset}
|
||||||
|
class={cn(
|
||||||
|
"bg-popover text-popover-foreground z-50 overflow-hidden rounded-md border px-3 py-1.5 text-sm shadow-md",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...$$restProps}
|
||||||
|
>
|
||||||
|
<slot />
|
||||||
|
</TooltipPrimitive.Content>
|
||||||
@@ -13,7 +13,7 @@ abstract class APIService {
|
|||||||
if (browser) {
|
if (browser) {
|
||||||
this.api.defaults.baseURL = '/api';
|
this.api.defaults.baseURL = '/api';
|
||||||
} else {
|
} else {
|
||||||
this.api.defaults.baseURL = process?.env?.INTERNAL_BACKEND_URL + '/api';
|
this.api.defaults.baseURL = process!.env!.INTERNAL_BACKEND_URL + '/api';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,14 +4,8 @@ import APIService from './api-service';
|
|||||||
|
|
||||||
class AuditLogService extends APIService {
|
class AuditLogService extends APIService {
|
||||||
async list(pagination?: PaginationRequest) {
|
async list(pagination?: PaginationRequest) {
|
||||||
const page = pagination?.page || 1;
|
|
||||||
const limit = pagination?.limit || 10;
|
|
||||||
|
|
||||||
const res = await this.api.get('/audit-logs', {
|
const res = await this.api.get('/audit-logs', {
|
||||||
params: {
|
params: pagination
|
||||||
page,
|
|
||||||
limit
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
return res.data as Paginated<AuditLog>;
|
return res.data as Paginated<AuditLog>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import type { Paginated, PaginationRequest } from '$lib/types/pagination.type';
|
|||||||
import APIService from './api-service';
|
import APIService from './api-service';
|
||||||
|
|
||||||
class OidcService extends APIService {
|
class OidcService extends APIService {
|
||||||
async authorize(clientId: string, scope: string, callbackURL : string, nonce?: string) {
|
async authorize(clientId: string, scope: string, callbackURL: string, nonce?: string) {
|
||||||
const res = await this.api.post('/oidc/authorize', {
|
const res = await this.api.post('/oidc/authorize', {
|
||||||
scope,
|
scope,
|
||||||
nonce,
|
nonce,
|
||||||
@@ -26,14 +26,10 @@ class OidcService extends APIService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async listClients(search?: string, pagination?: PaginationRequest) {
|
async listClients(search?: string, pagination?: PaginationRequest) {
|
||||||
const page = pagination?.page || 1;
|
|
||||||
const limit = pagination?.limit || 10;
|
|
||||||
|
|
||||||
const res = await this.api.get('/oidc/clients', {
|
const res = await this.api.get('/oidc/clients', {
|
||||||
params: {
|
params: {
|
||||||
search,
|
search,
|
||||||
page,
|
...pagination
|
||||||
limit
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
return res.data as Paginated<OidcClient>;
|
return res.data as Paginated<OidcClient>;
|
||||||
|
|||||||
43
frontend/src/lib/services/user-group-service.ts
Normal file
43
frontend/src/lib/services/user-group-service.ts
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
import type { Paginated, PaginationRequest } from '$lib/types/pagination.type';
|
||||||
|
import type {
|
||||||
|
UserGroupCreate,
|
||||||
|
UserGroupWithUserCount,
|
||||||
|
UserGroupWithUsers
|
||||||
|
} from '$lib/types/user-group.type';
|
||||||
|
import APIService from './api-service';
|
||||||
|
|
||||||
|
export default class UserGroupService extends APIService {
|
||||||
|
async list(search?: string, pagination?: PaginationRequest) {
|
||||||
|
const res = await this.api.get('/user-groups', {
|
||||||
|
params: {
|
||||||
|
search,
|
||||||
|
...pagination
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return res.data as Paginated<UserGroupWithUserCount>;
|
||||||
|
}
|
||||||
|
|
||||||
|
async get(id: string) {
|
||||||
|
const res = await this.api.get(`/user-groups/${id}`);
|
||||||
|
return res.data as UserGroupWithUsers;
|
||||||
|
}
|
||||||
|
|
||||||
|
async create(user: UserGroupCreate) {
|
||||||
|
const res = await this.api.post('/user-groups', user);
|
||||||
|
return res.data as UserGroupWithUsers;
|
||||||
|
}
|
||||||
|
|
||||||
|
async update(id: string, user: UserGroupCreate) {
|
||||||
|
const res = await this.api.put(`/user-groups/${id}`, user);
|
||||||
|
return res.data as UserGroupWithUsers;
|
||||||
|
}
|
||||||
|
|
||||||
|
async remove(id: string) {
|
||||||
|
await this.api.delete(`/user-groups/${id}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
async updateUsers(id: string, userIds: string[]) {
|
||||||
|
const res = await this.api.put(`/user-groups/${id}/users`, { userIds });
|
||||||
|
return res.data as UserGroupWithUsers;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,14 +4,10 @@ import APIService from './api-service';
|
|||||||
|
|
||||||
export default class UserService extends APIService {
|
export default class UserService extends APIService {
|
||||||
async list(search?: string, pagination?: PaginationRequest) {
|
async list(search?: string, pagination?: PaginationRequest) {
|
||||||
const page = pagination?.page || 1;
|
|
||||||
const limit = pagination?.limit || 10;
|
|
||||||
|
|
||||||
const res = await this.api.get('/users', {
|
const res = await this.api.get('/users', {
|
||||||
params: {
|
params: {
|
||||||
search,
|
search,
|
||||||
page,
|
...pagination
|
||||||
limit
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
return res.data as Paginated<User>;
|
return res.data as Paginated<User>;
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ export type PaginationResponse = {
|
|||||||
totalPages: number;
|
totalPages: number;
|
||||||
totalItems: number;
|
totalItems: number;
|
||||||
currentPage: number;
|
currentPage: number;
|
||||||
|
itemsPerPage: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type Paginated<T> = {
|
export type Paginated<T> = {
|
||||||
|
|||||||
18
frontend/src/lib/types/user-group.type.ts
Normal file
18
frontend/src/lib/types/user-group.type.ts
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import type { User } from './user.type';
|
||||||
|
|
||||||
|
export type UserGroup = {
|
||||||
|
id: string;
|
||||||
|
friendlyName: string;
|
||||||
|
name: string;
|
||||||
|
createdAt: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UserGroupWithUsers = UserGroup & {
|
||||||
|
users: User[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UserGroupWithUserCount = UserGroup & {
|
||||||
|
userCount: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UserGroupCreate = Pick<UserGroup, 'friendlyName' | 'name'>;
|
||||||
7
frontend/src/lib/utils/crypto-util.ts
Normal file
7
frontend/src/lib/utils/crypto-util.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export async function createSHA256hash(input: string) {
|
||||||
|
const msgUint8 = new TextEncoder().encode(input); // encode as (utf-8) Uint8Array
|
||||||
|
const hashBuffer = await crypto.subtle.digest('SHA-256', msgUint8); // hash the message
|
||||||
|
const hashArray = Array.from(new Uint8Array(hashBuffer)); // convert buffer to byte array
|
||||||
|
const hashHex = hashArray.map((b) => b.toString(16).padStart(2, '0')).join(''); // convert bytes to hex string
|
||||||
|
return hashHex;
|
||||||
|
}
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
import { getWebauthnErrorMessage } from '$lib/utils/error-util';
|
import { getWebauthnErrorMessage } from '$lib/utils/error-util';
|
||||||
import { startAuthentication } from '@simplewebauthn/browser';
|
import { startAuthentication } from '@simplewebauthn/browser';
|
||||||
import { AxiosError } from 'axios';
|
import { AxiosError } from 'axios';
|
||||||
import { LucideMail, LucideUser } from 'lucide-svelte';
|
import { LucideMail, LucideUser, LucideUsers } from 'lucide-svelte';
|
||||||
import { slide } from 'svelte/transition';
|
import { slide } from 'svelte/transition';
|
||||||
import type { PageData } from './$types';
|
import type { PageData } from './$types';
|
||||||
import ClientProviderImages from './components/client-provider-images.svelte';
|
import ClientProviderImages from './components/client-provider-images.svelte';
|
||||||
@@ -113,6 +113,13 @@
|
|||||||
description="View your profile information"
|
description="View your profile information"
|
||||||
/>
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
|
{#if scope!.includes('groups')}
|
||||||
|
<ScopeItem
|
||||||
|
icon={LucideUsers}
|
||||||
|
name="Groups"
|
||||||
|
description="View the groups you are a member of"
|
||||||
|
/>
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
</Card.Content>
|
</Card.Content>
|
||||||
</Card.Root>
|
</Card.Root>
|
||||||
|
|||||||
@@ -18,6 +18,7 @@
|
|||||||
links = [
|
links = [
|
||||||
...links,
|
...links,
|
||||||
{ href: '/settings/admin/users', label: 'Users' },
|
{ href: '/settings/admin/users', label: 'Users' },
|
||||||
|
{ href: '/settings/admin/user-groups', label: 'User Groups' },
|
||||||
{ href: '/settings/admin/oidc-clients', label: 'OIDC Clients' },
|
{ href: '/settings/admin/oidc-clients', label: 'OIDC Clients' },
|
||||||
{ href: '/settings/admin/application-configuration', label: 'Application Configuration' }
|
{ href: '/settings/admin/application-configuration', label: 'Application Configuration' }
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
id,
|
id,
|
||||||
imageClass,
|
imageClass,
|
||||||
label,
|
label,
|
||||||
image = $bindable<File | null>(null),
|
image = $bindable(),
|
||||||
imageURL,
|
imageURL,
|
||||||
accept = 'image/png, image/jpeg, image/svg+xml',
|
accept = 'image/png, image/jpeg, image/svg+xml',
|
||||||
...restProps
|
...restProps
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
import { beforeNavigate } from '$app/navigation';
|
import { beforeNavigate } from '$app/navigation';
|
||||||
import { page } from '$app/stores';
|
import { page } from '$app/stores';
|
||||||
import { openConfirmDialog } from '$lib/components/confirm-dialog';
|
import { openConfirmDialog } from '$lib/components/confirm-dialog';
|
||||||
|
import CopyToClipboard from '$lib/components/copy-to-clipboard.svelte';
|
||||||
import { Button } from '$lib/components/ui/button';
|
import { Button } from '$lib/components/ui/button';
|
||||||
import * as Card from '$lib/components/ui/card';
|
import * as Card from '$lib/components/ui/card';
|
||||||
import Label from '$lib/components/ui/label/label.svelte';
|
import Label from '$lib/components/ui/label/label.svelte';
|
||||||
@@ -89,7 +90,9 @@
|
|||||||
<div class="flex flex-col">
|
<div class="flex flex-col">
|
||||||
<div class="mb-2 flex">
|
<div class="mb-2 flex">
|
||||||
<Label class="mb-0 w-44">Client ID</Label>
|
<Label class="mb-0 w-44">Client ID</Label>
|
||||||
<span class="text-muted-foreground text-sm" data-testid="client-id"> {client.id}</span>
|
<CopyToClipboard value={client.id}>
|
||||||
|
<span class="text-muted-foreground text-sm" data-testid="client-id"> {client.id}</span>
|
||||||
|
</CopyToClipboard>
|
||||||
</div>
|
</div>
|
||||||
<div class="mb-2 mt-1 flex items-center">
|
<div class="mb-2 mt-1 flex items-center">
|
||||||
<Label class="w-44">Client secret</Label>
|
<Label class="w-44">Client secret</Label>
|
||||||
@@ -111,7 +114,9 @@
|
|||||||
{#each Object.entries(setupDetails) as [key, value]}
|
{#each Object.entries(setupDetails) as [key, value]}
|
||||||
<div class="mb-5 flex">
|
<div class="mb-5 flex">
|
||||||
<Label class="mb-0 w-44">{key}</Label>
|
<Label class="mb-0 w-44">{key}</Label>
|
||||||
<span class="text-muted-foreground text-sm">{value}</span>
|
<CopyToClipboard {value}>
|
||||||
|
<span class="text-muted-foreground text-sm">{value}</span>
|
||||||
|
</CopyToClipboard>
|
||||||
</div>
|
</div>
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
import UserGroupService from '$lib/services/user-group-service';
|
||||||
|
import type { PageServerLoad } from './$types';
|
||||||
|
|
||||||
|
export const load: PageServerLoad = async ({ cookies }) => {
|
||||||
|
const userGroupService = new UserGroupService(cookies.get('access_token'));
|
||||||
|
const userGroups = await userGroupService.list();
|
||||||
|
return userGroups;
|
||||||
|
};
|
||||||
73
frontend/src/routes/settings/admin/user-groups/+page.svelte
Normal file
73
frontend/src/routes/settings/admin/user-groups/+page.svelte
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { goto } from '$app/navigation';
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import * as Card from '$lib/components/ui/card';
|
||||||
|
import UserGroupService from '$lib/services/user-group-service';
|
||||||
|
import type { Paginated } from '$lib/types/pagination.type';
|
||||||
|
import type { UserGroupCreate, UserGroupWithUserCount } from '$lib/types/user-group.type';
|
||||||
|
import { axiosErrorToast } from '$lib/utils/error-util';
|
||||||
|
import { LucideMinus } from 'lucide-svelte';
|
||||||
|
import { toast } from 'svelte-sonner';
|
||||||
|
import { slide } from 'svelte/transition';
|
||||||
|
import UserGroupForm from './user-group-form.svelte';
|
||||||
|
import UserGroupList from './user-group-list.svelte';
|
||||||
|
|
||||||
|
let { data } = $props();
|
||||||
|
let userGroups: Paginated<UserGroupWithUserCount> = $state(data);
|
||||||
|
let expandAddUserGroup = $state(false);
|
||||||
|
|
||||||
|
const userGroupService = new UserGroupService();
|
||||||
|
|
||||||
|
async function createUserGroup(userGroup: UserGroupCreate) {
|
||||||
|
let success = true;
|
||||||
|
await userGroupService
|
||||||
|
.create(userGroup)
|
||||||
|
.then((createdUserGroup) => {
|
||||||
|
toast.success('User group created successfully');
|
||||||
|
goto(`/settings/admin/user-groups/${createdUserGroup.id}`);
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
axiosErrorToast(e);
|
||||||
|
success = false;
|
||||||
|
});
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:head>
|
||||||
|
<title>User Groups</title>
|
||||||
|
</svelte:head>
|
||||||
|
|
||||||
|
<Card.Root>
|
||||||
|
<Card.Header>
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<div>
|
||||||
|
<Card.Title>Create User Group</Card.Title>
|
||||||
|
<Card.Description>Create a new group that can be assigned to users.</Card.Description>
|
||||||
|
</div>
|
||||||
|
{#if !expandAddUserGroup}
|
||||||
|
<Button on:click={() => (expandAddUserGroup = true)}>Add Group</Button>
|
||||||
|
{:else}
|
||||||
|
<Button class="h-8 p-3" variant="ghost" on:click={() => (expandAddUserGroup = false)}>
|
||||||
|
<LucideMinus class="h-5 w-5" />
|
||||||
|
</Button>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</Card.Header>
|
||||||
|
{#if expandAddUserGroup}
|
||||||
|
<div transition:slide>
|
||||||
|
<Card.Content>
|
||||||
|
<UserGroupForm callback={createUserGroup} />
|
||||||
|
</Card.Content>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</Card.Root>
|
||||||
|
|
||||||
|
<Card.Root>
|
||||||
|
<Card.Header>
|
||||||
|
<Card.Title>Manage User Groups</Card.Title>
|
||||||
|
</Card.Header>
|
||||||
|
<Card.Content>
|
||||||
|
<UserGroupList {userGroups} />
|
||||||
|
</Card.Content>
|
||||||
|
</Card.Root>
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
import UserGroupService from '$lib/services/user-group-service';
|
||||||
|
import type { PageServerLoad } from './$types';
|
||||||
|
|
||||||
|
export const load: PageServerLoad = async ({ params, cookies }) => {
|
||||||
|
const userGroupService = new UserGroupService(cookies.get('access_token'));
|
||||||
|
const userGroup = await userGroupService.get(params.id);
|
||||||
|
|
||||||
|
return { userGroup };
|
||||||
|
};
|
||||||
@@ -0,0 +1,78 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import * as Card from '$lib/components/ui/card';
|
||||||
|
import UserGroupService from '$lib/services/user-group-service';
|
||||||
|
import UserService from '$lib/services/user-service';
|
||||||
|
import type { UserGroupCreate } from '$lib/types/user-group.type';
|
||||||
|
import { axiosErrorToast } from '$lib/utils/error-util';
|
||||||
|
import { LucideChevronLeft } from 'lucide-svelte';
|
||||||
|
import { toast } from 'svelte-sonner';
|
||||||
|
import UserGroupForm from '../user-group-form.svelte';
|
||||||
|
import UserSelection from '../user-selection.svelte';
|
||||||
|
|
||||||
|
let { data } = $props();
|
||||||
|
let userGroup = $state({
|
||||||
|
...data.userGroup,
|
||||||
|
userIds: data.userGroup.users.map((u) => u.id)
|
||||||
|
});
|
||||||
|
|
||||||
|
const userGroupService = new UserGroupService();
|
||||||
|
const userService = new UserService();
|
||||||
|
|
||||||
|
async function updateUserGroup(updatedUserGroup: UserGroupCreate) {
|
||||||
|
let success = true;
|
||||||
|
await userGroupService
|
||||||
|
.update(userGroup.id, updatedUserGroup)
|
||||||
|
.then(() => toast.success('User group updated successfully'))
|
||||||
|
.catch((e) => {
|
||||||
|
axiosErrorToast(e);
|
||||||
|
success = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function updateUserGroupUsers(userIds: string[]) {
|
||||||
|
await userGroupService
|
||||||
|
.updateUsers(userGroup.id, userIds)
|
||||||
|
.then(() => toast.success('Users updated successfully'))
|
||||||
|
.catch((e) => {
|
||||||
|
axiosErrorToast(e);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:head>
|
||||||
|
<title>User Group Details {userGroup.name}</title>
|
||||||
|
</svelte:head>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<a class="text-muted-foreground flex text-sm" href="/settings/admin/user-groups"
|
||||||
|
><LucideChevronLeft class="h-5 w-5" /> Back</a
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<Card.Root>
|
||||||
|
<Card.Header>
|
||||||
|
<Card.Title>Meta data</Card.Title>
|
||||||
|
</Card.Header>
|
||||||
|
|
||||||
|
<Card.Content>
|
||||||
|
<UserGroupForm existingUserGroup={userGroup} callback={updateUserGroup} />
|
||||||
|
</Card.Content>
|
||||||
|
</Card.Root>
|
||||||
|
|
||||||
|
<Card.Root>
|
||||||
|
<Card.Header>
|
||||||
|
<Card.Title>Users</Card.Title>
|
||||||
|
<Card.Description>Assign users to this group.</Card.Description>
|
||||||
|
</Card.Header>
|
||||||
|
|
||||||
|
<Card.Content>
|
||||||
|
{#await userService.list() then users}
|
||||||
|
<UserSelection {users} bind:selectedUserIds={userGroup.userIds} />
|
||||||
|
{/await}
|
||||||
|
<div class="mt-5 flex justify-end">
|
||||||
|
<Button on:click={() => updateUserGroupUsers(userGroup.userIds)}>Save</Button>
|
||||||
|
</div>
|
||||||
|
</Card.Content>
|
||||||
|
</Card.Root>
|
||||||
@@ -0,0 +1,82 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import FormInput from '$lib/components/form-input.svelte';
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import type { UserGroupCreate } from '$lib/types/user-group.type';
|
||||||
|
import { createForm } from '$lib/utils/form-util';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
let {
|
||||||
|
callback,
|
||||||
|
existingUserGroup
|
||||||
|
}: {
|
||||||
|
existingUserGroup?: UserGroupCreate;
|
||||||
|
callback: (userGroup: UserGroupCreate) => Promise<boolean>;
|
||||||
|
} = $props();
|
||||||
|
|
||||||
|
let isLoading = $state(false);
|
||||||
|
let hasManualNameEdit = $state(!!existingUserGroup?.friendlyName);
|
||||||
|
|
||||||
|
const userGroup = {
|
||||||
|
name: existingUserGroup?.name || '',
|
||||||
|
friendlyName: existingUserGroup?.friendlyName || ''
|
||||||
|
};
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
friendlyName: z.string().min(2).max(30),
|
||||||
|
name: z
|
||||||
|
.string()
|
||||||
|
.min(2)
|
||||||
|
.max(30)
|
||||||
|
.regex(/^[a-z0-9_]+$/, 'Name can only contain lowercase letters, numbers, and underscores')
|
||||||
|
});
|
||||||
|
type FormSchema = typeof formSchema;
|
||||||
|
|
||||||
|
const { inputs, ...form } = createForm<FormSchema>(formSchema, userGroup);
|
||||||
|
|
||||||
|
function onFriendlyNameInput(e: any) {
|
||||||
|
if (!hasManualNameEdit) {
|
||||||
|
$inputs.name.value = e.target!.value.toLowerCase().replace(/[^a-z0-9_]/g, '_');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onNameInput(_: Event) {
|
||||||
|
hasManualNameEdit = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onSubmit() {
|
||||||
|
const data = form.validate();
|
||||||
|
if (!data) return;
|
||||||
|
isLoading = true;
|
||||||
|
const success = await callback(data);
|
||||||
|
// Reset form if user group was successfully created
|
||||||
|
if (success && !existingUserGroup) {
|
||||||
|
form.reset();
|
||||||
|
hasManualNameEdit = false;
|
||||||
|
}
|
||||||
|
isLoading = false;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<form onsubmit={onSubmit}>
|
||||||
|
<div class="flex flex-col gap-3 sm:flex-row">
|
||||||
|
<div class="w-full">
|
||||||
|
<FormInput
|
||||||
|
label="Friendly Name"
|
||||||
|
description="Name that will be displayed in the UI"
|
||||||
|
bind:input={$inputs.friendlyName}
|
||||||
|
onInput={onFriendlyNameInput}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="w-full">
|
||||||
|
<FormInput
|
||||||
|
label="Name"
|
||||||
|
description={`Name that will be in the "groups" claim`}
|
||||||
|
bind:input={$inputs.name}
|
||||||
|
onInput={onNameInput}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="mt-5 flex justify-end">
|
||||||
|
<Button {isLoading} type="submit">Save</Button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import AdvancedTable from '$lib/components/advanced-table.svelte';
|
||||||
|
import { openConfirmDialog } from '$lib/components/confirm-dialog/';
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import * as DropdownMenu from '$lib/components/ui/dropdown-menu';
|
||||||
|
import * as Table from '$lib/components/ui/table';
|
||||||
|
import UserGroupService from '$lib/services/user-group-service';
|
||||||
|
import type { Paginated } from '$lib/types/pagination.type';
|
||||||
|
import type { UserGroup, UserGroupWithUserCount } from '$lib/types/user-group.type';
|
||||||
|
import { axiosErrorToast } from '$lib/utils/error-util';
|
||||||
|
import { LucidePencil, LucideTrash } from 'lucide-svelte';
|
||||||
|
import Ellipsis from 'lucide-svelte/icons/ellipsis';
|
||||||
|
import { toast } from 'svelte-sonner';
|
||||||
|
|
||||||
|
let { userGroups: initialUserGroups }: { userGroups: Paginated<UserGroupWithUserCount> } =
|
||||||
|
$props();
|
||||||
|
|
||||||
|
let userGroups = $state<Paginated<UserGroupWithUserCount>>(initialUserGroups);
|
||||||
|
|
||||||
|
const userGroupService = new UserGroupService();
|
||||||
|
|
||||||
|
async function deleteUserGroup(userGroup: UserGroup) {
|
||||||
|
openConfirmDialog({
|
||||||
|
title: `Delete ${userGroup.name}`,
|
||||||
|
message: 'Are you sure you want to delete this user group?',
|
||||||
|
confirm: {
|
||||||
|
label: 'Delete',
|
||||||
|
destructive: true,
|
||||||
|
action: async () => {
|
||||||
|
try {
|
||||||
|
await userGroupService.remove(userGroup.id);
|
||||||
|
userGroups = await userGroupService.list();
|
||||||
|
} catch (e) {
|
||||||
|
axiosErrorToast(e);
|
||||||
|
}
|
||||||
|
toast.success('User group deleted successfully');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchItems(search: string, page: number, limit: number) {
|
||||||
|
return userGroupService.list(search, { page, limit });
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<AdvancedTable items={userGroups} {fetchItems} columns={['Friendly Name', 'Name', 'User Count', {label: "Actions", hidden: true}]}>
|
||||||
|
{#snippet rows({ item })}
|
||||||
|
<Table.Cell>{item.friendlyName}</Table.Cell>
|
||||||
|
<Table.Cell>{item.name}</Table.Cell>
|
||||||
|
<Table.Cell>{item.userCount}</Table.Cell>
|
||||||
|
<Table.Cell class="flex justify-end">
|
||||||
|
<DropdownMenu.Root>
|
||||||
|
<DropdownMenu.Trigger asChild let:builder>
|
||||||
|
<Button aria-haspopup="true" size="icon" variant="ghost" builders={[builder]}>
|
||||||
|
<Ellipsis class="h-4 w-4" />
|
||||||
|
<span class="sr-only">Toggle menu</span>
|
||||||
|
</Button>
|
||||||
|
</DropdownMenu.Trigger>
|
||||||
|
<DropdownMenu.Content align="end">
|
||||||
|
<DropdownMenu.Item href="/settings/admin/user-groups/{item.id}"
|
||||||
|
><LucidePencil class="mr-2 h-4 w-4" /> Edit</DropdownMenu.Item
|
||||||
|
>
|
||||||
|
<DropdownMenu.Item
|
||||||
|
class="text-red-500 focus:!text-red-700"
|
||||||
|
on:click={() => deleteUserGroup(item)}
|
||||||
|
><LucideTrash class="mr-2 h-4 w-4" />Delete</DropdownMenu.Item
|
||||||
|
>
|
||||||
|
</DropdownMenu.Content>
|
||||||
|
</DropdownMenu.Root>
|
||||||
|
</Table.Cell>
|
||||||
|
{/snippet}
|
||||||
|
</AdvancedTable>
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import AdvancedTable from '$lib/components/advanced-table.svelte';
|
||||||
|
import * as Table from '$lib/components/ui/table';
|
||||||
|
import UserService from '$lib/services/user-service';
|
||||||
|
import type { Paginated } from '$lib/types/pagination.type';
|
||||||
|
import type { User } from '$lib/types/user.type';
|
||||||
|
|
||||||
|
let {
|
||||||
|
users: initialUsers,
|
||||||
|
selectedUserIds = $bindable()
|
||||||
|
}: { users: Paginated<User>; selectedUserIds: string[] } = $props();
|
||||||
|
|
||||||
|
const userService = new UserService();
|
||||||
|
|
||||||
|
let users = $state(initialUsers);
|
||||||
|
|
||||||
|
function fetchItems(search: string, page: number, limit: number) {
|
||||||
|
return userService.list(search, { page, limit });
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<AdvancedTable
|
||||||
|
items={users}
|
||||||
|
{fetchItems}
|
||||||
|
columns={['Name', 'Email']}
|
||||||
|
bind:selectedIds={selectedUserIds}
|
||||||
|
>
|
||||||
|
{#snippet rows({ item })}
|
||||||
|
<Table.Cell>{item.firstName} {item.lastName}</Table.Cell>
|
||||||
|
<Table.Cell>{item.email}</Table.Cell>
|
||||||
|
{/snippet}
|
||||||
|
</AdvancedTable>
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
import { LucideMinus } from 'lucide-svelte';
|
import { LucideMinus } from 'lucide-svelte';
|
||||||
import { toast } from 'svelte-sonner';
|
import { toast } from 'svelte-sonner';
|
||||||
import { slide } from 'svelte/transition';
|
import { slide } from 'svelte/transition';
|
||||||
import CreateUser from './user-form.svelte';
|
import UserForm from './user-form.svelte';
|
||||||
import UserList from './user-list.svelte';
|
import UserList from './user-list.svelte';
|
||||||
|
|
||||||
let { data } = $props();
|
let { data } = $props();
|
||||||
@@ -56,7 +56,7 @@
|
|||||||
{#if expandAddUser}
|
{#if expandAddUser}
|
||||||
<div transition:slide>
|
<div transition:slide>
|
||||||
<Card.Content>
|
<Card.Content>
|
||||||
<CreateUser callback={createUser} />
|
<UserForm callback={createUser} />
|
||||||
</Card.Content>
|
</Card.Content>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|||||||
@@ -38,3 +38,20 @@ export const oidcClients = {
|
|||||||
secondCallbackUrl: 'http://pingvin.share/auth/callback2'
|
secondCallbackUrl: 'http://pingvin.share/auth/callback2'
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const userGroups = {
|
||||||
|
developers: {
|
||||||
|
id: '4110f814-56f1-4b28-8998-752b69bc97c0e',
|
||||||
|
friendlyName: 'Developers',
|
||||||
|
name: 'developers'
|
||||||
|
},
|
||||||
|
designers: {
|
||||||
|
id: 'adab18bf-f89d-4087-9ee1-70ff15b48211',
|
||||||
|
friendlyName: 'Designers',
|
||||||
|
name: 'designers'
|
||||||
|
},
|
||||||
|
humanResources: {
|
||||||
|
friendlyName: 'Human Resources',
|
||||||
|
name: 'human_resources'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|||||||
74
frontend/tests/user-group.spec.ts
Normal file
74
frontend/tests/user-group.spec.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import test, { expect } from '@playwright/test';
|
||||||
|
import { userGroups, users } from './data';
|
||||||
|
import { cleanupBackend } from './utils/cleanup.util';
|
||||||
|
|
||||||
|
test.beforeEach(cleanupBackend);
|
||||||
|
|
||||||
|
test('Create user group', async ({ page }) => {
|
||||||
|
await page.goto('/settings/admin/user-groups');
|
||||||
|
const group = userGroups.humanResources;
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Add Group' }).click();
|
||||||
|
await page.getByLabel('Friendly Name').fill(group.friendlyName);
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('status')).toHaveText('User group created successfully');
|
||||||
|
expect(page.url()).toMatch(/\/settings\/admin\/user-groups\/[a-f0-9-]+/);
|
||||||
|
|
||||||
|
await expect(page.getByLabel('Friendly Name')).toHaveValue(group.friendlyName);
|
||||||
|
await expect(page.getByLabel('Name', { exact: true })).toHaveValue(group.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Edit user group', async ({ page }) => {
|
||||||
|
await page.goto('/settings/admin/user-groups');
|
||||||
|
const group = userGroups.developers;
|
||||||
|
|
||||||
|
await page.getByRole('row', { name: group.name }).getByRole('button').click();
|
||||||
|
await page.getByRole('menuitem', { name: 'Edit' }).click();
|
||||||
|
|
||||||
|
await page.getByLabel('Friendly Name').fill('Developers updated');
|
||||||
|
|
||||||
|
await expect(page.getByLabel('Name', { exact: true })).toHaveValue(group.name);
|
||||||
|
|
||||||
|
await page.getByLabel('Name', { exact: true }).fill('developers_updated');
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Save' }).nth(0).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('status')).toHaveText('User group updated successfully');
|
||||||
|
await expect(page.getByLabel('Friendly Name')).toHaveValue('Developers updated');
|
||||||
|
await expect(page.getByLabel('Name', { exact: true })).toHaveValue('developers_updated');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Update user group users', async ({ page }) => {
|
||||||
|
const group = userGroups.designers;
|
||||||
|
await page.goto(`/settings/admin/user-groups/${group.id}`);
|
||||||
|
|
||||||
|
await page.getByRole('row', { name: users.tim.email }).getByRole('checkbox').click();
|
||||||
|
await page.getByRole('row', { name: users.craig.email }).getByRole('checkbox').click();
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Save' }).nth(1).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('status')).toHaveText('Users updated successfully');
|
||||||
|
|
||||||
|
await page.reload();
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
page.getByRole('row', { name: users.tim.email }).getByRole('checkbox')
|
||||||
|
).toHaveAttribute('data-state', 'unchecked');
|
||||||
|
await expect(
|
||||||
|
page.getByRole('row', { name: users.craig.email }).getByRole('checkbox')
|
||||||
|
).toHaveAttribute('data-state', 'checked');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Delete user group', async ({ page }) => {
|
||||||
|
const group = userGroups.developers;
|
||||||
|
await page.goto('/settings/admin/user-groups');
|
||||||
|
|
||||||
|
await page.getByRole('row', { name: group.name }).getByRole('button').click();
|
||||||
|
await page.getByRole('menuitem', { name: 'Delete' }).click();
|
||||||
|
await page.getByRole('button', { name: 'Delete' }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('status')).toHaveText('User group deleted successfully');
|
||||||
|
await expect(page.getByRole('row', { name: group.name })).not.toBeVisible();
|
||||||
|
});
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
:80 {
|
:80 {
|
||||||
reverse_proxy /api/* http://localhost:{$BACKEND_PORT:8080}
|
reverse_proxy /api/* http://localhost:{$BACKEND_PORT:8080}
|
||||||
reverse_proxy /.well-known/* http://localhost:{$BACKEND_PORT:8080}
|
reverse_proxy /.well-known/* http://localhost:{$BACKEND_PORT:8080}
|
||||||
reverse_proxy /* http://localhost:{$PORT:3000}
|
reverse_proxy /* http://localhost:{$PORT:3000}
|
||||||
|
|
||||||
log {
|
log {
|
||||||
output file /var/log/caddy/access.log
|
output file /var/log/caddy/access.log
|
||||||
level WARN
|
level WARN
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
:80 {
|
:80 {
|
||||||
reverse_proxy /api/* http://localhost:{$BACKEND_PORT:8080} {
|
reverse_proxy /api/* http://localhost:{$BACKEND_PORT:8080} {
|
||||||
trusted_proxies 0.0.0.0/0
|
trusted_proxies 0.0.0.0/0
|
||||||
}
|
}
|
||||||
reverse_proxy /.well-known/* http://localhost:{$BACKEND_PORT:8080} {
|
reverse_proxy /.well-known/* http://localhost:{$BACKEND_PORT:8080} {
|
||||||
trusted_proxies 0.0.0.0/0
|
trusted_proxies 0.0.0.0/0
|
||||||
}
|
}
|
||||||
reverse_proxy /* http://localhost:{$PORT:3000} {
|
reverse_proxy /* http://localhost:{$PORT:3000} {
|
||||||
trusted_proxies 0.0.0.0/0
|
trusted_proxies 0.0.0.0/0
|
||||||
}
|
}
|
||||||
|
|
||||||
log {
|
log {
|
||||||
output file /var/log/caddy/access.log
|
output file /var/log/caddy/access.log
|
||||||
level WARN
|
level WARN
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user