423 lines
12 KiB
Go
423 lines
12 KiB
Go
// SiYuan - Build Your Eternal Digital Garden
|
|
// Copyright (c) 2020-present, b3log.org
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
package bazaar
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/88250/gulu"
|
|
"github.com/88250/lute"
|
|
"github.com/PuerkitoBio/goquery"
|
|
"github.com/araddon/dateparse"
|
|
"github.com/imroc/req/v3"
|
|
"github.com/siyuan-note/filelock"
|
|
"github.com/siyuan-note/httpclient"
|
|
"github.com/siyuan-note/logging"
|
|
"github.com/siyuan-note/siyuan/kernel/util"
|
|
textUnicode "golang.org/x/text/encoding/unicode"
|
|
"golang.org/x/text/transform"
|
|
)
|
|
|
|
type Package struct {
|
|
Author string `json:"author"`
|
|
URL string `json:"url"`
|
|
Version string `json:"version"`
|
|
|
|
Name string `json:"name"`
|
|
RepoURL string `json:"repoURL"`
|
|
RepoHash string `json:"repoHash"`
|
|
PreviewURL string `json:"previewURL"`
|
|
PreviewURLThumb string `json:"previewURLThumb"`
|
|
|
|
README string `json:"readme"`
|
|
|
|
Installed bool `json:"installed"`
|
|
Outdated bool `json:"outdated"`
|
|
Current bool `json:"current"`
|
|
Updated string `json:"updated"`
|
|
Stars int `json:"stars"`
|
|
OpenIssues int `json:"openIssues"`
|
|
Size int64 `json:"size"`
|
|
HSize string `json:"hSize"`
|
|
InstallSize int64 `json:"installSize"`
|
|
HInstallSize string `json:"hInstallSize"`
|
|
HInstallDate string `json:"hInstallDate"`
|
|
HUpdated string `json:"hUpdated"`
|
|
Downloads int `json:"downloads"`
|
|
}
|
|
|
|
func WidgetJSON(widgetDirName string) (ret map[string]interface{}, err error) {
|
|
p := filepath.Join(util.DataDir, "widgets", widgetDirName, "widget.json")
|
|
if !gulu.File.IsExist(p) {
|
|
err = os.ErrNotExist
|
|
return
|
|
}
|
|
data, err := os.ReadFile(p)
|
|
if nil != err {
|
|
logging.LogErrorf("read widget.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if err = gulu.JSON.UnmarshalJSON(data, &ret); nil != err {
|
|
logging.LogErrorf("parse widget.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if 4 > len(ret) {
|
|
logging.LogWarnf("invalid widget.json [%s]", p)
|
|
return nil, errors.New("invalid widget.json")
|
|
}
|
|
return
|
|
}
|
|
|
|
func IconJSON(iconDirName string) (ret map[string]interface{}, err error) {
|
|
p := filepath.Join(util.IconsPath, iconDirName, "icon.json")
|
|
if !gulu.File.IsExist(p) {
|
|
err = os.ErrNotExist
|
|
return
|
|
}
|
|
data, err := os.ReadFile(p)
|
|
if nil != err {
|
|
logging.LogErrorf("read icon.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if err = gulu.JSON.UnmarshalJSON(data, &ret); nil != err {
|
|
logging.LogErrorf("parse icon.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if 4 > len(ret) {
|
|
logging.LogWarnf("invalid icon.json [%s]", p)
|
|
return nil, errors.New("invalid icon.json")
|
|
}
|
|
return
|
|
}
|
|
|
|
func TemplateJSON(templateDirName string) (ret map[string]interface{}, err error) {
|
|
p := filepath.Join(util.DataDir, "templates", templateDirName, "template.json")
|
|
if !gulu.File.IsExist(p) {
|
|
err = os.ErrNotExist
|
|
return
|
|
}
|
|
data, err := os.ReadFile(p)
|
|
if nil != err {
|
|
logging.LogErrorf("read template.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if err = gulu.JSON.UnmarshalJSON(data, &ret); nil != err {
|
|
logging.LogErrorf("parse template.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if 4 > len(ret) {
|
|
logging.LogWarnf("invalid template.json [%s]", p)
|
|
return nil, errors.New("invalid template.json")
|
|
}
|
|
return
|
|
}
|
|
|
|
func ThemeJSON(themeDirName string) (ret map[string]interface{}, err error) {
|
|
p := filepath.Join(util.ThemesPath, themeDirName, "theme.json")
|
|
if !gulu.File.IsExist(p) {
|
|
err = os.ErrNotExist
|
|
return
|
|
}
|
|
data, err := os.ReadFile(p)
|
|
if nil != err {
|
|
logging.LogErrorf("read theme.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if err = gulu.JSON.UnmarshalJSON(data, &ret); nil != err {
|
|
logging.LogErrorf("parse theme.json [%s] failed: %s", p, err)
|
|
return
|
|
}
|
|
if 5 > len(ret) {
|
|
logging.LogWarnf("invalid theme.json [%s]", p)
|
|
return nil, errors.New("invalid theme.json")
|
|
}
|
|
return
|
|
}
|
|
|
|
func getPkgIndex(pkgType string) (ret map[string]interface{}, err error) {
|
|
ret, err = util.GetRhyResult(false)
|
|
if nil != err {
|
|
return
|
|
}
|
|
|
|
bazaarHash := ret["bazaar"].(string)
|
|
ret = map[string]interface{}{}
|
|
request := httpclient.NewBrowserRequest()
|
|
u := util.BazaarOSSServer + "/bazaar@" + bazaarHash + "/stage/" + pkgType + ".json"
|
|
resp, reqErr := request.SetSuccessResult(&ret).Get(u)
|
|
if nil != reqErr {
|
|
logging.LogErrorf("get community stage index [%s] failed: %s", u, reqErr)
|
|
return
|
|
}
|
|
if 200 != resp.StatusCode {
|
|
logging.LogErrorf("get community stage index [%s] failed: %d", u, resp.StatusCode)
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
func isOutdatedTheme(theme *Theme, bazaarThemes []*Theme) bool {
|
|
if !strings.HasPrefix(theme.URL, "https://github.com/") {
|
|
return false
|
|
}
|
|
|
|
repo := strings.TrimPrefix(theme.URL, "https://github.com/")
|
|
parts := strings.Split(repo, "/")
|
|
if 2 != len(parts) || "" == strings.TrimSpace(parts[1]) {
|
|
return false
|
|
}
|
|
|
|
for _, pkg := range bazaarThemes {
|
|
if theme.URL == pkg.URL && theme.Name == pkg.Name && theme.Author == pkg.Author && theme.Version < pkg.Version {
|
|
theme.RepoHash = pkg.RepoHash
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func isOutdatedIcon(icon *Icon, bazaarIcons []*Icon) bool {
|
|
if !strings.HasPrefix(icon.URL, "https://github.com/") {
|
|
return false
|
|
}
|
|
|
|
repo := strings.TrimPrefix(icon.URL, "https://github.com/")
|
|
parts := strings.Split(repo, "/")
|
|
if 2 != len(parts) || "" == strings.TrimSpace(parts[1]) {
|
|
return false
|
|
}
|
|
|
|
for _, pkg := range bazaarIcons {
|
|
if icon.URL == pkg.URL && icon.Name == pkg.Name && icon.Author == pkg.Author && icon.Version < pkg.Version {
|
|
icon.RepoHash = pkg.RepoHash
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func isOutdatedWidget(widget *Widget, bazaarWidgets []*Widget) bool {
|
|
if !strings.HasPrefix(widget.URL, "https://github.com/") {
|
|
return false
|
|
}
|
|
|
|
repo := strings.TrimPrefix(widget.URL, "https://github.com/")
|
|
parts := strings.Split(repo, "/")
|
|
if 2 != len(parts) || "" == strings.TrimSpace(parts[1]) {
|
|
return false
|
|
}
|
|
|
|
for _, pkg := range bazaarWidgets {
|
|
if widget.URL == pkg.URL && widget.Name == pkg.Name && widget.Author == pkg.Author && widget.Version < pkg.Version {
|
|
widget.RepoHash = pkg.RepoHash
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func isOutdatedTemplate(template *Template, bazaarTemplates []*Template) bool {
|
|
if !strings.HasPrefix(template.URL, "https://github.com/") {
|
|
return false
|
|
}
|
|
|
|
repo := strings.TrimPrefix(template.URL, "https://github.com/")
|
|
parts := strings.Split(repo, "/")
|
|
if 2 != len(parts) || "" == strings.TrimSpace(parts[1]) {
|
|
return false
|
|
}
|
|
|
|
for _, pkg := range bazaarTemplates {
|
|
if template.URL == pkg.URL && template.Name == pkg.Name && template.Author == pkg.Author && template.Version < pkg.Version {
|
|
template.RepoHash = pkg.RepoHash
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func GetPackageREADME(repoURL, repoHash string, systemID string) (ret string) {
|
|
repoURLHash := repoURL + "@" + repoHash
|
|
data, err := downloadPackage(repoURLHash+"/README.md", false, systemID)
|
|
if nil != err {
|
|
ret = "Load bazaar package's README.md failed: " + err.Error()
|
|
return
|
|
}
|
|
|
|
if 2 < len(data) {
|
|
if 255 == data[0] && 254 == data[1] {
|
|
data, _, err = transform.Bytes(textUnicode.UTF16(textUnicode.LittleEndian, textUnicode.ExpectBOM).NewDecoder(), data)
|
|
} else if 254 == data[1] && 255 == data[0] {
|
|
data, _, err = transform.Bytes(textUnicode.UTF16(textUnicode.BigEndian, textUnicode.ExpectBOM).NewDecoder(), data)
|
|
}
|
|
}
|
|
|
|
ret, err = renderREADME(repoURL, data)
|
|
return
|
|
}
|
|
|
|
func renderREADME(repoURL string, mdData []byte) (ret string, err error) {
|
|
luteEngine := lute.New()
|
|
luteEngine.SetSoftBreak2HardBreak(false)
|
|
luteEngine.SetCodeSyntaxHighlight(false)
|
|
linkBase := repoURL + "/blob/main/"
|
|
luteEngine.SetLinkBase(linkBase)
|
|
ret = luteEngine.Md2HTML(string(mdData))
|
|
doc, err := goquery.NewDocumentFromReader(strings.NewReader(ret))
|
|
if nil != err {
|
|
logging.LogErrorf("parse HTML failed: %s", err)
|
|
return
|
|
}
|
|
|
|
doc.Find("a").Each(func(i int, selection *goquery.Selection) {
|
|
if href, ok := selection.Attr("href"); ok && util.IsRelativePath(href) {
|
|
selection.SetAttr("href", linkBase+href)
|
|
}
|
|
})
|
|
|
|
ret, _ = doc.Find("body").Html()
|
|
return
|
|
}
|
|
|
|
func downloadPackage(repoURLHash string, pushProgress bool, systemID string) (data []byte, err error) {
|
|
// repoURLHash: https://github.com/88250/Comfortably-Numb@6286912c381ef3f83e455d06ba4d369c498238dc
|
|
pushID := repoURLHash[:strings.LastIndex(repoURLHash, "@")]
|
|
repoURLHash = strings.TrimPrefix(repoURLHash, "https://github.com/")
|
|
u := util.BazaarOSSServer + "/package/" + repoURLHash
|
|
buf := &bytes.Buffer{}
|
|
resp, err := httpclient.NewBrowserRequest().SetOutput(buf).SetDownloadCallback(func(info req.DownloadInfo) {
|
|
if pushProgress {
|
|
progress := float32(info.DownloadedSize) / float32(info.Response.ContentLength)
|
|
//logging.LogDebugf("downloading bazaar package [%f]", progress)
|
|
util.PushDownloadProgress(pushID, progress)
|
|
}
|
|
}).Get(u)
|
|
if nil != err {
|
|
logging.LogErrorf("get bazaar package [%s] failed: %s", u, err)
|
|
return nil, errors.New("get bazaar package failed")
|
|
}
|
|
if 200 != resp.StatusCode {
|
|
logging.LogErrorf("get bazaar package [%s] failed: %d", u, resp.StatusCode)
|
|
return nil, errors.New("get bazaar package failed")
|
|
}
|
|
data = buf.Bytes()
|
|
|
|
go incPackageDownloads(repoURLHash, systemID)
|
|
return
|
|
}
|
|
|
|
func incPackageDownloads(repoURLHash, systemID string) {
|
|
if strings.Contains(repoURLHash, ".md") {
|
|
return
|
|
}
|
|
|
|
repo := strings.Split(repoURLHash, "@")[0]
|
|
u := util.AliyunServer + "/apis/siyuan/bazaar/addBazaarPackageDownloadCount"
|
|
httpclient.NewCloudRequest30s().SetBody(
|
|
map[string]interface{}{
|
|
"systemID": systemID,
|
|
"repo": repo,
|
|
}).Post(u)
|
|
}
|
|
|
|
func installPackage(data []byte, installPath string) (err error) {
|
|
dir := filepath.Join(util.TempDir, "bazaar", "package")
|
|
if err = os.MkdirAll(dir, 0755); nil != err {
|
|
return
|
|
}
|
|
name := gulu.Rand.String(7)
|
|
tmp := filepath.Join(dir, name+".zip")
|
|
if err = os.WriteFile(tmp, data, 0644); nil != err {
|
|
return
|
|
}
|
|
|
|
unzipPath := filepath.Join(dir, name)
|
|
if err = gulu.Zip.Unzip(tmp, unzipPath); nil != err {
|
|
logging.LogErrorf("write file [%s] failed: %s", installPath, err)
|
|
err = errors.New("write file failed")
|
|
return
|
|
}
|
|
|
|
dirs, err := os.ReadDir(unzipPath)
|
|
if nil != err {
|
|
return
|
|
}
|
|
for _, d := range dirs {
|
|
if d.IsDir() && strings.Contains(d.Name(), "-") {
|
|
dir = d.Name()
|
|
break
|
|
}
|
|
}
|
|
srcPath := filepath.Join(unzipPath, dir)
|
|
if err = filelock.Copy(srcPath, installPath); nil != err {
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
func formatUpdated(updated string) (ret string) {
|
|
t, e := dateparse.ParseIn(updated, time.Now().Location())
|
|
if nil == e {
|
|
ret = t.Format("2006-01-02")
|
|
} else {
|
|
if strings.Contains(updated, "T") {
|
|
ret = updated[:strings.Index(updated, "T")]
|
|
} else {
|
|
ret = strings.ReplaceAll(strings.ReplaceAll(updated, "T", ""), "Z", "")
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
type bazaarPackage struct {
|
|
Name string `json:"name"`
|
|
Downloads int `json:"downloads"`
|
|
}
|
|
|
|
var cachedBazaarIndex = map[string]*bazaarPackage{}
|
|
var bazaarIndexCacheTime int64
|
|
var bazaarIndexLock = sync.Mutex{}
|
|
|
|
func getBazaarIndex() map[string]*bazaarPackage {
|
|
bazaarIndexLock.Lock()
|
|
defer bazaarIndexLock.Unlock()
|
|
|
|
now := time.Now().Unix()
|
|
if 3600 >= now-bazaarIndexCacheTime {
|
|
return cachedBazaarIndex
|
|
}
|
|
|
|
request := httpclient.NewBrowserRequest()
|
|
u := util.BazaarStatServer + "/bazaar/index.json"
|
|
resp, reqErr := request.SetSuccessResult(&cachedBazaarIndex).Get(u)
|
|
if nil != reqErr {
|
|
logging.LogErrorf("get bazaar index [%s] failed: %s", u, reqErr)
|
|
return cachedBazaarIndex
|
|
}
|
|
if 200 != resp.StatusCode {
|
|
logging.LogErrorf("get bazaar index [%s] failed: %d", u, resp.StatusCode)
|
|
return cachedBazaarIndex
|
|
}
|
|
bazaarIndexCacheTime = now
|
|
return cachedBazaarIndex
|
|
}
|