Feat: finish object storage
This commit is contained in:
@@ -7,10 +7,13 @@ import (
|
|||||||
|
|
||||||
"gitea.konchin.com/go2025/backend/handlers/api"
|
"gitea.konchin.com/go2025/backend/handlers/api"
|
||||||
"gitea.konchin.com/go2025/backend/handlers/auth"
|
"gitea.konchin.com/go2025/backend/handlers/auth"
|
||||||
|
"gitea.konchin.com/go2025/backend/handlers/img"
|
||||||
"gitea.konchin.com/go2025/backend/implements"
|
"gitea.konchin.com/go2025/backend/implements"
|
||||||
"gitea.konchin.com/go2025/backend/middlewares"
|
"gitea.konchin.com/go2025/backend/middlewares"
|
||||||
"gitea.konchin.com/go2025/backend/tracing"
|
"gitea.konchin.com/go2025/backend/tracing"
|
||||||
"gitea.konchin.com/go2025/backend/utils"
|
"gitea.konchin.com/go2025/backend/utils"
|
||||||
|
"github.com/minio/minio-go/v7"
|
||||||
|
"github.com/minio/minio-go/v7/pkg/credentials"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
httpSwagger "github.com/swaggo/http-swagger"
|
httpSwagger "github.com/swaggo/http-swagger"
|
||||||
@@ -57,39 +60,38 @@ var serveCmd = &cobra.Command{
|
|||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
// Initialize MinIO instance
|
||||||
// Initialize MinIO instance
|
mc, err := minio.New(viper.GetString("minio-host"), &minio.Options{
|
||||||
mc, err := minio.New(viper.GetString("minio-host"), &minio.Options{
|
Creds: credentials.NewStaticV4(
|
||||||
Creds: credentials.NewStaticV4(
|
viper.GetString("minio-accesskey"),
|
||||||
viper.GetString("minio-accesskey"),
|
viper.GetString("minio-secretkey"),
|
||||||
viper.GetString("minio-secretkey"),
|
"",
|
||||||
"",
|
),
|
||||||
),
|
Secure: viper.GetBool("minio-usessl"),
|
||||||
Secure: viper.GetBool("minio-usessl"),
|
})
|
||||||
})
|
if err != nil {
|
||||||
if err != nil {
|
tracing.Logger.Ctx(ctx).
|
||||||
tracing.Logger.Ctx(ctx).
|
Error("failed to create minio client",
|
||||||
Error("failed to create minio client",
|
zap.Error(err))
|
||||||
zap.Error(err))
|
panic(err)
|
||||||
panic(err)
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if err := utils.InitMinIO(ctx, mc); err != nil {
|
if err := utils.InitMinIO(ctx, mc); err != nil {
|
||||||
tracing.Logger.Ctx(ctx).
|
tracing.Logger.Ctx(ctx).
|
||||||
Error("failed to minio init",
|
Error("failed to minio init",
|
||||||
zap.Error(err))
|
zap.Error(err))
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
// Initialize custom interfaces
|
// Initialize custom interfaces
|
||||||
db := implements.NewBunDatabase(bunDB)
|
db := implements.NewBunDatabase(bunDB)
|
||||||
// s3 := implements.NewMinIOObjectStorage(mc)
|
s3 := implements.NewMinIOObjectStorage(mc)
|
||||||
|
|
||||||
// Initialize handlers
|
// Initialize handlers
|
||||||
midHandlers := middlewares.NewHandlers(db)
|
midHandlers := middlewares.NewHandlers(db)
|
||||||
apis := api.NewHandlers(db)
|
apis := api.NewHandlers(db, s3)
|
||||||
auths := auth.NewHandlers(db)
|
auths := auth.NewHandlers(db)
|
||||||
|
imgs := img.NewHandlers(s3)
|
||||||
|
|
||||||
// Initialize backend router
|
// Initialize backend router
|
||||||
router := bunrouter.New()
|
router := bunrouter.New()
|
||||||
@@ -103,12 +105,17 @@ var serveCmd = &cobra.Command{
|
|||||||
Use(midHandlers.CheckRefreshToken).
|
Use(midHandlers.CheckRefreshToken).
|
||||||
Use(midHandlers.CheckAccessToken)
|
Use(midHandlers.CheckAccessToken)
|
||||||
apiGroup.GET("/images", apis.GetImages)
|
apiGroup.GET("/images", apis.GetImages)
|
||||||
|
apiGroup.GET("/aliases", apis.GetAliases)
|
||||||
|
apiGroup.PUT("/image/:id/aliases", apis.PutImageAliases)
|
||||||
|
|
||||||
authGroup := backend.NewGroup("/auth")
|
authGroup := backend.NewGroup("/auth")
|
||||||
authGroup.POST("/login", auths.PostLogin)
|
authGroup.POST("/login", auths.PostLogin)
|
||||||
authGroup.POST("/gen-login-url",
|
authGroup.POST("/gen-login-url",
|
||||||
midHandlers.CheckPresharedKey(auths.PostGenLoginUrl))
|
midHandlers.CheckPresharedKey(auths.PostGenLoginUrl))
|
||||||
|
|
||||||
|
imgGroup := backend.NewGroup("/img")
|
||||||
|
imgGroup.GET("/:filename", imgs.Get)
|
||||||
|
|
||||||
if viper.GetBool("swagger") {
|
if viper.GetBool("swagger") {
|
||||||
backend.GET("/swagger/*any",
|
backend.GET("/swagger/*any",
|
||||||
bunrouter.HTTPHandlerFunc(
|
bunrouter.HTTPHandlerFunc(
|
||||||
|
|||||||
16
docs/docs.go
16
docs/docs.go
@@ -143,10 +143,7 @@ const docTemplate = `{
|
|||||||
"in": "body",
|
"in": "body",
|
||||||
"required": true,
|
"required": true,
|
||||||
"schema": {
|
"schema": {
|
||||||
"type": "array",
|
"$ref": "#/definitions/api.putImageAliasesInput"
|
||||||
"items": {
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
@@ -285,6 +282,17 @@ const docTemplate = `{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"api.putImageAliasesInput": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"aliases": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"auth.postGenLoginUrlInput": {
|
"auth.postGenLoginUrlInput": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
|||||||
@@ -135,10 +135,7 @@
|
|||||||
"in": "body",
|
"in": "body",
|
||||||
"required": true,
|
"required": true,
|
||||||
"schema": {
|
"schema": {
|
||||||
"type": "array",
|
"$ref": "#/definitions/api.putImageAliasesInput"
|
||||||
"items": {
|
|
||||||
"type": "string"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
@@ -277,6 +274,17 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"api.putImageAliasesInput": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"aliases": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"auth.postGenLoginUrlInput": {
|
"auth.postGenLoginUrlInput": {
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
|
|||||||
@@ -20,6 +20,13 @@ definitions:
|
|||||||
uploadedUserId:
|
uploadedUserId:
|
||||||
type: string
|
type: string
|
||||||
type: object
|
type: object
|
||||||
|
api.putImageAliasesInput:
|
||||||
|
properties:
|
||||||
|
aliases:
|
||||||
|
items:
|
||||||
|
type: string
|
||||||
|
type: array
|
||||||
|
type: object
|
||||||
auth.postGenLoginUrlInput:
|
auth.postGenLoginUrlInput:
|
||||||
properties:
|
properties:
|
||||||
userId:
|
userId:
|
||||||
@@ -127,9 +134,7 @@ paths:
|
|||||||
name: payload
|
name: payload
|
||||||
required: true
|
required: true
|
||||||
schema:
|
schema:
|
||||||
items:
|
$ref: '#/definitions/api.putImageAliasesInput'
|
||||||
type: string
|
|
||||||
type: array
|
|
||||||
responses:
|
responses:
|
||||||
"200":
|
"200":
|
||||||
description: OK
|
description: OK
|
||||||
|
|||||||
@@ -29,8 +29,8 @@ func (self *Handlers) GetImages(
|
|||||||
) error {
|
) error {
|
||||||
ctx := req.Context()
|
ctx := req.Context()
|
||||||
|
|
||||||
rawReqImages := strings.Split(req.Param("images"), ",")
|
rawReqImages := strings.Split(req.URL.Query().Get("images"), ",")
|
||||||
rawReqAliases := strings.Split(req.Param("aliases"), ",")
|
rawReqAliases := strings.Split(req.URL.Query().Get("aliases"), ",")
|
||||||
|
|
||||||
if (len(rawReqImages) == 0 && len(rawReqAliases) == 0) ||
|
if (len(rawReqImages) == 0 && len(rawReqAliases) == 0) ||
|
||||||
(len(rawReqImages) > 0 && len(rawReqAliases) > 0) {
|
(len(rawReqImages) > 0 && len(rawReqAliases) > 0) {
|
||||||
|
|||||||
@@ -4,8 +4,15 @@ import "gitea.konchin.com/go2025/backend/interfaces"
|
|||||||
|
|
||||||
type Handlers struct {
|
type Handlers struct {
|
||||||
db interfaces.Database
|
db interfaces.Database
|
||||||
|
s3 interfaces.ObjectStorage
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHandlers(db interfaces.Database) *Handlers {
|
func NewHandlers(
|
||||||
return &Handlers{db: db}
|
db interfaces.Database,
|
||||||
|
s3 interfaces.ObjectStorage,
|
||||||
|
) *Handlers {
|
||||||
|
return &Handlers{
|
||||||
|
db: db,
|
||||||
|
s3: s3,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,11 @@ package api
|
|||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"gitea.konchin.com/go2025/backend/middlewares"
|
"gitea.konchin.com/go2025/backend/middlewares"
|
||||||
|
"gitea.konchin.com/go2025/backend/models"
|
||||||
|
"gitea.konchin.com/go2025/backend/types"
|
||||||
"gitea.konchin.com/go2025/backend/utils"
|
"gitea.konchin.com/go2025/backend/utils"
|
||||||
"github.com/uptrace/bunrouter"
|
"github.com/uptrace/bunrouter"
|
||||||
)
|
)
|
||||||
@@ -21,6 +24,16 @@ import (
|
|||||||
func (self *Handlers) PostImage(
|
func (self *Handlers) PostImage(
|
||||||
w http.ResponseWriter, req bunrouter.Request,
|
w http.ResponseWriter, req bunrouter.Request,
|
||||||
) error {
|
) error {
|
||||||
|
ctx := req.Context()
|
||||||
|
|
||||||
|
claim, ok := ctx.Value(types.AccessToken("")).(models.AccessTokenClaim)
|
||||||
|
if !ok {
|
||||||
|
return middlewares.HTTPError{
|
||||||
|
StatusCode: http.StatusUnauthorized,
|
||||||
|
Message: "missing access token",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
typeHeader := strings.Split(req.Header.Get("Content-Type"), "/")
|
typeHeader := strings.Split(req.Header.Get("Content-Type"), "/")
|
||||||
if len(typeHeader) != 2 || typeHeader[0] != "image" {
|
if len(typeHeader) != 2 || typeHeader[0] != "image" {
|
||||||
return middlewares.HTTPError{
|
return middlewares.HTTPError{
|
||||||
@@ -29,6 +42,28 @@ func (self *Handlers) PostImage(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO
|
image := models.Image{
|
||||||
|
Extension: typeHeader[1],
|
||||||
|
Uploader: claim.UserId,
|
||||||
|
UploadTS: time.Now(),
|
||||||
|
}
|
||||||
|
err := self.db.InsertImage(ctx, &image)
|
||||||
|
if err != nil {
|
||||||
|
return middlewares.HTTPError{
|
||||||
|
StatusCode: http.StatusInternalServerError,
|
||||||
|
Message: "failed to insert image to db",
|
||||||
|
OriginError: err,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
err = self.s3.PutImage(ctx,
|
||||||
|
image.Filename(), req.Body, req.ContentLength)
|
||||||
|
if err != nil {
|
||||||
|
return middlewares.HTTPError{
|
||||||
|
StatusCode: http.StatusInternalServerError,
|
||||||
|
Message: "failed to put image to s3",
|
||||||
|
OriginError: err,
|
||||||
|
}
|
||||||
|
}
|
||||||
return utils.Success(w)
|
return utils.Success(w)
|
||||||
}
|
}
|
||||||
|
|||||||
35
handlers/img/get.go
Normal file
35
handlers/img/get.go
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
package img
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"gitea.konchin.com/go2025/backend/middlewares"
|
||||||
|
"github.com/uptrace/bunrouter"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (self *Handlers) Get(
|
||||||
|
w http.ResponseWriter,
|
||||||
|
req bunrouter.Request,
|
||||||
|
) error {
|
||||||
|
ctx := req.Context()
|
||||||
|
|
||||||
|
filename := req.Param("filename")
|
||||||
|
|
||||||
|
r, err := self.s3.GetImage(ctx, filename)
|
||||||
|
if err != nil {
|
||||||
|
return middlewares.HTTPError{
|
||||||
|
StatusCode: http.StatusInternalServerError,
|
||||||
|
Message: "failed to get image",
|
||||||
|
OriginError: err,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if _, err := io.Copy(w, r); err != nil {
|
||||||
|
return middlewares.HTTPError{
|
||||||
|
StatusCode: http.StatusInternalServerError,
|
||||||
|
Message: "failed to copy buffer",
|
||||||
|
OriginError: err,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
11
handlers/img/handlers.go
Normal file
11
handlers/img/handlers.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package img
|
||||||
|
|
||||||
|
import "gitea.konchin.com/go2025/backend/interfaces"
|
||||||
|
|
||||||
|
type Handlers struct {
|
||||||
|
s3 interfaces.ObjectStorage
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHandlers(s3 interfaces.ObjectStorage) *Handlers {
|
||||||
|
return &Handlers{s3: s3}
|
||||||
|
}
|
||||||
@@ -184,3 +184,13 @@ func (self *BunDatabase) UpdateAliases(
|
|||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (self *BunDatabase) InsertImage(
|
||||||
|
ctx context.Context,
|
||||||
|
image *models.Image,
|
||||||
|
) error {
|
||||||
|
_, err := self.db.NewInsert().
|
||||||
|
Model(image).
|
||||||
|
Exec(ctx)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,4 +1,36 @@
|
|||||||
package implements
|
package implements
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/minio/minio-go/v7"
|
||||||
|
"github.com/spf13/viper"
|
||||||
|
)
|
||||||
|
|
||||||
type MinIOObjectStorage struct {
|
type MinIOObjectStorage struct {
|
||||||
|
mc *minio.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMinIOObjectStorage(mc *minio.Client) *MinIOObjectStorage {
|
||||||
|
return &MinIOObjectStorage{mc: mc}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *MinIOObjectStorage) PutImage(
|
||||||
|
ctx context.Context,
|
||||||
|
name string,
|
||||||
|
r io.Reader,
|
||||||
|
size int64,
|
||||||
|
) error {
|
||||||
|
_, err := self.mc.PutObject(ctx,
|
||||||
|
viper.GetString("minio-bucket"), name, r, size, minio.PutObjectOptions{})
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (self *MinIOObjectStorage) GetImage(
|
||||||
|
ctx context.Context,
|
||||||
|
name string,
|
||||||
|
) (io.Reader, error) {
|
||||||
|
return self.mc.GetObject(ctx,
|
||||||
|
viper.GetString("minio-bucket"), name, minio.GetObjectOptions{})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,4 +37,9 @@ type Database interface {
|
|||||||
imageId int64,
|
imageId int64,
|
||||||
aliasNames []string,
|
aliasNames []string,
|
||||||
) error
|
) error
|
||||||
|
|
||||||
|
InsertImage(
|
||||||
|
ctx context.Context,
|
||||||
|
image *models.Image,
|
||||||
|
) error
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,19 @@
|
|||||||
package interfaces
|
package interfaces
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
type ObjectStorage interface {
|
type ObjectStorage interface {
|
||||||
|
PutImage(
|
||||||
|
ctx context.Context,
|
||||||
|
name string,
|
||||||
|
r io.Reader,
|
||||||
|
size int64,
|
||||||
|
) error
|
||||||
|
GetImage(
|
||||||
|
ctx context.Context,
|
||||||
|
name string,
|
||||||
|
) (io.Reader, error)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -79,7 +79,7 @@ func (self *Handlers) CheckAccessToken(
|
|||||||
var claim models.AccessTokenClaim
|
var claim models.AccessTokenClaim
|
||||||
token, err := jwt.ParseWithClaims(accessTokenString, &claim,
|
token, err := jwt.ParseWithClaims(accessTokenString, &claim,
|
||||||
func(*jwt.Token) (interface{}, error) {
|
func(*jwt.Token) (interface{}, error) {
|
||||||
return []byte(viper.GetString("ACCESS_TOKEN_SECRET")), nil
|
return []byte(viper.GetString("access-token-secret")), nil
|
||||||
})
|
})
|
||||||
if err != nil || !token.Valid {
|
if err != nil || !token.Valid {
|
||||||
accessTokenString, err = refreshAccessToken(ctx, self.db, w, req)
|
accessTokenString, err = refreshAccessToken(ctx, self.db, w, req)
|
||||||
@@ -92,7 +92,7 @@ func (self *Handlers) CheckAccessToken(
|
|||||||
}
|
}
|
||||||
token, err := jwt.ParseWithClaims(accessTokenString, &claim,
|
token, err := jwt.ParseWithClaims(accessTokenString, &claim,
|
||||||
func(*jwt.Token) (interface{}, error) {
|
func(*jwt.Token) (interface{}, error) {
|
||||||
return []byte(viper.GetString("ACCESS_TOKEN_SECRET")), nil
|
return []byte(viper.GetString("access-token-secret")), nil
|
||||||
})
|
})
|
||||||
if err != nil || !token.Valid {
|
if err != nil || !token.Valid {
|
||||||
return HTTPError{
|
return HTTPError{
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ func (self *Handlers) CheckRefreshToken(
|
|||||||
var claim models.RefreshTokenClaim
|
var claim models.RefreshTokenClaim
|
||||||
token, err := jwt.ParseWithClaims(refreshTokenCookie.Value, &claim,
|
token, err := jwt.ParseWithClaims(refreshTokenCookie.Value, &claim,
|
||||||
func(*jwt.Token) (interface{}, error) {
|
func(*jwt.Token) (interface{}, error) {
|
||||||
return []byte(viper.GetString("REFRESH_TOKEN_SECRET")), nil
|
return []byte(viper.GetString("refresh-token-secret")), nil
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return HTTPError{
|
return HTTPError{
|
||||||
@@ -48,7 +48,7 @@ func (self *Handlers) CheckRefreshToken(
|
|||||||
|
|
||||||
// check time and refresh
|
// check time and refresh
|
||||||
timeLeft := claim.ExpiresAt.Time.Sub(time.Now()) / time.Second
|
timeLeft := claim.ExpiresAt.Time.Sub(time.Now()) / time.Second
|
||||||
if int64(timeLeft) < viper.GetInt64("REFRESH_TOKEN_TIMEOUT")/2 {
|
if int64(timeLeft) < viper.GetInt64("refresh-token-timeout")/2 {
|
||||||
session, err := self.db.UpdateRefreshToken(ctx, claim.UserId)
|
session, err := self.db.UpdateRefreshToken(ctx, claim.UserId)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return HTTPError{
|
return HTTPError{
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package models
|
package models
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/uptrace/bun"
|
"github.com/uptrace/bun"
|
||||||
@@ -9,9 +10,14 @@ import (
|
|||||||
type Image struct {
|
type Image struct {
|
||||||
bun.BaseModel `bun:"table:image"`
|
bun.BaseModel `bun:"table:image"`
|
||||||
|
|
||||||
Id int64 `bun:"id,pk,autoincrement"`
|
Id int64 `bun:"id,pk,autoincrement"`
|
||||||
Uploader string `bun:"uploader"`
|
Extension string `bun:"extension"`
|
||||||
UploadTS time.Time `bun:"upload_timestamp"`
|
Uploader string `bun:"uploader"`
|
||||||
|
UploadTS time.Time `bun:"upload_timestamp"`
|
||||||
|
|
||||||
Aliases []Alias `bun:"m2m:alias_image,join:Image=Alias"`
|
Aliases []Alias `bun:"m2m:alias_image,join:Image=Alias"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (self *Image) Filename() string {
|
||||||
|
return fmt.Sprintf("%d.%s", self.Id, self.Extension)
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user