// 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 . 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/httpclient" "github.com/siyuan-note/logging" "github.com/siyuan-note/siyuan/kernel/filesys" "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.SetResult(&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.NewBrowserDownloadRequest().SetOutput(buf).SetDownloadCallback(func(info req.DownloadInfo) { if pushProgress { util.PushDownloadProgress(pushID, float32(info.DownloadedSize)/float32(info.Response.ContentLength)) } }).Get(u) if nil != err { u = util.BazaarOSSServer + "/package/" + repoURLHash resp, err = httpclient.NewBrowserDownloadRequest().SetOutput(buf).SetDownloadCallback(func(info req.DownloadInfo) { if pushProgress { util.PushDownloadProgress(pushID, float32(info.DownloadedSize)/float32(info.Response.ContentLength)) } }).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.NewCloudRequest().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 = filesys.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.SetResult(&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 }