mirror of
https://github.com/pocket-id/pocket-id.git
synced 2025-12-12 00:03:00 +03:00
Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d2b3b7647d | ||
|
|
025378d14e | ||
|
|
e033ba6d45 | ||
|
|
e09562824a | ||
|
|
08f7fd16a9 | ||
|
|
be45eed125 | ||
|
|
9e94a436cc | ||
|
|
f82020ccfb | ||
|
|
a4a90a16a9 | ||
|
|
365734ec5d | ||
|
|
d02d8931a0 | ||
|
|
24c948e6a6 | ||
|
|
7a54d3ae20 | ||
|
|
5e1d19e0a4 | ||
|
|
d6a9bb4c09 | ||
|
|
3c67765992 | ||
|
|
6bb613e0e7 |
@@ -23,6 +23,9 @@ jobs:
|
|||||||
username: ${{ secrets.DOCKER_REGISTRY_USERNAME }}
|
username: ${{ secrets.DOCKER_REGISTRY_USERNAME }}
|
||||||
password: ${{ secrets.DOCKER_REGISTRY_PASSWORD }}
|
password: ${{ secrets.DOCKER_REGISTRY_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Download GeoLite2 City database
|
||||||
|
run: MAXMIND_LICENSE_KEY=${{ secrets.MAXMIND_LICENSE_KEY }} sh scripts/download-ip-database.sh
|
||||||
|
|
||||||
- name: Build and push
|
- name: Build and push
|
||||||
uses: docker/build-push-action@v4
|
uses: docker/build-push-action@v4
|
||||||
with:
|
with:
|
||||||
|
|||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -34,4 +34,5 @@ vite.config.ts.timestamp-*
|
|||||||
# Application specific
|
# Application specific
|
||||||
data
|
data
|
||||||
/frontend/tests/.auth
|
/frontend/tests/.auth
|
||||||
pocket-id-backend
|
pocket-id-backend
|
||||||
|
/backend/GeoLite2-City.mmdb
|
||||||
47
CHANGELOG.md
47
CHANGELOG.md
@@ -1,3 +1,50 @@
|
|||||||
|
## [](https://github.com/stonith404/pocket-id/compare/v0.7.1...v) (2024-10-04)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add location based on ip to the audit log ([025378d](https://github.com/stonith404/pocket-id/commit/025378d14edd2d72da76e90799a0ccdd42cf672c))
|
||||||
|
|
||||||
|
## [](https://github.com/stonith404/pocket-id/compare/v0.7.0...v) (2024-10-03)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* initials don't get displayed if Gravatar avatar doesn't exist ([e095628](https://github.com/stonith404/pocket-id/commit/e09562824a794bc7d240e9d229709d4b389db7d5))
|
||||||
|
|
||||||
|
## [](https://github.com/stonith404/pocket-id/compare/v0.6.0...v) (2024-10-03)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* add ability to set light and dark mode logo
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* add ability to set light and dark mode logo ([be45eed](https://github.com/stonith404/pocket-id/commit/be45eed125e33e9930572660a034d5f12dc310ce))
|
||||||
|
|
||||||
|
## [](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)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* add space to "Firstname" and "Lastname" label ([#31](https://github.com/stonith404/pocket-id/issues/31)) ([d6a9bb4](https://github.com/stonith404/pocket-id/commit/d6a9bb4c09efb8102da172e49c36c070b341f0fc))
|
||||||
|
* port environment variables get ignored in caddyfile ([3c67765](https://github.com/stonith404/pocket-id/commit/3c67765992d7369a79812bc8cd216c9ba12fd96e))
|
||||||
|
|
||||||
## [](https://github.com/stonith404/pocket-id/compare/v0.5.1...v) (2024-09-19)
|
## [](https://github.com/stonith404/pocket-id/compare/v0.5.1...v) (2024-09-19)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ COPY --from=frontend-builder /app/frontend/package.json ./frontend/package.json
|
|||||||
|
|
||||||
COPY --from=backend-builder /app/backend/pocket-id-backend ./backend/pocket-id-backend
|
COPY --from=backend-builder /app/backend/pocket-id-backend ./backend/pocket-id-backend
|
||||||
COPY --from=backend-builder /app/backend/migrations ./backend/migrations
|
COPY --from=backend-builder /app/backend/migrations ./backend/migrations
|
||||||
|
COPY --from=backend-builder /app/backend/GeoLite2-City.mmdb ./backend/GeoLite2-City.mmdb
|
||||||
COPY --from=backend-builder /app/backend/email-templates ./backend/email-templates
|
COPY --from=backend-builder /app/backend/email-templates ./backend/email-templates
|
||||||
COPY --from=backend-builder /app/backend/images ./backend/images
|
COPY --from=backend-builder /app/backend/images ./backend/images
|
||||||
|
|
||||||
|
|||||||
@@ -68,6 +68,10 @@ Required tools:
|
|||||||
cd ..
|
cd ..
|
||||||
pm2 start pocket-id-backend --name pocket-id-backend
|
pm2 start pocket-id-backend --name pocket-id-backend
|
||||||
|
|
||||||
|
# Optional: Download the GeoLite2 city database.
|
||||||
|
# If not downloaded the ip location in the audit log will be empty.
|
||||||
|
MAXMIND_LICENSE_KEY=<your-key> sh scripts/download-ip-database.sh
|
||||||
|
|
||||||
# Start the frontend
|
# Start the frontend
|
||||||
cd ../frontend
|
cd ../frontend
|
||||||
npm install
|
npm install
|
||||||
@@ -94,7 +98,7 @@ You may need the following information:
|
|||||||
- **Userinfo URL**: `https://<your-domain>/api/oidc/userinfo`
|
- **Userinfo URL**: `https://<your-domain>/api/oidc/userinfo`
|
||||||
- **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.
|
- **Scopes**: At least `openid email`. Optionally you can add `profile` and `groups`.
|
||||||
|
|
||||||
### Proxy Services with Pocket ID
|
### Proxy Services with Pocket ID
|
||||||
|
|
||||||
@@ -131,6 +135,9 @@ docker compose up -d
|
|||||||
cd ..
|
cd ..
|
||||||
pm2 start pocket-id-backend --name pocket-id-backend
|
pm2 start pocket-id-backend --name pocket-id-backend
|
||||||
|
|
||||||
|
# Optional: Update the GeoLite2 city database
|
||||||
|
MAXMIND_LICENSE_KEY=<your-key> sh scripts/download-ip-database.sh
|
||||||
|
|
||||||
# Start the frontend
|
# Start the frontend
|
||||||
cd ../frontend
|
cd ../frontend
|
||||||
npm install
|
npm install
|
||||||
|
|||||||
@@ -9,9 +9,15 @@
|
|||||||
<div class="content">
|
<div class="content">
|
||||||
<h2>New Sign-In Detected</h2>
|
<h2>New Sign-In Detected</h2>
|
||||||
<div class="grid">
|
<div class="grid">
|
||||||
|
{{ if and .Data.City .Data.Country }}
|
||||||
|
<div>
|
||||||
|
<p class="label">Approximate Location</p>
|
||||||
|
<p>{{ .Data.City }}, {{ .Data.Country }}</p>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
<div>
|
<div>
|
||||||
<p class="label">IP Address</p>
|
<p class="label">IP Address</p>
|
||||||
<p>{{ .Data.IPAddress}}</p>
|
<p>{{ .Data.IPAddress }}</p>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<p class="label">Device</p>
|
<p class="label">Device</p>
|
||||||
@@ -19,7 +25,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<p class="label">Sign-In Time</p>
|
<p class="label">Sign-In Time</p>
|
||||||
<p>{{ .Data.DateTime.Format "2006-01-02 15:04:05 UTC"}}</p>
|
<p>{{ .Data.DateTime.Format "2006-01-02 15:04:05 UTC" }}</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p class="message">
|
<p class="message">
|
||||||
@@ -27,4 +33,4 @@
|
|||||||
safely ignore this message. If not, please review your account and security settings.
|
safely ignore this message. If not, please review your account and security settings.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
{{ end -}}
|
{{ end -}}
|
||||||
@@ -2,6 +2,9 @@
|
|||||||
New Sign-In Detected
|
New Sign-In Detected
|
||||||
====================
|
====================
|
||||||
|
|
||||||
|
{{ if and .Data.City .Data.Country }}
|
||||||
|
Approximate Location: {{ .Data.City }}, {{ .Data.Country }}
|
||||||
|
{{ end }}
|
||||||
IP Address: {{ .Data.IPAddress }}
|
IP Address: {{ .Data.IPAddress }}
|
||||||
Device: {{ .Data.Device }}
|
Device: {{ .Data.Device }}
|
||||||
Time: {{ .Data.DateTime.Format "2006-01-02 15:04:05 UTC"}}
|
Time: {{ .Data.DateTime.Format "2006-01-02 15:04:05 UTC"}}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module github.com/stonith404/pocket-id/backend
|
module github.com/stonith404/pocket-id/backend
|
||||||
|
|
||||||
go 1.23
|
go 1.23.1
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/caarlos0/env/v11 v11.2.2
|
github.com/caarlos0/env/v11 v11.2.2
|
||||||
@@ -15,6 +15,7 @@ require (
|
|||||||
github.com/google/uuid v1.6.0
|
github.com/google/uuid v1.6.0
|
||||||
github.com/joho/godotenv v1.5.1
|
github.com/joho/godotenv v1.5.1
|
||||||
github.com/mileusna/useragent v1.3.4
|
github.com/mileusna/useragent v1.3.4
|
||||||
|
github.com/oschwald/maxminddb-golang/v2 v2.0.0-beta.1
|
||||||
golang.org/x/crypto v0.26.0
|
golang.org/x/crypto v0.26.0
|
||||||
golang.org/x/time v0.6.0
|
golang.org/x/time v0.6.0
|
||||||
gorm.io/driver/sqlite v1.5.6
|
gorm.io/driver/sqlite v1.5.6
|
||||||
|
|||||||
@@ -90,6 +90,8 @@ github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w
|
|||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
||||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
|
||||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
|
||||||
|
github.com/oschwald/maxminddb-golang/v2 v2.0.0-beta.1 h1:UihPOz+oIJ5X0JsO7wEkL50fheCODsoZ9r86mJWfNMc=
|
||||||
|
github.com/oschwald/maxminddb-golang/v2 v2.0.0-beta.1/go.mod h1:vPpFrres6g9B5+meBwAd9xnp335KFcLEFW7EqJxBHy0=
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
|
github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
|
github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
|
||||||
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
|
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
|
||||||
|
|||||||
3
backend/images/logoDark.svg
Normal file
3
backend/images/logoDark.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" id="a" viewBox="0 0 1015 1015">
|
||||||
|
<path fill="white" d="M506.6,0c209.52,0,379.98,170.45,379.98,379.96,0,82.33-25.9,160.68-74.91,226.54-48.04,64.59-113.78,111.51-190.13,135.71l-21.1,6.7-50.29-248.04,13.91-6.73c45.41-21.95,74.76-68.71,74.76-119.11,0-72.91-59.31-132.23-132.21-132.23s-132.23,59.32-132.23,132.23c0,50.4,29.36,97.16,74.77,119.11l13.65,6.61-81.01,499.24h-226.36V0h351.18Z"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 434 B |
3
backend/images/logoLight.svg
Normal file
3
backend/images/logoLight.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" id="a" viewBox="0 0 1015 1015">
|
||||||
|
<path fill="black" d="M506.6,0c209.52,0,379.98,170.45,379.98,379.96,0,82.33-25.9,160.68-74.91,226.54-48.04,64.59-113.78,111.51-190.13,135.71l-21.1,6.7-50.29-248.04,13.91-6.73c45.41-21.95,74.76-68.71,74.76-119.11,0-72.91-59.31-132.23-132.21-132.23s-132.23,59.32-132.23,132.23c0,50.4,29.36,97.16,74.77,119.11l13.65,6.61-81.01,499.24h-226.36V0h351.18Z"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 434 B |
@@ -5,24 +5,53 @@ import (
|
|||||||
"github.com/stonith404/pocket-id/backend/internal/utils"
|
"github.com/stonith404/pocket-id/backend/internal/utils"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// initApplicationImages copies the images from the images directory to the application-images directory
|
||||||
func initApplicationImages() {
|
func initApplicationImages() {
|
||||||
dirPath := common.EnvConfig.UploadPath + "/application-images"
|
dirPath := common.EnvConfig.UploadPath + "/application-images"
|
||||||
|
|
||||||
files, err := os.ReadDir(dirPath)
|
sourceFiles, err := os.ReadDir("./images")
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil && !os.IsNotExist(err) {
|
||||||
log.Fatalf("Error reading directory: %v", err)
|
log.Fatalf("Error reading directory: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Skip if files already exist
|
destinationFiles, err := os.ReadDir(dirPath)
|
||||||
if len(files) > 1 {
|
if err != nil && !os.IsNotExist(err) {
|
||||||
return
|
log.Fatalf("Error reading directory: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy files from source to destination
|
// Copy images from the images directory to the application-images directory if they don't already exist
|
||||||
err = utils.CopyDirectory("./images", dirPath)
|
for _, sourceFile := range sourceFiles {
|
||||||
if err != nil {
|
if sourceFile.IsDir() || imageAlreadyExists(sourceFile.Name(), destinationFiles) {
|
||||||
log.Fatalf("Error copying directory: %v", err)
|
continue
|
||||||
|
}
|
||||||
|
srcFilePath := "./images/" + sourceFile.Name()
|
||||||
|
destFilePath := dirPath + "/" + sourceFile.Name()
|
||||||
|
|
||||||
|
err := utils.CopyFile(srcFilePath, destFilePath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Error copying file: %v", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func imageAlreadyExists(fileName string, destinationFiles []os.DirEntry) bool {
|
||||||
|
for _, destinationFile := range destinationFiles {
|
||||||
|
sourceFileWithoutExtension := getImageNameWithoutExtension(fileName)
|
||||||
|
destinationFileWithoutExtension := getImageNameWithoutExtension(destinationFile.Name())
|
||||||
|
|
||||||
|
if sourceFileWithoutExtension == destinationFileWithoutExtension {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func getImageNameWithoutExtension(fileName string) string {
|
||||||
|
splitted := strings.Split(fileName, ".")
|
||||||
|
return strings.Join(splitted[:len(splitted)-1], ".")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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")
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -91,8 +91,20 @@ func (acc *AppConfigController) updateAppConfigHandler(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (acc *AppConfigController) getLogoHandler(c *gin.Context) {
|
func (acc *AppConfigController) getLogoHandler(c *gin.Context) {
|
||||||
imageType := acc.appConfigService.DbConfig.LogoImageType.Value
|
lightLogo := c.DefaultQuery("light", "true") == "true"
|
||||||
acc.getImage(c, "logo", imageType)
|
|
||||||
|
var imageName string
|
||||||
|
var imageType string
|
||||||
|
|
||||||
|
if lightLogo {
|
||||||
|
imageName = "logoLight"
|
||||||
|
imageType = acc.appConfigService.DbConfig.LogoLightImageType.Value
|
||||||
|
} else {
|
||||||
|
imageName = "logoDark"
|
||||||
|
imageType = acc.appConfigService.DbConfig.LogoDarkImageType.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
acc.getImage(c, imageName, imageType)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (acc *AppConfigController) getFaviconHandler(c *gin.Context) {
|
func (acc *AppConfigController) getFaviconHandler(c *gin.Context) {
|
||||||
@@ -105,8 +117,20 @@ func (acc *AppConfigController) getBackgroundImageHandler(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (acc *AppConfigController) updateLogoHandler(c *gin.Context) {
|
func (acc *AppConfigController) updateLogoHandler(c *gin.Context) {
|
||||||
imageType := acc.appConfigService.DbConfig.LogoImageType.Value
|
lightLogo := c.DefaultQuery("light", "true") == "true"
|
||||||
acc.updateImage(c, "logo", imageType)
|
|
||||||
|
var imageName string
|
||||||
|
var imageType string
|
||||||
|
|
||||||
|
if lightLogo {
|
||||||
|
imageName = "logoLight"
|
||||||
|
imageType = acc.appConfigService.DbConfig.LogoLightImageType.Value
|
||||||
|
} else {
|
||||||
|
imageName = "logoDark"
|
||||||
|
imageType = acc.appConfigService.DbConfig.LogoDarkImageType.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
acc.updateImage(c, imageName, imageType)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (acc *AppConfigController) updateFaviconHandler(c *gin.Context) {
|
func (acc *AppConfigController) updateFaviconHandler(c *gin.Context) {
|
||||||
|
|||||||
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)
|
||||||
|
}
|
||||||
@@ -11,6 +11,8 @@ type AuditLogDto struct {
|
|||||||
|
|
||||||
Event model.AuditLogEvent `json:"event"`
|
Event model.AuditLogEvent `json:"event"`
|
||||||
IpAddress string `json:"ipAddress"`
|
IpAddress string `json:"ipAddress"`
|
||||||
|
Country string `json:"country"`
|
||||||
|
City string `json:"city"`
|
||||||
Device string `json:"device"`
|
Device string `json:"device"`
|
||||||
UserID string `json:"userID"`
|
UserID string `json:"userID"`
|
||||||
Data model.AuditLogData `json:"data"`
|
Data model.AuditLogData `json:"data"`
|
||||||
|
|||||||
@@ -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)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,7 +11,8 @@ type AppConfigVariable struct {
|
|||||||
type AppConfig struct {
|
type AppConfig struct {
|
||||||
AppName AppConfigVariable
|
AppName AppConfigVariable
|
||||||
BackgroundImageType AppConfigVariable
|
BackgroundImageType AppConfigVariable
|
||||||
LogoImageType AppConfigVariable
|
LogoLightImageType AppConfigVariable
|
||||||
|
LogoDarkImageType AppConfigVariable
|
||||||
SessionDuration AppConfigVariable
|
SessionDuration AppConfigVariable
|
||||||
|
|
||||||
EmailEnabled AppConfigVariable
|
EmailEnabled AppConfigVariable
|
||||||
|
|||||||
@@ -11,6 +11,8 @@ type AuditLog struct {
|
|||||||
|
|
||||||
Event AuditLogEvent
|
Event AuditLogEvent
|
||||||
IpAddress string
|
IpAddress string
|
||||||
|
Country string
|
||||||
|
City string
|
||||||
UserAgent string
|
UserAgent string
|
||||||
UserID string
|
UserID string
|
||||||
Data AuditLogData
|
Data AuditLogData
|
||||||
|
|||||||
@@ -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;"`
|
||||||
|
}
|
||||||
@@ -47,8 +47,14 @@ var defaultDbConfig = model.AppConfig{
|
|||||||
IsInternal: true,
|
IsInternal: true,
|
||||||
Value: "jpg",
|
Value: "jpg",
|
||||||
},
|
},
|
||||||
LogoImageType: model.AppConfigVariable{
|
LogoLightImageType: model.AppConfigVariable{
|
||||||
Key: "logoImageType",
|
Key: "logoLightImageType",
|
||||||
|
Type: "string",
|
||||||
|
IsInternal: true,
|
||||||
|
Value: "svg",
|
||||||
|
},
|
||||||
|
LogoDarkImageType: model.AppConfigVariable{
|
||||||
|
Key: "logoDarkImageType",
|
||||||
Type: "string",
|
Type: "string",
|
||||||
IsInternal: true,
|
IsInternal: true,
|
||||||
Value: "svg",
|
Value: "svg",
|
||||||
|
|||||||
@@ -2,11 +2,13 @@ package service
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
userAgentParser "github.com/mileusna/useragent"
|
userAgentParser "github.com/mileusna/useragent"
|
||||||
|
"github.com/oschwald/maxminddb-golang/v2"
|
||||||
"github.com/stonith404/pocket-id/backend/internal/model"
|
"github.com/stonith404/pocket-id/backend/internal/model"
|
||||||
"github.com/stonith404/pocket-id/backend/internal/utils"
|
"github.com/stonith404/pocket-id/backend/internal/utils"
|
||||||
"github.com/stonith404/pocket-id/backend/internal/utils/email"
|
"github.com/stonith404/pocket-id/backend/internal/utils/email"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
"log"
|
"log"
|
||||||
|
"net/netip"
|
||||||
)
|
)
|
||||||
|
|
||||||
type AuditLogService struct {
|
type AuditLogService struct {
|
||||||
@@ -21,9 +23,16 @@ func NewAuditLogService(db *gorm.DB, appConfigService *AppConfigService, emailSe
|
|||||||
|
|
||||||
// Create creates a new audit log entry in the database
|
// Create creates a new audit log entry in the database
|
||||||
func (s *AuditLogService) Create(event model.AuditLogEvent, ipAddress, userAgent, userID string, data model.AuditLogData) model.AuditLog {
|
func (s *AuditLogService) Create(event model.AuditLogEvent, ipAddress, userAgent, userID string, data model.AuditLogData) model.AuditLog {
|
||||||
|
country, city, err := s.GetIpLocation(ipAddress)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Failed to get IP location: %v\n", err)
|
||||||
|
}
|
||||||
|
|
||||||
auditLog := model.AuditLog{
|
auditLog := model.AuditLog{
|
||||||
Event: event,
|
Event: event,
|
||||||
IpAddress: ipAddress,
|
IpAddress: ipAddress,
|
||||||
|
Country: country,
|
||||||
|
City: city,
|
||||||
UserAgent: userAgent,
|
UserAgent: userAgent,
|
||||||
UserID: userID,
|
UserID: userID,
|
||||||
Data: data,
|
Data: data,
|
||||||
@@ -61,6 +70,8 @@ func (s *AuditLogService) CreateNewSignInWithEmail(ipAddress, userAgent, userID
|
|||||||
Email: user.Email,
|
Email: user.Email,
|
||||||
}, NewLoginTemplate, &NewLoginTemplateData{
|
}, NewLoginTemplate, &NewLoginTemplateData{
|
||||||
IPAddress: ipAddress,
|
IPAddress: ipAddress,
|
||||||
|
Country: createdAuditLog.Country,
|
||||||
|
City: createdAuditLog.City,
|
||||||
Device: s.DeviceStringFromUserAgent(userAgent),
|
Device: s.DeviceStringFromUserAgent(userAgent),
|
||||||
DateTime: createdAuditLog.CreatedAt.UTC(),
|
DateTime: createdAuditLog.CreatedAt.UTC(),
|
||||||
})
|
})
|
||||||
@@ -86,3 +97,29 @@ func (s *AuditLogService) DeviceStringFromUserAgent(userAgent string) string {
|
|||||||
ua := userAgentParser.Parse(userAgent)
|
ua := userAgentParser.Parse(userAgent)
|
||||||
return ua.Name + " on " + ua.OS + " " + ua.OSVersion
|
return ua.Name + " on " + ua.OS + " " + ua.OSVersion
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *AuditLogService) GetIpLocation(ipAddress string) (country, city string, err error) {
|
||||||
|
db, err := maxminddb.Open("GeoLite2-City.mmdb")
|
||||||
|
if err != nil {
|
||||||
|
return "", "", err
|
||||||
|
}
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
|
addr := netip.MustParseAddr(ipAddress)
|
||||||
|
|
||||||
|
var record struct {
|
||||||
|
City struct {
|
||||||
|
Names map[string]string `maxminddb:"names"`
|
||||||
|
} `maxminddb:"city"`
|
||||||
|
Country struct {
|
||||||
|
Names map[string]string `maxminddb:"names"`
|
||||||
|
} `maxminddb:"country"`
|
||||||
|
}
|
||||||
|
|
||||||
|
err = db.Lookup(addr).Decode(&record)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return record.Country.Names["en"], record.City.Names["en"], nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -29,6 +29,8 @@ var NewLoginTemplate = email.Template[NewLoginTemplateData]{
|
|||||||
|
|
||||||
type NewLoginTemplateData struct {
|
type NewLoginTemplateData struct {
|
||||||
IPAddress string
|
IPAddress string
|
||||||
|
Country string
|
||||||
|
City string
|
||||||
Device string
|
Device string
|
||||||
DateTime time.Time
|
DateTime time.Time
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
|
}
|
||||||
@@ -38,7 +38,7 @@ func CopyDirectory(srcDir, destDir string) error {
|
|||||||
srcFilePath := filepath.Join(srcDir, file.Name())
|
srcFilePath := filepath.Join(srcDir, file.Name())
|
||||||
destFilePath := filepath.Join(destDir, file.Name())
|
destFilePath := filepath.Join(destDir, file.Name())
|
||||||
|
|
||||||
err := copyFile(srcFilePath, destFilePath)
|
err := CopyFile(srcFilePath, destFilePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -47,7 +47,7 @@ func CopyDirectory(srcDir, destDir string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func copyFile(srcFilePath, destFilePath string) error {
|
func CopyFile(srcFilePath, destFilePath string) error {
|
||||||
srcFile, err := os.Open(srcFilePath)
|
srcFile, err := os.Open(srcFilePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|||||||
@@ -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
|
||||||
|
);
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
ALTER TABLE audit_logs DROP COLUMN country;
|
||||||
|
ALTER TABLE audit_logs DROP COLUMN city;
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
ALTER TABLE audit_logs ADD COLUMN country TEXT;
|
||||||
|
ALTER TABLE audit_logs ADD COLUMN city TEXT;
|
||||||
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",
|
||||||
|
|||||||
@@ -97,16 +97,4 @@
|
|||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
src: url('/fonts/PlayfairDisplay-Bold.woff') format('woff');
|
src: url('/fonts/PlayfairDisplay-Bold.woff') format('woff');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@layer components {
|
|
||||||
.application-images-grid {
|
|
||||||
@apply flex flex-wrap justify-between gap-x-5 gap-y-8;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 1127px) {
|
|
||||||
.application-images-grid {
|
|
||||||
justify-content: flex-start;
|
|
||||||
@apply gap-x-20;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
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}?d=404`;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
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" />
|
||||||
|
|||||||
@@ -1 +1,10 @@
|
|||||||
<img class={$$restProps.class} src="/api/application-configuration/logo" alt="Logo" />
|
<script lang="ts">
|
||||||
|
import { mode } from 'mode-watcher';
|
||||||
|
import type { HTMLAttributes } from 'svelte/elements';
|
||||||
|
|
||||||
|
let { ...props }: HTMLAttributes<HTMLImageElement> = $props();
|
||||||
|
|
||||||
|
const isDarkMode = $derived($mode === 'dark');
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<img {...props} src="/api/application-configuration/logo?light={!isDarkMode}" alt="Logo" />
|
||||||
|
|||||||
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';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,4 @@
|
|||||||
import type {
|
import type { AllAppConfig, AppConfigRawResponse } from '$lib/types/application-configuration';
|
||||||
AllAppConfig,
|
|
||||||
AppConfigRawResponse
|
|
||||||
} from '$lib/types/application-configuration';
|
|
||||||
import APIService from './api-service';
|
import APIService from './api-service';
|
||||||
|
|
||||||
export default class AppConfigService extends APIService {
|
export default class AppConfigService extends APIService {
|
||||||
@@ -33,11 +30,13 @@ export default class AppConfigService extends APIService {
|
|||||||
await this.api.put(`/application-configuration/favicon`, formData);
|
await this.api.put(`/application-configuration/favicon`, formData);
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateLogo(logo: File) {
|
async updateLogo(logo: File, light = true) {
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
formData.append('file', logo!);
|
formData.append('file', logo!);
|
||||||
|
|
||||||
await this.api.put(`/application-configuration/logo`, formData);
|
await this.api.put(`/application-configuration/logo`, formData, {
|
||||||
|
params: { light }
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateBackgroundImage(backgroundImage: File) {
|
async updateBackgroundImage(backgroundImage: File) {
|
||||||
|
|||||||
@@ -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>;
|
||||||
|
|||||||
@@ -2,6 +2,8 @@ export type AuditLog = {
|
|||||||
id: string;
|
id: string;
|
||||||
event: string;
|
event: string;
|
||||||
ipAddress: string;
|
ipAddress: string;
|
||||||
|
country?: string;
|
||||||
|
city?: string;
|
||||||
device: string;
|
device: string;
|
||||||
createdAt: string;
|
createdAt: string;
|
||||||
data: any;
|
data: any;
|
||||||
|
|||||||
@@ -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' }
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -38,10 +38,10 @@
|
|||||||
<form onsubmit={onSubmit}>
|
<form onsubmit={onSubmit}>
|
||||||
<div class="flex flex-col gap-3 sm:flex-row">
|
<div class="flex flex-col gap-3 sm:flex-row">
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
<FormInput label="Firstname" bind:input={$inputs.firstName} />
|
<FormInput label="First name" bind:input={$inputs.firstName} />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
<FormInput label="Lastname" bind:input={$inputs.lastName} />
|
<FormInput label="Last name" bind:input={$inputs.lastName} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-3 flex flex-col gap-3 sm:flex-row">
|
<div class="mt-3 flex flex-col gap-3 sm:flex-row">
|
||||||
|
|||||||
@@ -28,17 +28,19 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function updateImages(
|
async function updateImages(
|
||||||
logo: File | null,
|
logoLight: File | null,
|
||||||
|
logoDark: File | null,
|
||||||
backgroundImage: File | null,
|
backgroundImage: File | null,
|
||||||
favicon: File | null
|
favicon: File | null
|
||||||
) {
|
) {
|
||||||
const faviconPromise = favicon ? appConfigService.updateFavicon(favicon) : Promise.resolve();
|
const faviconPromise = favicon ? appConfigService.updateFavicon(favicon) : Promise.resolve();
|
||||||
const logoPromise = logo ? appConfigService.updateLogo(logo) : Promise.resolve();
|
const lightLogoPromise = logoLight ? appConfigService.updateLogo(logoLight, true) : Promise.resolve();
|
||||||
|
const darkLogoPromise = logoDark ? appConfigService.updateLogo(logoDark, false) : Promise.resolve();
|
||||||
const backgroundImagePromise = backgroundImage
|
const backgroundImagePromise = backgroundImage
|
||||||
? appConfigService.updateBackgroundImage(backgroundImage)
|
? appConfigService.updateBackgroundImage(backgroundImage)
|
||||||
: Promise.resolve();
|
: Promise.resolve();
|
||||||
|
|
||||||
await Promise.all([logoPromise, backgroundImagePromise, faviconPromise])
|
await Promise.all([lightLogoPromise, darkLogoPromise, backgroundImagePromise, faviconPromise])
|
||||||
.then(() => toast.success('Images updated successfully'))
|
.then(() => toast.success('Images updated successfully'))
|
||||||
.catch(axiosErrorToast);
|
.catch(axiosErrorToast);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,9 +8,10 @@
|
|||||||
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',
|
||||||
|
forceColorScheme,
|
||||||
...restProps
|
...restProps
|
||||||
}: HTMLAttributes<HTMLDivElement> & {
|
}: HTMLAttributes<HTMLDivElement> & {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -18,6 +19,7 @@
|
|||||||
label: string;
|
label: string;
|
||||||
image: File | null;
|
image: File | null;
|
||||||
imageURL: string;
|
imageURL: string;
|
||||||
|
forceColorScheme?: 'light' | 'dark';
|
||||||
accept?: string;
|
accept?: string;
|
||||||
} = $props();
|
} = $props();
|
||||||
|
|
||||||
@@ -37,10 +39,16 @@
|
|||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div {...restProps}>
|
<div class="flex flex-col items-start md:flex-row md:items-center" {...restProps}>
|
||||||
<Label for={id}>{label}</Label>
|
<Label class="w-52" for={id}>{label}</Label>
|
||||||
<FileInput {id} variant="secondary" {accept} onchange={onImageChange}>
|
<FileInput {id} variant="secondary" {accept} onchange={onImageChange}>
|
||||||
<div class="bg-muted group relative flex items-center rounded">
|
<div
|
||||||
|
class="{forceColorScheme === 'light'
|
||||||
|
? 'bg-[#F1F1F5]'
|
||||||
|
: forceColorScheme === 'dark'
|
||||||
|
? 'bg-[#27272A]'
|
||||||
|
: 'bg-muted'} group relative flex items-center rounded"
|
||||||
|
>
|
||||||
<img
|
<img
|
||||||
class={cn(
|
class={cn(
|
||||||
'h-full w-full rounded object-cover p-3 transition-opacity duration-200 group-hover:opacity-10',
|
'h-full w-full rounded object-cover p-3 transition-opacity duration-200 group-hover:opacity-10',
|
||||||
|
|||||||
@@ -5,15 +5,21 @@
|
|||||||
let {
|
let {
|
||||||
callback
|
callback
|
||||||
}: {
|
}: {
|
||||||
callback: (logo: File | null, backgroundImage: File | null, favicon: File | null) => void;
|
callback: (
|
||||||
|
logoLight: File | null,
|
||||||
|
logoDark: File | null,
|
||||||
|
backgroundImage: File | null,
|
||||||
|
favicon: File | null
|
||||||
|
) => void;
|
||||||
} = $props();
|
} = $props();
|
||||||
|
|
||||||
let logo = $state<File | null>(null);
|
let logoLight = $state<File | null>(null);
|
||||||
|
let logoDark = $state<File | null>(null);
|
||||||
let backgroundImage = $state<File | null>(null);
|
let backgroundImage = $state<File | null>(null);
|
||||||
let favicon = $state<File | null>(null);
|
let favicon = $state<File | null>(null);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="application-images-grid">
|
<div class="flex flex-col gap-8">
|
||||||
<ApplicationImage
|
<ApplicationImage
|
||||||
id="favicon"
|
id="favicon"
|
||||||
imageClass="h-14 w-14 p-2"
|
imageClass="h-14 w-14 p-2"
|
||||||
@@ -23,15 +29,23 @@
|
|||||||
accept="image/x-icon"
|
accept="image/x-icon"
|
||||||
/>
|
/>
|
||||||
<ApplicationImage
|
<ApplicationImage
|
||||||
id="logo"
|
id="logo-light"
|
||||||
imageClass="h-32 w-32"
|
imageClass="h-32 w-32"
|
||||||
label="Logo"
|
label="Light Mode Logo"
|
||||||
bind:image={logo}
|
bind:image={logoLight}
|
||||||
imageURL="/api/application-configuration/logo"
|
imageURL="/api/application-configuration/logo?light=true"
|
||||||
|
forceColorScheme="light"
|
||||||
|
/>
|
||||||
|
<ApplicationImage
|
||||||
|
id="logo-dark"
|
||||||
|
imageClass="h-32 w-32"
|
||||||
|
label="Dark Mode Logo"
|
||||||
|
bind:image={logoDark}
|
||||||
|
imageURL="/api/application-configuration/logo?light=false"
|
||||||
|
forceColorScheme="dark"
|
||||||
/>
|
/>
|
||||||
<ApplicationImage
|
<ApplicationImage
|
||||||
id="background-image"
|
id="background-image"
|
||||||
class="basis-full lg:basis-auto"
|
|
||||||
imageClass="h-[350px] max-w-[500px]"
|
imageClass="h-[350px] max-w-[500px]"
|
||||||
label="Background Image"
|
label="Background Image"
|
||||||
bind:image={backgroundImage}
|
bind:image={backgroundImage}
|
||||||
@@ -39,5 +53,7 @@
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex justify-end">
|
<div class="flex justify-end">
|
||||||
<Button class="mt-5" onclick={() => callback(logo, backgroundImage, favicon)}>Save</Button>
|
<Button class="mt-5" onclick={() => callback(logoLight, logoDark, backgroundImage, favicon)}
|
||||||
|
>Save</Button
|
||||||
|
>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -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';
|
||||||
@@ -26,7 +27,6 @@
|
|||||||
'Token URL': `https://${$page.url.hostname}/api/oidc/token`,
|
'Token URL': `https://${$page.url.hostname}/api/oidc/token`,
|
||||||
'Userinfo URL': `https://${$page.url.hostname}/api/oidc/userinfo`,
|
'Userinfo URL': `https://${$page.url.hostname}/api/oidc/userinfo`,
|
||||||
'Certificate URL': `https://${$page.url.hostname}/.well-known/jwks.json`,
|
'Certificate URL': `https://${$page.url.hostname}/.well-known/jwks.json`,
|
||||||
PKCE: 'Disabled'
|
|
||||||
};
|
};
|
||||||
|
|
||||||
async function updateClient(updatedClient: OidcClientCreateWithLogo) {
|
async function updateClient(updatedClient: OidcClientCreateWithLogo) {
|
||||||
@@ -89,7 +89,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 +113,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}
|
||||||
|
|||||||
@@ -56,10 +56,10 @@
|
|||||||
<form onsubmit={onSubmit}>
|
<form onsubmit={onSubmit}>
|
||||||
<div class="flex flex-col gap-3 sm:flex-row">
|
<div class="flex flex-col gap-3 sm:flex-row">
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
<FormInput label="Firstname" bind:input={$inputs.firstName} />
|
<FormInput label="First name" bind:input={$inputs.firstName} />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
<FormInput label="Lastname" bind:input={$inputs.lastName} />
|
<FormInput label="Last name" bind:input={$inputs.lastName} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-3 flex flex-col gap-3 sm:flex-row">
|
<div class="mt-3 flex flex-col gap-3 sm:flex-row">
|
||||||
|
|||||||
@@ -30,6 +30,7 @@
|
|||||||
<Table.Row>
|
<Table.Row>
|
||||||
<Table.Head>Time</Table.Head>
|
<Table.Head>Time</Table.Head>
|
||||||
<Table.Head>Event</Table.Head>
|
<Table.Head>Event</Table.Head>
|
||||||
|
<Table.Head>Approximate Location</Table.Head>
|
||||||
<Table.Head>IP Address</Table.Head>
|
<Table.Head>IP Address</Table.Head>
|
||||||
<Table.Head>Device</Table.Head>
|
<Table.Head>Device</Table.Head>
|
||||||
<Table.Head>Client</Table.Head>
|
<Table.Head>Client</Table.Head>
|
||||||
@@ -47,6 +48,7 @@
|
|||||||
<Table.Cell>
|
<Table.Cell>
|
||||||
<Badge variant="outline">{toFriendlyEventString(auditLog.event)}</Badge>
|
<Badge variant="outline">{toFriendlyEventString(auditLog.event)}</Badge>
|
||||||
</Table.Cell>
|
</Table.Cell>
|
||||||
|
<Table.Cell>{auditLog.city && auditLog.country ? `${auditLog.city}, ${auditLog.country}` : 'Unknown'}</Table.Cell>
|
||||||
<Table.Cell>{auditLog.ipAddress}</Table.Cell>
|
<Table.Cell>{auditLog.ipAddress}</Table.Cell>
|
||||||
<Table.Cell>{auditLog.device}</Table.Cell>
|
<Table.Cell>{auditLog.device}</Table.Cell>
|
||||||
<Table.Cell>{auditLog.data.clientName}</Table.Cell>
|
<Table.Cell>{auditLog.data.clientName}</Table.Cell>
|
||||||
|
|||||||
@@ -8,8 +8,8 @@ test.beforeEach(cleanupBackend);
|
|||||||
test('Update account details', async ({ page }) => {
|
test('Update account details', async ({ page }) => {
|
||||||
await page.goto('/settings/account');
|
await page.goto('/settings/account');
|
||||||
|
|
||||||
await page.getByLabel('Firstname').fill('Timothy');
|
await page.getByLabel('First name').fill('Timothy');
|
||||||
await page.getByLabel('Lastname').fill('Apple');
|
await page.getByLabel('Last name').fill('Apple');
|
||||||
await page.getByLabel('Email').fill('timothy.apple@test.com');
|
await page.getByLabel('Email').fill('timothy.apple@test.com');
|
||||||
await page.getByLabel('Username').fill('timothy');
|
await page.getByLabel('Username').fill('timothy');
|
||||||
await page.getByRole('button', { name: 'Save' }).click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|||||||
@@ -52,7 +52,8 @@ test('Update application images', async ({ page }) => {
|
|||||||
await page.goto('/settings/admin/application-configuration');
|
await page.goto('/settings/admin/application-configuration');
|
||||||
|
|
||||||
await page.getByLabel('Favicon').setInputFiles('tests/assets/w3-schools-favicon.ico');
|
await page.getByLabel('Favicon').setInputFiles('tests/assets/w3-schools-favicon.ico');
|
||||||
await page.getByLabel('Logo').setInputFiles('tests/assets/pingvin-share-logo.png');
|
await page.getByLabel('Light Mode Logo').setInputFiles('tests/assets/pingvin-share-logo.png');
|
||||||
|
await page.getByLabel('Dark Mode Logo').setInputFiles('tests/assets/nextcloud-logo.png');
|
||||||
await page.getByLabel('Background Image').setInputFiles('tests/assets/clouds.jpg');
|
await page.getByLabel('Background Image').setInputFiles('tests/assets/clouds.jpg');
|
||||||
await page.getByRole('button', { name: 'Save' }).nth(1).click();
|
await page.getByRole('button', { name: 'Save' }).nth(1).click();
|
||||||
|
|
||||||
@@ -62,9 +63,11 @@ test('Update application images', async ({ page }) => {
|
|||||||
.get('/api/application-configuration/favicon')
|
.get('/api/application-configuration/favicon')
|
||||||
.then((res) => expect.soft(res.status()).toBe(200));
|
.then((res) => expect.soft(res.status()).toBe(200));
|
||||||
await page.request
|
await page.request
|
||||||
.get('/api/application-configuration/logo')
|
.get('/api/application-configuration/logo?light=true')
|
||||||
|
.then((res) => expect.soft(res.status()).toBe(200));
|
||||||
|
await page.request
|
||||||
|
.get('/api/application-configuration/logo?light=false')
|
||||||
.then((res) => expect.soft(res.status()).toBe(200));
|
.then((res) => expect.soft(res.status()).toBe(200));
|
||||||
|
|
||||||
await page.request
|
await page.request
|
||||||
.get('/api/application-configuration/background-image')
|
.get('/api/application-configuration/background-image')
|
||||||
.then((res) => expect.soft(res.status()).toBe(200));
|
.then((res) => expect.soft(res.status()).toBe(200));
|
||||||
|
|||||||
@@ -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();
|
||||||
|
});
|
||||||
@@ -10,8 +10,8 @@ test('Create user', async ({ page }) => {
|
|||||||
await page.goto('/settings/admin/users');
|
await page.goto('/settings/admin/users');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Add User' }).click();
|
await page.getByRole('button', { name: 'Add User' }).click();
|
||||||
await page.getByLabel('Firstname').fill(user.firstname);
|
await page.getByLabel('First name').fill(user.firstname);
|
||||||
await page.getByLabel('Lastname').fill(user.lastname);
|
await page.getByLabel('Last name').fill(user.lastname);
|
||||||
await page.getByLabel('Email').fill(user.email);
|
await page.getByLabel('Email').fill(user.email);
|
||||||
await page.getByLabel('Username').fill(user.username);
|
await page.getByLabel('Username').fill(user.username);
|
||||||
await page.getByRole('button', { name: 'Save' }).click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
@@ -26,8 +26,8 @@ test('Create user fails with already taken email', async ({ page }) => {
|
|||||||
await page.goto('/settings/admin/users');
|
await page.goto('/settings/admin/users');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Add User' }).click();
|
await page.getByRole('button', { name: 'Add User' }).click();
|
||||||
await page.getByLabel('Firstname').fill(user.firstname);
|
await page.getByLabel('First name').fill(user.firstname);
|
||||||
await page.getByLabel('Lastname').fill(user.lastname);
|
await page.getByLabel('Last name').fill(user.lastname);
|
||||||
await page.getByLabel('Email').fill(users.tim.email);
|
await page.getByLabel('Email').fill(users.tim.email);
|
||||||
await page.getByLabel('Username').fill(user.username);
|
await page.getByLabel('Username').fill(user.username);
|
||||||
await page.getByRole('button', { name: 'Save' }).click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
@@ -41,8 +41,8 @@ test('Create user fails with already taken username', async ({ page }) => {
|
|||||||
await page.goto('/settings/admin/users');
|
await page.goto('/settings/admin/users');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Add User' }).click();
|
await page.getByRole('button', { name: 'Add User' }).click();
|
||||||
await page.getByLabel('Firstname').fill(user.firstname);
|
await page.getByLabel('First name').fill(user.firstname);
|
||||||
await page.getByLabel('Lastname').fill(user.lastname);
|
await page.getByLabel('Last name').fill(user.lastname);
|
||||||
await page.getByLabel('Email').fill(user.email);
|
await page.getByLabel('Email').fill(user.email);
|
||||||
await page.getByLabel('Username').fill(users.tim.username);
|
await page.getByLabel('Username').fill(users.tim.username);
|
||||||
await page.getByRole('button', { name: 'Save' }).click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
@@ -91,8 +91,8 @@ test('Update user', async ({ page }) => {
|
|||||||
.click();
|
.click();
|
||||||
await page.getByRole('menuitem', { name: 'Edit' }).click();
|
await page.getByRole('menuitem', { name: 'Edit' }).click();
|
||||||
|
|
||||||
await page.getByLabel('Firstname').fill('Crack');
|
await page.getByLabel('First name').fill('Crack');
|
||||||
await page.getByLabel('Lastname').fill('Apple');
|
await page.getByLabel('Last name').fill('Apple');
|
||||||
await page.getByLabel('Email').fill('crack.apple@test.com');
|
await page.getByLabel('Email').fill('crack.apple@test.com');
|
||||||
await page.getByLabel('Username').fill('crack');
|
await page.getByLabel('Username').fill('crack');
|
||||||
await page.getByRole('button', { name: 'Save' }).click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
:80 {
|
:80 {
|
||||||
reverse_proxy /api/* http://localhost:8080
|
reverse_proxy /api/* http://localhost:{$BACKEND_PORT:8080}
|
||||||
reverse_proxy /.well-known/* http://localhost:8080
|
reverse_proxy /.well-known/* http://localhost:{$BACKEND_PORT:8080}
|
||||||
reverse_proxy /* http://localhost: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: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: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: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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
31
scripts/download-ip-database.sh
Normal file
31
scripts/download-ip-database.sh
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Check if the license key environment variable is set
|
||||||
|
if [ -z "$MAXMIND_LICENSE_KEY" ]; then
|
||||||
|
echo "Error: MAXMIND_LICENSE_KEY environment variable is not set."
|
||||||
|
echo "Please set it using 'export MAXMIND_LICENSE_KEY=your_license_key' and try again."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo $MAXMIND_LICENSE_KEY
|
||||||
|
# GeoLite2 City Database URL
|
||||||
|
URL="https://download.maxmind.com/app/geoip_download?edition_id=GeoLite2-City&license_key=${MAXMIND_LICENSE_KEY}&suffix=tar.gz"
|
||||||
|
|
||||||
|
# Download directory
|
||||||
|
DOWNLOAD_DIR="./geolite2_db"
|
||||||
|
TARGET_PATH=./backend/GeoLite2-City.mmdb
|
||||||
|
mkdir -p $DOWNLOAD_DIR
|
||||||
|
|
||||||
|
# Download the database
|
||||||
|
echo "Downloading GeoLite2 City database..."
|
||||||
|
curl -L -o "$DOWNLOAD_DIR/GeoLite2-City.tar.gz" "$URL"
|
||||||
|
|
||||||
|
# Extract the downloaded file
|
||||||
|
echo "Extracting GeoLite2 City database..."
|
||||||
|
tar -xzf "$DOWNLOAD_DIR/GeoLite2-City.tar.gz" -C $DOWNLOAD_DIR --strip-components=1
|
||||||
|
|
||||||
|
mv "$DOWNLOAD_DIR/GeoLite2-City.mmdb" $TARGET_PATH
|
||||||
|
|
||||||
|
# Clean up
|
||||||
|
rm -rf "$DOWNLOAD_DIR"
|
||||||
|
|
||||||
|
echo "GeoLite2 City database downloaded and extracted to $TARGET_PATH"
|
||||||
Reference in New Issue
Block a user