wp-go/actions/common/common.go

269 lines
7.9 KiB
Go
Raw Normal View History

2022-09-18 04:34:48 +00:00
package common
import (
2022-09-20 04:00:09 +00:00
"context"
2022-09-20 08:55:13 +00:00
"database/sql"
2022-09-18 04:34:48 +00:00
"fmt"
2022-09-19 09:39:00 +00:00
"github/fthvgb1/wp-go/cache"
2022-09-18 04:34:48 +00:00
"github/fthvgb1/wp-go/models"
2022-09-19 11:11:36 +00:00
"github/fthvgb1/wp-go/vars"
2022-09-19 12:15:10 +00:00
"log"
2022-09-18 14:06:27 +00:00
"strings"
"sync"
2022-09-19 09:39:00 +00:00
"time"
2022-09-18 04:34:48 +00:00
)
2022-09-18 14:06:27 +00:00
var PostsCache sync.Map
2022-09-19 11:11:36 +00:00
var PostContextCache sync.Map
2022-09-18 14:06:27 +00:00
2022-09-19 12:34:20 +00:00
var archivesCaches *Arch
2022-09-19 11:11:36 +00:00
var categoryCaches *cache.SliceCache[models.WpTermsMy]
var recentPostsCaches *cache.SliceCache[models.WpPosts]
2022-09-20 08:55:13 +00:00
var monthCaches *cache.MapCache[string, []models.WpPosts]
2022-09-20 13:16:51 +00:00
var recentCommentsCaches *cache.SliceCache[models.WpComments]
2022-09-19 11:11:36 +00:00
func InitCache() {
2022-09-19 12:34:20 +00:00
archivesCaches = &Arch{
2022-09-19 12:15:10 +00:00
mutex: &sync.Mutex{},
setCacheFunc: archives,
}
2022-09-19 11:11:36 +00:00
categoryCaches = cache.NewSliceCache[models.WpTermsMy](categories, vars.Conf.CategoryCacheTime)
recentPostsCaches = cache.NewSliceCache[models.WpPosts](recentPosts, vars.Conf.RecentPostCacheTime)
2022-09-20 08:55:13 +00:00
monthCaches = cache.NewMapCache[string, []models.WpPosts](getMonthPost, 30*time.Minute)
2022-09-20 13:16:51 +00:00
recentCommentsCaches = cache.NewSliceCache[models.WpComments](recentComments, vars.Conf.RecentCommentsCacheTime)
2022-09-19 11:11:36 +00:00
}
2022-09-19 12:34:20 +00:00
type Arch struct {
data []models.PostArchive
2022-09-19 12:15:10 +00:00
mutex *sync.Mutex
2022-09-19 12:34:20 +00:00
setCacheFunc func() ([]models.PostArchive, error)
2022-09-19 12:15:10 +00:00
month time.Month
}
2022-09-19 12:34:20 +00:00
func (c *Arch) GetCache() []models.PostArchive {
2022-09-19 12:15:10 +00:00
l := len(c.data)
m := time.Now().Month()
if l > 0 && c.month != m || l < 1 {
r, err := c.setCacheFunc()
if err != nil {
log.Printf("set cache err[%s]", err)
return nil
}
c.mutex.Lock()
defer c.mutex.Unlock()
c.month = m
c.data = r
}
return c.data
}
2022-09-19 11:11:36 +00:00
type PostContext struct {
Prev models.WpPosts
Next models.WpPosts
expireTime time.Duration
setTime time.Time
}
2022-09-20 08:55:13 +00:00
func GetMonthPost(ctx context.Context, year, month string) ([]models.WpPosts, error) {
return monthCaches.GetCache(ctx, fmt.Sprintf("%s%s", year, month), time.Second, year, month)
}
2022-09-20 13:16:51 +00:00
func RecentComments(ctx context.Context) (r []models.WpComments) {
r, _ = recentCommentsCaches.GetCache(ctx, time.Second)
return
}
func recentComments(...any) (r []models.WpComments, err error) {
return models.Find[models.WpComments](models.SqlBuilder{
{"comment_approved", "1"},
{"post_status", "publish"},
}, "comment_ID,comment_author,comment_post_ID,post_title", "", models.SqlBuilder{{"comment_date_gmt", "desc"}}, models.SqlBuilder{
{"a", "left join", "wp_posts b", "a.comment_post_ID=b.ID"},
}, 5)
}
2022-09-20 08:55:13 +00:00
func getMonthPost(args ...any) ([]models.WpPosts, error) {
y := args[0].(string)
m := args[1].(string)
where := models.SqlBuilder{
{"post_type", "in", ""},
{"post_status", "in", ""},
{"month(post_date)", m},
{"year(post_date)", y},
}
return models.Find[models.WpPosts](where, "ID", "",
models.SqlBuilder{{"post_date", "asc"}},
nil, 0, []interface{}{"post"}, []interface{}{"publish"},
)
}
2022-09-19 11:11:36 +00:00
func GetContextPost(id uint64, t time.Time) (prev, next models.WpPosts, err error) {
post, ok := PostContextCache.Load(id)
if ok {
c := post.(PostContext)
isExp := c.expireTime/time.Second+time.Duration(c.setTime.Unix()) < time.Duration(time.Now().Unix())
if !isExp && (c.Prev.Id > 0 || c.Next.Id > 0) {
return c.Prev, c.Next, nil
}
}
prev, next, err = getPostContext(t)
post = PostContext{
Prev: prev,
Next: next,
expireTime: vars.Conf.ContextPostCacheTime,
setTime: time.Now(),
}
PostContextCache.Store(id, post)
return
}
func getPostContext(t time.Time) (prev, next models.WpPosts, err error) {
next, err = models.FirstOne[models.WpPosts](models.SqlBuilder{
{"post_date", ">", t.Format("2006-01-02 15:04:05")},
2022-09-19 12:24:14 +00:00
{"post_status", "in", ""},
2022-09-19 11:11:36 +00:00
{"post_type", "post"},
2022-09-19 12:24:14 +00:00
}, "ID,post_title,post_password", nil, []interface{}{"publish", "private"})
2022-09-19 11:11:36 +00:00
prev, err = models.FirstOne[models.WpPosts](models.SqlBuilder{
{"post_date", "<", t.Format("2006-01-02 15:04:05")},
2022-09-19 12:24:14 +00:00
{"post_status", "in", ""},
2022-09-19 11:11:36 +00:00
{"post_type", "post"},
2022-09-19 12:24:14 +00:00
}, "ID,post_title", models.SqlBuilder{{"post_date", "desc"}}, []interface{}{"publish", "private"})
2022-09-20 08:55:13 +00:00
if err == sql.ErrNoRows {
err = nil
}
2022-09-19 11:11:36 +00:00
return
}
2022-09-19 09:39:00 +00:00
2022-09-18 14:06:27 +00:00
func GetPostFromCache(Id uint64) (r models.WpPosts) {
p, ok := PostsCache.Load(Id)
if ok {
r = *p.(*models.WpPosts)
}
return
}
func QueryAndSetPostCache(postIds []models.WpPosts) (err error) {
var all []uint64
var needQuery []interface{}
for _, wpPosts := range postIds {
all = append(all, wpPosts.Id)
if _, ok := PostsCache.Load(wpPosts.Id); !ok {
needQuery = append(needQuery, wpPosts.Id)
}
}
if len(needQuery) > 0 {
rawPosts, er := models.Find[models.WpPosts](models.SqlBuilder{{
"Id", "in", "",
}}, "a.*,ifnull(d.name,'') category_name,ifnull(taxonomy,'') `taxonomy`", "", nil, models.SqlBuilder{{
"a", "left join", "wp_term_relationships b", "a.Id=b.object_id",
}, {
"left join", "wp_term_taxonomy c", "b.term_taxonomy_id=c.term_taxonomy_id",
}, {
"left join", "wp_terms d", "c.term_id=d.term_id",
}}, 0, needQuery)
if er != nil {
err = er
return
}
postsMap := make(map[uint64]*models.WpPosts)
for i, post := range rawPosts {
v, ok := postsMap[post.Id]
if !ok {
v = &rawPosts[i]
}
if post.Taxonomy == "category" {
v.Categories = append(v.Categories, post.CategoryName)
} else if post.Taxonomy == "post_tag" {
v.Tags = append(v.Tags, post.CategoryName)
}
postsMap[post.Id] = v
}
for _, pp := range postsMap {
if len(pp.Categories) > 0 {
t := make([]string, 0, len(pp.Categories))
for _, cat := range pp.Categories {
t = append(t, fmt.Sprintf(`<a href="/p/category/%s" rel="category tag">%s</a>`, cat, cat))
}
pp.CategoriesHtml = strings.Join(t, "、")
}
if len(pp.Tags) > 0 {
t := make([]string, 0, len(pp.Tags))
for _, cat := range pp.Tags {
t = append(t, fmt.Sprintf(`<a href="/p/tag/%s" rel="tag">%s</a>`, cat, cat))
}
pp.TagsHtml = strings.Join(t, "、")
}
PostsCache.Store(pp.Id, pp)
}
}
return
}
2022-09-19 09:39:00 +00:00
func archives() ([]models.PostArchive, error) {
return models.Find[models.PostArchive](models.SqlBuilder{
2022-09-18 04:34:48 +00:00
{"post_type", "post"}, {"post_status", "publish"},
}, "YEAR(post_date) AS `year`, MONTH(post_date) AS `month`, count(ID) as posts", "year,month", models.SqlBuilder{{"year", "desc"}, {"month", "desc"}}, nil, 0)
}
2022-09-19 09:39:00 +00:00
func Archives() (r []models.PostArchive) {
return archivesCaches.GetCache()
}
2022-09-20 04:00:09 +00:00
func Categories(ctx context.Context) []models.WpTermsMy {
2022-09-20 08:11:20 +00:00
r, _ := categoryCaches.GetCache(ctx, time.Second)
return r
2022-09-19 09:39:00 +00:00
}
2022-09-20 04:00:09 +00:00
func categories(...any) (terms []models.WpTermsMy, err error) {
2022-09-18 04:34:48 +00:00
var in = []interface{}{"category"}
terms, err = models.Find[models.WpTermsMy](models.SqlBuilder{
{"tt.count", ">", "0", "int"},
{"tt.taxonomy", "in", ""},
}, "t.term_id", "", models.SqlBuilder{
{"t.name", "asc"},
}, models.SqlBuilder{
{"t", "inner join", "wp_term_taxonomy tt", "t.term_id = tt.term_id"},
}, 0, in)
for i := 0; i < len(terms); i++ {
if v, ok := models.Terms[terms[i].WpTerms.TermId]; ok {
terms[i].WpTerms = v
}
if v, ok := models.TermTaxonomy[terms[i].WpTerms.TermId]; ok {
terms[i].WpTermTaxonomy = v
}
}
return
}
2022-09-20 04:00:09 +00:00
func RecentPosts(ctx context.Context) (r []models.WpPosts) {
2022-09-20 08:11:20 +00:00
r, _ = recentPostsCaches.GetCache(ctx, time.Second)
return
2022-09-19 09:39:00 +00:00
}
2022-09-20 04:00:09 +00:00
func recentPosts(...any) (r []models.WpPosts, err error) {
2022-09-18 04:34:48 +00:00
r, err = models.Find[models.WpPosts](models.SqlBuilder{{
"post_type", "post",
}, {"post_status", "publish"}}, "ID,post_title,post_password", "", models.SqlBuilder{{"post_date", "desc"}}, nil, 5)
2022-09-18 14:06:27 +00:00
for i, post := range r {
if post.PostPassword != "" {
PasswordProjectTitle(&r[i])
}
}
2022-09-18 04:34:48 +00:00
return
}
2022-09-18 14:06:27 +00:00
func PasswordProjectTitle(post *models.WpPosts) {
if post.PostPassword != "" {
2022-09-18 04:34:48 +00:00
post.PostTitle = fmt.Sprintf("密码保护:%s", post.PostTitle)
}
2022-09-18 14:06:27 +00:00
}
func PasswdProjectContent(post *models.WpPosts) {
2022-09-18 04:34:48 +00:00
if post.PostContent != "" {
format := `
<form action="/login" class="post-password-form" method="post">
<p>此内容受密码保护如需查阅请在下列字段中输入您的密码</p>
<p><label for="pwbox-%d">密码 <input name="post_password" id="pwbox-%d" type="password" size="20"></label> <input type="submit" name="Submit" value="提交"></p>
</form>`
post.PostContent = fmt.Sprintf(format, post.Id, post.Id)
}
}