123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- // 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 (
- "errors"
- "os"
- "path/filepath"
- "sort"
- "strings"
- "sync"
- "github.com/dustin/go-humanize"
- ants "github.com/panjf2000/ants/v2"
- "github.com/siyuan-note/httpclient"
- "github.com/siyuan-note/logging"
- "github.com/siyuan-note/siyuan/kernel/util"
- )
- type Theme struct {
- Package
- Modes []string `json:"modes"`
- }
- func Themes() (ret []*Theme) {
- ret = []*Theme{}
- pkgIndex, err := getPkgIndex("themes")
- if nil != err {
- return
- }
- bazaarIndex := getBazaarIndex()
- repos := pkgIndex["repos"].([]interface{})
- waitGroup := &sync.WaitGroup{}
- lock := &sync.Mutex{}
- p, _ := ants.NewPoolWithFunc(8, func(arg interface{}) {
- defer waitGroup.Done()
- repo := arg.(map[string]interface{})
- repoURL := repo["url"].(string)
- theme := &Theme{}
- innerU := util.BazaarOSSServer + "/package/" + repoURL + "/theme.json"
- innerResp, innerErr := httpclient.NewBrowserRequest().SetSuccessResult(theme).Get(innerU)
- if nil != innerErr {
- logging.LogErrorf("get bazaar package [%s] failed: %s", innerU, innerErr)
- return
- }
- if 200 != innerResp.StatusCode {
- logging.LogErrorf("get bazaar package [%s] failed: %d", innerU, innerResp.StatusCode)
- return
- }
- theme.URL = strings.TrimSuffix(theme.URL, "/")
- repoURLHash := strings.Split(repoURL, "@")
- theme.RepoURL = "https://github.com/" + repoURLHash[0]
- theme.RepoHash = repoURLHash[1]
- theme.PreviewURL = util.BazaarOSSServer + "/package/" + repoURL + "/preview.png?imageslim"
- theme.PreviewURLThumb = util.BazaarOSSServer + "/package/" + repoURL + "/preview.png?imageView2/2/w/436/h/232"
- theme.Updated = repo["updated"].(string)
- theme.Stars = int(repo["stars"].(float64))
- theme.OpenIssues = int(repo["openIssues"].(float64))
- theme.Size = int64(repo["size"].(float64))
- theme.HSize = humanize.Bytes(uint64(theme.Size))
- theme.HUpdated = formatUpdated(theme.Updated)
- pkg := bazaarIndex[strings.Split(repoURL, "@")[0]]
- if nil != pkg {
- theme.Downloads = pkg.Downloads
- }
- lock.Lock()
- ret = append(ret, theme)
- lock.Unlock()
- })
- for _, repo := range repos {
- waitGroup.Add(1)
- p.Invoke(repo)
- }
- waitGroup.Wait()
- p.Release()
- sort.Slice(ret, func(i, j int) bool { return ret[i].Updated > ret[j].Updated })
- return
- }
- func InstalledThemes() (ret []*Theme) {
- ret = []*Theme{}
- themeDirs, err := os.ReadDir(util.ThemesPath)
- if nil != err {
- logging.LogWarnf("read appearance themes folder failed: %s", err)
- return
- }
- bazaarThemes := Themes()
- for _, themeDir := range themeDirs {
- if !themeDir.IsDir() {
- continue
- }
- dirName := themeDir.Name()
- if isBuiltInTheme(dirName) {
- continue
- }
- themeConf, parseErr := ThemeJSON(dirName)
- if nil != parseErr || nil == themeConf {
- continue
- }
- installPath := filepath.Join(util.ThemesPath, dirName)
- theme := &Theme{}
- theme.Installed = true
- theme.Name = themeConf["name"].(string)
- theme.Author = themeConf["author"].(string)
- theme.URL = themeConf["url"].(string)
- theme.URL = strings.TrimSuffix(theme.URL, "/")
- theme.Version = themeConf["version"].(string)
- for _, mode := range themeConf["modes"].([]interface{}) {
- theme.Modes = append(theme.Modes, mode.(string))
- }
- theme.RepoURL = theme.URL
- theme.PreviewURL = "/appearance/themes/" + dirName + "/preview.png"
- theme.PreviewURLThumb = "/appearance/themes/" + dirName + "/preview.png"
- info, statErr := os.Stat(filepath.Join(installPath, "README.md"))
- if nil != statErr {
- logging.LogWarnf("stat install theme README.md failed: %s", statErr)
- continue
- }
- theme.HInstallDate = info.ModTime().Format("2006-01-02")
- installSize, _ := util.SizeOfDirectory(installPath)
- theme.InstallSize = installSize
- theme.HInstallSize = humanize.Bytes(uint64(installSize))
- readme, readErr := os.ReadFile(filepath.Join(installPath, "README.md"))
- if nil != readErr {
- logging.LogWarnf("read install theme README.md failed: %s", readErr)
- continue
- }
- theme.README, _ = renderREADME(theme.URL, readme)
- theme.Outdated = isOutdatedTheme(theme, bazaarThemes)
- ret = append(ret, theme)
- }
- return
- }
- func isBuiltInTheme(dirName string) bool {
- return "daylight" == dirName || "midnight" == dirName
- }
- func InstallTheme(repoURL, repoHash, installPath string, systemID string) error {
- repoURLHash := repoURL + "@" + repoHash
- data, err := downloadPackage(repoURLHash, true, systemID)
- if nil != err {
- return err
- }
- return installPackage(data, installPath)
- }
- func UninstallTheme(installPath string) error {
- if err := os.RemoveAll(installPath); nil != err {
- logging.LogErrorf("remove theme [%s] failed: %s", installPath, err)
- return errors.New("remove community theme failed")
- }
- //logging.Logger.Infof("uninstalled theme [%s]", installPath)
- return nil
- }
|