2024-10-31 13:07:53 +00:00
|
|
|
package grpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"git.solsynth.dev/hypernet/nexus/pkg/proto"
|
|
|
|
"git.solsynth.dev/hypernet/passport/pkg/authkit/models"
|
2024-12-08 12:21:40 +00:00
|
|
|
localCache "git.solsynth.dev/hypernet/passport/pkg/internal/cache"
|
2024-10-31 13:07:53 +00:00
|
|
|
"git.solsynth.dev/hypernet/passport/pkg/internal/database"
|
2024-12-08 12:21:40 +00:00
|
|
|
"git.solsynth.dev/hypernet/passport/pkg/internal/services"
|
|
|
|
"github.com/eko/gocache/lib/v4/cache"
|
|
|
|
"github.com/eko/gocache/lib/v4/marshaler"
|
2024-10-31 13:07:53 +00:00
|
|
|
"github.com/samber/lo"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2024-12-08 12:21:40 +00:00
|
|
|
"gorm.io/gorm"
|
2024-10-31 13:07:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func (v *App) GetUser(ctx context.Context, request *proto.GetUserRequest) (*proto.UserInfo, error) {
|
2024-12-08 12:21:40 +00:00
|
|
|
cacheManager := cache.New[any](localCache.S)
|
|
|
|
marshal := marshaler.New(cacheManager)
|
|
|
|
contx := context.Background()
|
|
|
|
|
|
|
|
var account models.Account
|
|
|
|
|
2024-11-02 17:47:50 +00:00
|
|
|
tx := database.C
|
2024-12-08 12:21:40 +00:00
|
|
|
hitCache := false
|
2024-11-02 17:47:50 +00:00
|
|
|
if request.UserId != nil {
|
2024-12-08 12:21:40 +00:00
|
|
|
if val, err := marshal.Get(contx, services.GetAccountCacheKey(request.GetUserId()), new(models.Account)); err == nil {
|
|
|
|
account = *val.(*models.Account)
|
|
|
|
hitCache = true
|
|
|
|
} else {
|
|
|
|
tx = tx.Where("id = ?", uint(request.GetUserId()))
|
|
|
|
}
|
2024-11-02 17:47:50 +00:00
|
|
|
}
|
|
|
|
if request.Name != nil {
|
2024-12-08 12:21:40 +00:00
|
|
|
if val, err := marshal.Get(contx, services.GetAccountCacheKey(request.GetName()), new(models.Account)); err == nil {
|
|
|
|
account = *val.(*models.Account)
|
|
|
|
hitCache = true
|
|
|
|
} else {
|
|
|
|
tx = tx.Where("name = ?", request.GetName())
|
|
|
|
}
|
2024-11-02 17:47:50 +00:00
|
|
|
}
|
|
|
|
|
2024-12-08 12:21:40 +00:00
|
|
|
if !hitCache {
|
|
|
|
if err := tx.
|
|
|
|
Preload("Profile").
|
|
|
|
Preload("Badges", func(db *gorm.DB) *gorm.DB {
|
|
|
|
return db.Order("badges.type DESC")
|
|
|
|
}).
|
|
|
|
First(&account).Error; err != nil {
|
|
|
|
return nil, status.Errorf(codes.NotFound, fmt.Sprintf("requested user with id %d was not found", request.GetUserId()))
|
|
|
|
}
|
|
|
|
|
|
|
|
groups, err := services.GetUserAccountGroup(account)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, fmt.Sprintf("unable to get account groups: %v", err))
|
|
|
|
}
|
|
|
|
for _, group := range groups {
|
|
|
|
for k, v := range group.PermNodes {
|
|
|
|
if _, ok := account.PermNodes[k]; !ok {
|
|
|
|
account.PermNodes[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
services.CacheAccount(account)
|
2024-10-31 13:07:53 +00:00
|
|
|
}
|
2024-12-08 12:21:40 +00:00
|
|
|
|
2024-10-31 13:07:53 +00:00
|
|
|
return account.EncodeToUserInfo(), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *App) ListUser(ctx context.Context, request *proto.ListUserRequest) (*proto.MultipleUserInfo, error) {
|
|
|
|
var accounts []models.Account
|
|
|
|
if err := database.C.
|
|
|
|
Where("id IN ?", lo.Map(request.GetUserId(), func(id uint64, _ int) interface{} { return id })).
|
|
|
|
Find(&accounts).Error; err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, fmt.Sprintf("failed to list users: %v", err))
|
|
|
|
}
|
|
|
|
return &proto.MultipleUserInfo{
|
|
|
|
Data: lo.Map(request.GetUserId(), func(item uint64, index int) *proto.UserInfo {
|
|
|
|
val, ok := lo.Find(accounts, func(x models.Account) bool {
|
|
|
|
return uint(item) == x.ID
|
|
|
|
})
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return val.EncodeToUserInfo()
|
|
|
|
}),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (v *App) ListUserRelative(ctx context.Context, request *proto.ListUserRelativeRequest) (*proto.ListUserRelativeResponse, error) {
|
2024-12-11 15:58:19 +00:00
|
|
|
tx := database.C.Preload("Account").Preload("Related").Where("status = ?", request.GetStatus())
|
2024-10-31 13:07:53 +00:00
|
|
|
|
|
|
|
if request.GetIsRelated() {
|
|
|
|
tx = tx.Where("related_id = ?", request.GetUserId())
|
|
|
|
} else {
|
|
|
|
tx = tx.Where("account_id = ?", request.GetUserId())
|
|
|
|
}
|
|
|
|
|
|
|
|
var data []models.AccountRelationship
|
|
|
|
if err := tx.Find(&data).Error; err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &proto.ListUserRelativeResponse{
|
|
|
|
Data: lo.Map(data, func(item models.AccountRelationship, index int) *proto.UserInfo {
|
2024-12-11 15:58:19 +00:00
|
|
|
account := lo.Ternary(request.GetIsRelated(), item.Account, item.Related)
|
2024-10-31 13:07:53 +00:00
|
|
|
val := &proto.UserInfo{
|
2024-12-11 15:58:19 +00:00
|
|
|
Id: uint64(account.ID),
|
|
|
|
Name: account.Name,
|
2024-10-31 13:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return val
|
|
|
|
}),
|
|
|
|
}, nil
|
|
|
|
}
|