mirror of https://github.com/go-sonic/sonic.git
feat: add manage apis for ApplicationPassword
parent
56157ba32a
commit
b77bcce0c6
@ -0,0 +1,51 @@
|
||||
package wp
|
||||
|
||||
import (
|
||||
"github.com/gin-gonic/gin"
|
||||
"github.com/go-sonic/sonic/model/dto/wp"
|
||||
"github.com/go-sonic/sonic/model/entity"
|
||||
"github.com/go-sonic/sonic/model/param"
|
||||
"github.com/go-sonic/sonic/service"
|
||||
)
|
||||
|
||||
type CategoryHandler struct {
|
||||
CategoryService service.CategoryService
|
||||
}
|
||||
|
||||
func NewCategoryHandler(categoryService service.CategoryService) *CategoryHandler {
|
||||
return &CategoryHandler{
|
||||
CategoryService: categoryService,
|
||||
}
|
||||
}
|
||||
|
||||
func (c *CategoryHandler) List(ctx *gin.Context) (interface{}, error) {
|
||||
sort := ¶m.Sort{
|
||||
Fields: []string{"name,desc"},
|
||||
}
|
||||
categoryEntities, err := c.CategoryService.ListAll(ctx, sort)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
categoryDTOList := make([]*wp.CategoryDTO, len(categoryEntities))
|
||||
for _, categoryEntity := range categoryEntities {
|
||||
categoryDTOList = append(categoryDTOList, convertToCategoryDTO(categoryEntity))
|
||||
}
|
||||
|
||||
return categoryDTOList, nil
|
||||
}
|
||||
|
||||
func convertToCategoryDTO(categoryEntity *entity.Category) *wp.CategoryDTO {
|
||||
categoryDTO := &wp.CategoryDTO{
|
||||
ID: categoryEntity.ID,
|
||||
Count: 0,
|
||||
Description: categoryEntity.Description,
|
||||
Link: "",
|
||||
Name: categoryEntity.Name,
|
||||
Slug: categoryEntity.Slug,
|
||||
Taxonomy: "",
|
||||
Parent: categoryEntity.ParentID,
|
||||
Meta: nil,
|
||||
}
|
||||
return categoryDTO
|
||||
}
|
@ -0,0 +1,11 @@
|
||||
package wp
|
||||
|
||||
import "github.com/go-sonic/sonic/injection"
|
||||
|
||||
func init() {
|
||||
injection.Provide(
|
||||
NewPostHandler,
|
||||
NewUserHandler,
|
||||
NewCategoryHandler,
|
||||
)
|
||||
}
|
@ -0,0 +1,128 @@
|
||||
package wp
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"github.com/gin-gonic/gin"
|
||||
"github.com/go-sonic/sonic/consts"
|
||||
"github.com/go-sonic/sonic/log"
|
||||
"github.com/go-sonic/sonic/model/dto/wp"
|
||||
"github.com/go-sonic/sonic/model/entity"
|
||||
"github.com/go-sonic/sonic/model/param"
|
||||
"github.com/go-sonic/sonic/service"
|
||||
"github.com/go-sonic/sonic/util"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
type PostHandler struct {
|
||||
PostService service.PostService
|
||||
}
|
||||
|
||||
func NewPostHandler(postService service.PostService) *PostHandler {
|
||||
return &PostHandler{
|
||||
PostService: postService,
|
||||
}
|
||||
}
|
||||
|
||||
func (handler *PostHandler) Create(ctx *gin.Context) (interface{}, error) {
|
||||
var wpPost param.WpPost
|
||||
err := ctx.ShouldBindJSON(&wpPost)
|
||||
if err != nil {
|
||||
return nil, util.WrapJsonBindErr(err)
|
||||
}
|
||||
|
||||
bytes, err := json.Marshal(wpPost)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
log.CtxInfo(ctx, "wpPost: "+string(bytes))
|
||||
|
||||
postParam := convertToPostParam(&wpPost)
|
||||
|
||||
create, err := handler.PostService.Create(ctx, postParam)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return convertToWpPost(create), nil
|
||||
}
|
||||
|
||||
func convertToPostParam(wpPost *param.WpPost) *param.Post {
|
||||
var paramPostStatus = consts.PostStatusPublished
|
||||
if strings.ToLower(wpPost.Content) == "draft" {
|
||||
paramPostStatus = consts.PostStatusDraft
|
||||
}
|
||||
|
||||
createTime := time.Now().Unix()
|
||||
|
||||
return ¶m.Post{
|
||||
Title: wpPost.Title,
|
||||
Status: paramPostStatus,
|
||||
Slug: wpPost.Slug,
|
||||
EditorType: nil,
|
||||
OriginalContent: wpPost.Content,
|
||||
Summary: "",
|
||||
Thumbnail: "",
|
||||
DisallowComment: false,
|
||||
Password: wpPost.Password,
|
||||
Template: "",
|
||||
TopPriority: 0,
|
||||
CreateTime: &createTime,
|
||||
MetaKeywords: "",
|
||||
MetaDescription: "",
|
||||
TagIDs: make([]int32, 0),
|
||||
CategoryIDs: make([]int32, 0),
|
||||
MetaParam: nil,
|
||||
Content: wpPost.Content,
|
||||
EditTime: nil,
|
||||
UpdateTime: nil,
|
||||
}
|
||||
}
|
||||
|
||||
func convertToWpPost(postEntity *entity.Post) *wp.PostDTO {
|
||||
timeFormat := time.RFC3339
|
||||
var wpStatus = "publish"
|
||||
var wpCommentStatus = "open"
|
||||
var wpContent = make(map[string]interface{})
|
||||
|
||||
if postEntity.Status == consts.PostStatusDraft {
|
||||
wpStatus = "draft"
|
||||
}
|
||||
|
||||
if postEntity.DisallowComment {
|
||||
wpCommentStatus = "close"
|
||||
}
|
||||
|
||||
wpContent["rendered"] = postEntity.OriginalContent
|
||||
wpContent["protected"] = false
|
||||
|
||||
var postDTO = &wp.PostDTO{
|
||||
Date: postEntity.CreateTime.Format(timeFormat),
|
||||
DateGmt: postEntity.CreateTime.UTC().Format(timeFormat),
|
||||
Guid: nil,
|
||||
Id: postEntity.ID,
|
||||
Link: "",
|
||||
Modified: postEntity.UpdateTime.Format(timeFormat),
|
||||
ModifiedGmt: postEntity.UpdateTime.UTC().Format(timeFormat),
|
||||
Slug: "",
|
||||
Status: wpStatus,
|
||||
Type: "post",
|
||||
Password: "standard",
|
||||
PermalinkTemplate: "",
|
||||
GeneratedSlug: "",
|
||||
Title: "",
|
||||
Content: wpContent,
|
||||
Author: 0,
|
||||
Excerpt: nil,
|
||||
FeaturedMedia: 0,
|
||||
CommentStatus: wpCommentStatus,
|
||||
PingStatus: "open",
|
||||
Format: "standard",
|
||||
Meta: nil,
|
||||
Sticky: false,
|
||||
Template: "",
|
||||
Categories: make([]int32, 0),
|
||||
Tags: make([]int32, 0),
|
||||
}
|
||||
return postDTO
|
||||
}
|
@ -0,0 +1,31 @@
|
||||
package wp
|
||||
|
||||
import (
|
||||
"github.com/gin-gonic/gin"
|
||||
"github.com/go-sonic/sonic/model/dto"
|
||||
"github.com/go-sonic/sonic/service"
|
||||
)
|
||||
|
||||
type UserHandler struct {
|
||||
UserService service.UserService
|
||||
}
|
||||
|
||||
func NewUserHandler(userService service.UserService) *UserHandler {
|
||||
return &UserHandler{
|
||||
UserService: userService,
|
||||
}
|
||||
}
|
||||
|
||||
func (u *UserHandler) List(ctx *gin.Context) (interface{}, error) {
|
||||
allUser, err := u.UserService.GetAllUser(ctx)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
userDTOList := make([]*dto.User, len(allUser))
|
||||
for _, user := range allUser {
|
||||
userDTO := u.UserService.ConvertToDTO(ctx, user)
|
||||
userDTOList = append(userDTOList, userDTO)
|
||||
}
|
||||
return userDTOList, nil
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
package middleware
|
||||
|
||||
import (
|
||||
"encoding/base64"
|
||||
"fmt"
|
||||
"github.com/gin-gonic/gin"
|
||||
"github.com/go-sonic/sonic/consts"
|
||||
"github.com/go-sonic/sonic/model/dto"
|
||||
"github.com/go-sonic/sonic/service"
|
||||
"net/http"
|
||||
"regexp"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type ApplicationPasswordMiddleware struct {
|
||||
PasswordService service.ApplicationPasswordService
|
||||
UserService service.UserService
|
||||
}
|
||||
|
||||
func NewApplicationPasswordMiddleware(passwordService service.ApplicationPasswordService, userService service.UserService) *ApplicationPasswordMiddleware {
|
||||
m := &ApplicationPasswordMiddleware{
|
||||
PasswordService: passwordService,
|
||||
UserService: userService,
|
||||
}
|
||||
return m
|
||||
}
|
||||
|
||||
func (a *ApplicationPasswordMiddleware) Get() error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func (a *ApplicationPasswordMiddleware) GetWrapHandler() gin.HandlerFunc {
|
||||
|
||||
return func(ctx *gin.Context) {
|
||||
header := ctx.GetHeader("Authorization")
|
||||
if len(header) == 0 {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
match := verifyHeader(header)
|
||||
if !match {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
bytes, err := base64.StdEncoding.DecodeString(header[6:])
|
||||
if err != nil {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
userPass := string(bytes)
|
||||
|
||||
if !strings.Contains(userPass, ":") {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
splits := strings.SplitN(userPass, ":", 2)
|
||||
|
||||
userEntity, err := a.UserService.GetByUsername(ctx, splits[0])
|
||||
if err != nil {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
pwdEntity, err := a.PasswordService.Verify(ctx, userEntity.ID, splits[1])
|
||||
if err != nil || pwdEntity == nil {
|
||||
abortUnauthorized(ctx)
|
||||
return
|
||||
}
|
||||
|
||||
err = a.PasswordService.Update(ctx, *pwdEntity.ID, ctx.ClientIP())
|
||||
if err != nil {
|
||||
ctx.AbortWithStatusJSON(http.StatusInternalServerError, &dto.BaseDTO{
|
||||
Status: http.StatusInternalServerError,
|
||||
Message: fmt.Sprintf("Update application password entity error, err=%s", err),
|
||||
})
|
||||
return
|
||||
}
|
||||
ctx.Set(consts.AuthorizedUser, userEntity)
|
||||
}
|
||||
}
|
||||
|
||||
func abortUnauthorized(ctx *gin.Context) {
|
||||
ctx.AbortWithStatusJSON(http.StatusUnauthorized, &dto.BaseDTO{
|
||||
Status: http.StatusUnauthorized,
|
||||
Message: "Unauthorized",
|
||||
})
|
||||
}
|
||||
|
||||
func verifyHeader(header string) bool {
|
||||
compile, err := regexp.Compile("^Basic [a-z\\d/+]*={0,2}")
|
||||
return err != nil || compile.MatchString(header)
|
||||
}
|
@ -0,0 +1,13 @@
|
||||
package wp
|
||||
|
||||
type CategoryDTO struct {
|
||||
ID int32 `json:"id"`
|
||||
Count int32 `json:"count"`
|
||||
Description string `json:"description"`
|
||||
Link string `json:"link"`
|
||||
Name string `json:"name"`
|
||||
Slug string `json:"slug"`
|
||||
Taxonomy string `json:"taxonomy"`
|
||||
Parent int32 `json:"parent"`
|
||||
Meta map[string]interface{} `json:"meta"`
|
||||
}
|
@ -0,0 +1,30 @@
|
||||
package wp
|
||||
|
||||
type PostDTO struct {
|
||||
Date string `json:"date"`
|
||||
DateGmt string `json:"date_gmt"`
|
||||
Guid map[string]interface{} `json:"guid"`
|
||||
Id int32 `json:"id"`
|
||||
Link string `json:"link"`
|
||||
Modified string `json:"modified"`
|
||||
ModifiedGmt string `json:"modified_gmt"`
|
||||
Slug string `json:"slug"`
|
||||
Status string `json:"status"`
|
||||
Type string `json:"type"`
|
||||
Password string `json:"password"`
|
||||
PermalinkTemplate string `json:"permalink_template"`
|
||||
GeneratedSlug string `json:"generated_slug"`
|
||||
Title string `json:"title"`
|
||||
Content map[string]interface{} `json:"content"`
|
||||
Author int32 `json:"author"`
|
||||
Excerpt map[string]interface{} `json:"excerpt"`
|
||||
FeaturedMedia int32 `json:"featured_media"`
|
||||
CommentStatus string `json:"comment_status"`
|
||||
PingStatus string `json:"ping_status"`
|
||||
Format string `json:"format"`
|
||||
Meta map[string]interface{} `json:"meta"`
|
||||
Sticky bool `json:"sticky"`
|
||||
Template string `json:"template"`
|
||||
Categories []int32 `json:"categories"`
|
||||
Tags []int32 `json:"tags"`
|
||||
}
|
@ -0,0 +1,4 @@
|
||||
package wp
|
||||
|
||||
type UserDTO struct {
|
||||
}
|
@ -0,0 +1,30 @@
|
||||
package param
|
||||
|
||||
type WpPost struct {
|
||||
Date string `json:"date"`
|
||||
DateGmt string `json:"date_gmt"`
|
||||
Guid map[string]interface{} `json:"guid"`
|
||||
Id int32 `json:"id"`
|
||||
Link string `json:"link"`
|
||||
Modified string `json:"modified"`
|
||||
ModifiedGmt string `json:"modified_gmt"`
|
||||
Slug string `json:"slug"`
|
||||
Status string `json:"status"`
|
||||
Type string `json:"type"`
|
||||
Password string `json:"password"`
|
||||
PermalinkTemplate string `json:"permalink_template"`
|
||||
GeneratedSlug string `json:"generated_slug"`
|
||||
Title string `json:"title"`
|
||||
Content string `json:"content"`
|
||||
Author int32 `json:"author"`
|
||||
Excerpt map[string]interface{} `json:"excerpt"`
|
||||
FeaturedMedia int32 `json:"featured_media"`
|
||||
CommentStatus string `json:"comment_status"`
|
||||
PingStatus string `json:"ping_status"`
|
||||
Format string `json:"format"`
|
||||
Meta map[string]interface{} `json:"meta"`
|
||||
Sticky bool `json:"sticky"`
|
||||
Template string `json:"template"`
|
||||
Categories []int32 `json:"categories"`
|
||||
Tags []int32 `json:"tags"`
|
||||
}
|
Loading…
Reference in New Issue