release: v1.0.1 version
This commit is contained in:
parent
23136281f1
commit
2e55397574
22
README.md
Normal file
22
README.md
Normal file
@ -0,0 +1,22 @@
|
||||
# mder
|
||||
|
||||
> 一个极速静态网站生成工具
|
||||
|
||||
快速开始
|
||||
|
||||
```shell
|
||||
# 执行安装
|
||||
go install gitter.top/mder/mder@latest
|
||||
# 创建第一个项目
|
||||
mder init --name "my_first_blog"
|
||||
# 本地预览
|
||||
mder serve
|
||||
# 本地部署
|
||||
mder deploy
|
||||
```
|
||||
|
||||
## changelog
|
||||
|
||||
| 时间节点 | 描述 |
|
||||
|------------|-----------|
|
||||
| 2022-07-25 | 服务端基本功能完成 |
|
28
cmd.go
28
cmd.go
@ -2,6 +2,12 @@ package main
|
||||
|
||||
import (
|
||||
"github.com/spf13/cobra"
|
||||
"github.com/yuin/goldmark"
|
||||
emoji "github.com/yuin/goldmark-emoji"
|
||||
meta "github.com/yuin/goldmark-meta"
|
||||
"github.com/yuin/goldmark/extension"
|
||||
"github.com/yuin/goldmark/parser"
|
||||
"go.abhg.dev/goldmark/mermaid"
|
||||
)
|
||||
|
||||
var (
|
||||
@ -9,13 +15,31 @@ var (
|
||||
Use: "mder",
|
||||
Short: "mder is a very fast static site generator",
|
||||
}
|
||||
|
||||
markdown = goldmark.New(
|
||||
goldmark.WithParserOptions(parser.WithAutoHeadingID()),
|
||||
goldmark.WithExtensions(extension.GFM, meta.Meta, emoji.Emoji, &mermaid.Extender{}),
|
||||
)
|
||||
)
|
||||
|
||||
func init() {
|
||||
// create a new mder folder
|
||||
// create a new site folder
|
||||
rootCmd.AddCommand(initCmd())
|
||||
// generate static website
|
||||
rootCmd.AddCommand(generateCmd())
|
||||
// new post or page
|
||||
rootCmd.AddCommand(newCmd())
|
||||
// run serve locally
|
||||
rootCmd.AddCommand(serveCmd())
|
||||
// auto update
|
||||
rootCmd.AddCommand(updateCmd())
|
||||
// deploy project
|
||||
rootCmd.AddCommand(deployCmd())
|
||||
}
|
||||
|
||||
func main() {
|
||||
rootCmd.Execute()
|
||||
err := rootCmd.Execute()
|
||||
if err != nil {
|
||||
logger.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
113
config.go
Normal file
113
config.go
Normal file
@ -0,0 +1,113 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"os"
|
||||
|
||||
"gopkg.in/yaml.v3"
|
||||
)
|
||||
|
||||
// Logo 头像配置
|
||||
type Logo struct {
|
||||
Enabled bool `yaml:"enabled"` // 显示或隐藏
|
||||
Width int64 `yaml:"width"` // 宽度控制
|
||||
Height int64 `yaml:"height"` // 高度控制
|
||||
Url string `yaml:"url"` // 源url
|
||||
}
|
||||
|
||||
// Favicon 自定义favicon
|
||||
type Favicon struct {
|
||||
Url string `yaml:"url"`
|
||||
}
|
||||
|
||||
// SocialLinks 社交媒体链接
|
||||
type SocialLinks struct {
|
||||
Github string `yaml:"github"`
|
||||
Email string `yaml:"email"`
|
||||
QQ string `yaml:"qq"`
|
||||
Wechat string `yaml:"wechat"`
|
||||
Twitter string `yaml:"twitter"`
|
||||
Telegram string `yaml:"telegram"`
|
||||
}
|
||||
|
||||
// ICP 备案配置
|
||||
type ICP struct {
|
||||
Enabled bool `yaml:"enabled"` // 显示或隐藏
|
||||
Url string `yaml:"url"` // 重定向地址
|
||||
Text string `yaml:"text"` // 内容
|
||||
}
|
||||
|
||||
// CDN 厂商信息
|
||||
type CDN struct {
|
||||
Enabled bool `yaml:"enabled"`
|
||||
Url string `yaml:"url"`
|
||||
Image string `yaml:"image"`
|
||||
Text string `yaml:"text"`
|
||||
}
|
||||
|
||||
// Comment 评论功能
|
||||
type Comment struct {
|
||||
Enabled bool `yaml:"enabled"`
|
||||
}
|
||||
|
||||
// Site 站点配置
|
||||
type Site struct {
|
||||
Title string `yaml:"title"` // 网站标题
|
||||
Subtitle string `yaml:"subtitle"` // 副标题
|
||||
Description string `yaml:"description"` // 描述
|
||||
Keywords string `yaml:"keywords"` // 关键字
|
||||
Author string `yaml:"author"` // 您的名字
|
||||
Summary string `yaml:"summary"` // 个人总结
|
||||
Theme string `yaml:"theme"` // 主题
|
||||
}
|
||||
|
||||
// PageConfig 页面配置
|
||||
type PageConfig struct {
|
||||
Paginate bool `yaml:"paginate"` // 是否开启分页
|
||||
Size int64 `yaml:"size"` // 每页数
|
||||
Total int // 页总数
|
||||
CurrentSize int // 当前页数
|
||||
}
|
||||
|
||||
// Config 配置文件
|
||||
type Config struct {
|
||||
Logo Logo `yaml:"logo"`
|
||||
Favicon Favicon `yaml:"favicon"`
|
||||
SocialLinks SocialLinks `yaml:"social_links"`
|
||||
ICP ICP `yaml:"icp"`
|
||||
CDN CDN `yaml:"cdn"`
|
||||
Comment Comment `yaml:"comment"`
|
||||
PageConfig PageConfig `yaml:"page"`
|
||||
Site Site `yaml:"site"` // 站点配置信息
|
||||
Deploy Deploy `yaml:"deploy"` // 部署配置
|
||||
}
|
||||
|
||||
var BaseDir string
|
||||
|
||||
func (c *Config) load() error {
|
||||
var file = "config.yaml"
|
||||
if BaseDir != "" {
|
||||
file = BaseDir + "/config.yaml"
|
||||
}
|
||||
configBuffer, err := os.ReadFile(file)
|
||||
if err != nil {
|
||||
logger.Errorf("read config file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
if err := yaml.Unmarshal(configBuffer, c); err != nil {
|
||||
logger.Errorf("read config file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type DeployType string
|
||||
|
||||
const (
|
||||
GitDeploy DeployType = "git"
|
||||
UpyunDeploy DeployType = "upyun"
|
||||
)
|
||||
|
||||
type Deploy struct {
|
||||
Type DeployType `yaml:"type"`
|
||||
UpyunAuth string `yaml:"upyun_auth"`
|
||||
}
|
59
deploy_command.go
Normal file
59
deploy_command.go
Normal file
@ -0,0 +1,59 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"github.com/spf13/cobra"
|
||||
"strings"
|
||||
)
|
||||
|
||||
func deployCmd() *cobra.Command {
|
||||
var path string
|
||||
var outter Outter
|
||||
cmd := &cobra.Command{
|
||||
Use: "deploy",
|
||||
Aliases: []string{"d"},
|
||||
Short: "deploy project to upyun",
|
||||
PreRun: func(cmd *cobra.Command, args []string) {
|
||||
if path != "" {
|
||||
BaseDir = strings.TrimSuffix(path, "/")
|
||||
}
|
||||
|
||||
if err := outter.loadConfig(); err != nil {
|
||||
logger.Fatalf("load config failed: %v", err)
|
||||
}
|
||||
config := outter.Config.Deploy
|
||||
switch config.Type {
|
||||
case UpyunDeploy:
|
||||
if !isCommandExist("upx") {
|
||||
if err := goInstall("github.com/upyun/upx/cmd/upx"); err != nil {
|
||||
logger.Errorf("install upx failed: %+v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
case GitDeploy:
|
||||
}
|
||||
|
||||
// generate source
|
||||
if err := generateCmd().Execute(); err != nil {
|
||||
logger.Errorf("generate project failed: %v", err)
|
||||
return
|
||||
}
|
||||
},
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
config := outter.Config.Deploy
|
||||
switch config.Type {
|
||||
case UpyunDeploy:
|
||||
if config.UpyunAuth == "" {
|
||||
logger.Errorf("please config upyun auth string: upx auth [bucket] [operator] [password]")
|
||||
return
|
||||
}
|
||||
if err := uploadToUpyun(config.UpyunAuth, path); err != nil {
|
||||
logger.Errorf("upload dist to upyun failed: %v", err)
|
||||
return
|
||||
}
|
||||
logger.Infof("deploy to upyun success")
|
||||
}
|
||||
},
|
||||
}
|
||||
cmd.Flags().StringVar(&path, "path", ".", "mder project path")
|
||||
return cmd
|
||||
}
|
933
generate_command.go
Normal file
933
generate_command.go
Normal file
@ -0,0 +1,933 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"math"
|
||||
"os"
|
||||
"os/exec"
|
||||
"sort"
|
||||
"strings"
|
||||
"text/template"
|
||||
"time"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
meta "github.com/yuin/goldmark-meta"
|
||||
"github.com/yuin/goldmark/parser"
|
||||
"github.com/yuin/goldmark/text"
|
||||
"go.abhg.dev/goldmark/toc"
|
||||
)
|
||||
|
||||
func generateCmd() *cobra.Command {
|
||||
var startAt time.Time
|
||||
var outter = newOutter()
|
||||
var path string
|
||||
cmd := &cobra.Command{
|
||||
Use: "generate",
|
||||
Aliases: []string{"g"},
|
||||
Short: "generate project to dist folder",
|
||||
PreRun: func(cmd *cobra.Command, args []string) {
|
||||
if path != "" {
|
||||
BaseDir = strings.TrimSuffix(path, "/")
|
||||
}
|
||||
startAt = time.Now()
|
||||
// 读取配置文件
|
||||
if err := outter.loadConfig(); err != nil {
|
||||
logger.Fatalf("load config file failed: %v", err)
|
||||
}
|
||||
},
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
// 读取资源文件
|
||||
if err := outter.readDataSource(); err != nil {
|
||||
logger.Errorf("read data source failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 读取主题模板文件
|
||||
if err := outter.readTheme(outter.Config.Site.Theme); err != nil {
|
||||
logger.Errorf("read theme source failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 读取页面列表
|
||||
if err := outter.readAllPages(); err != nil {
|
||||
logger.Errorf("read page source failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 读取文章列表
|
||||
if err := outter.readAllPosts(); err != nil {
|
||||
logger.Errorf("read post source failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 数据写入模板文件
|
||||
outter.generate()
|
||||
},
|
||||
PostRun: func(cmd *cobra.Command, args []string) {
|
||||
endAt := time.Since(startAt)
|
||||
logger.Infof("generate used: %s", endAt.String())
|
||||
},
|
||||
}
|
||||
cmd.Flags().StringVar(&path, "path", ".", "mder project path")
|
||||
return cmd
|
||||
}
|
||||
|
||||
// readAllPosts 读取所有post文章
|
||||
func (o *Outter) readAllPosts() error {
|
||||
var postDir = BaseDir + "/posts"
|
||||
dirs, err := os.ReadDir(postDir)
|
||||
if err != nil {
|
||||
logger.Errorf("read directory failed: %v", err)
|
||||
return err
|
||||
}
|
||||
for _, info := range dirs {
|
||||
// 获取文件信息
|
||||
fsInfo, err := info.Info()
|
||||
if err != nil {
|
||||
logger.Errorf("read file info failed: %v", err)
|
||||
continue
|
||||
}
|
||||
// 不是目录 没有分类
|
||||
if info.IsDir() {
|
||||
o.Posts = append(o.Posts, o.readPosts(postDir, info.Name())...)
|
||||
continue
|
||||
}
|
||||
if !strings.HasSuffix(info.Name(), ".md") {
|
||||
continue
|
||||
}
|
||||
post, err := o.readPost(fmt.Sprintf("%s/%s", postDir, info.Name()))
|
||||
if err != nil {
|
||||
logger.Errorf("read post file failed: %v", err)
|
||||
continue
|
||||
}
|
||||
post.UpdatedAtFormat = fsInfo.ModTime().Format(time.DateOnly)
|
||||
post.FileBasename = strings.ReplaceAll(fsInfo.Name(), ".md", "")
|
||||
post.Link = fmt.Sprintf("/default/%s.html", post.FileBasename)
|
||||
post.Category = "default"
|
||||
if isDraft(post.Tags) {
|
||||
post.Link = fmt.Sprintf("/draft/%s.html", post.FileBasename)
|
||||
o.DraftPosts = append(o.DraftPosts, post)
|
||||
} else {
|
||||
o.Posts = append(o.Posts, post)
|
||||
}
|
||||
}
|
||||
sort.Slice(o.Posts, func(i, j int) bool {
|
||||
return o.Posts[i].CreatedAt.Unix() > o.Posts[j].CreatedAt.Unix()
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
func (o *Outter) readPost(fp string) (*Post, error) {
|
||||
var post = new(Post)
|
||||
content, err := os.ReadFile(fp)
|
||||
if err != nil {
|
||||
logger.Errorf("read file %s failed: %v", fp, err)
|
||||
return nil, err
|
||||
}
|
||||
var buf bytes.Buffer
|
||||
var parserContext = parser.NewContext()
|
||||
doc := markdown.Parser().Parse(text.NewReader(content), parser.WithContext(parserContext))
|
||||
|
||||
tocTree, err := toc.Inspect(doc, content)
|
||||
if err != nil {
|
||||
logger.Errorf("read content toc tree failed: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
|
||||
list := toc.RenderList(tocTree)
|
||||
if list != nil {
|
||||
if err := markdown.Renderer().Render(&buf, content, list); err != nil {
|
||||
logger.Errorf("render toc content failed: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
post.TOC = buf.String()
|
||||
buf.Reset()
|
||||
}
|
||||
if err := markdown.Renderer().Render(&buf, content, doc); err != nil {
|
||||
logger.Errorf("render content failed: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
|
||||
metaData := meta.Get(parserContext)
|
||||
post.CreatedAt, err = datetimeStringToTime(mustString(metaData["date"]))
|
||||
if err != nil {
|
||||
logger.Errorf("get post %s created_at time failed, please check file content", post.FileBasename)
|
||||
return nil, err
|
||||
}
|
||||
post.CreatedAtFormat = post.CreatedAt.Format("2006-01-02")
|
||||
post.Title = mustString(metaData["title"])
|
||||
post.Tags = mustStringSlice(metaData["tags"])
|
||||
post.Category = mustString(metaData["category"])
|
||||
post.MD = buf.String()
|
||||
return post, nil
|
||||
}
|
||||
|
||||
func (o *Outter) readPage(fp string) (*Page, error) {
|
||||
var page = new(Page)
|
||||
content, err := os.ReadFile(fp)
|
||||
if err != nil {
|
||||
logger.Errorf("read page file failed: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
var buf bytes.Buffer
|
||||
context := parser.NewContext()
|
||||
if err := markdown.Convert(content, &buf, parser.WithContext(context)); err != nil {
|
||||
logger.Errorf("render content failed: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
metaData := meta.Get(context)
|
||||
page.Title = mustString(metaData["title"])
|
||||
page.MD = buf.String()
|
||||
return page, nil
|
||||
}
|
||||
|
||||
func (o *Outter) readPosts(base, category string) []*Post {
|
||||
var dir = fmt.Sprintf("%s/%s", base, category)
|
||||
dirs, err := os.ReadDir(dir)
|
||||
if err != nil {
|
||||
logger.Errorf("read directory failed: %v", err)
|
||||
return nil
|
||||
}
|
||||
var posts []*Post
|
||||
for _, info := range dirs {
|
||||
if info.IsDir() {
|
||||
continue
|
||||
}
|
||||
if !strings.HasSuffix(info.Name(), ".md") {
|
||||
continue
|
||||
}
|
||||
// 获取文件信息
|
||||
fsInfo, err := info.Info()
|
||||
if err != nil {
|
||||
logger.Errorf("read file info failed: %v", err)
|
||||
return nil
|
||||
}
|
||||
post, err := o.readPost(fmt.Sprintf("%s/%s", dir, info.Name()))
|
||||
if err != nil {
|
||||
logger.Errorf("read post failed: %v", err)
|
||||
continue
|
||||
}
|
||||
post.UpdatedAtFormat = fsInfo.ModTime().Format(time.DateOnly)
|
||||
post.FileBasename = strings.ReplaceAll(fsInfo.Name(), ".md", "")
|
||||
post.Link = fmt.Sprintf("/%s/%s.html", strings.ToLower(category), post.FileBasename)
|
||||
post.Category = category
|
||||
if isDraft(post.Tags) {
|
||||
post.Link = fmt.Sprintf("/draft/%s.html", post.FileBasename)
|
||||
o.DraftPosts = append(o.DraftPosts, post)
|
||||
} else {
|
||||
posts = append(posts, post)
|
||||
}
|
||||
}
|
||||
return posts
|
||||
}
|
||||
|
||||
func (o *Outter) readAllPages() error {
|
||||
var dir = BaseDir + "/pages"
|
||||
dirs, err := os.ReadDir(dir)
|
||||
if err != nil {
|
||||
logger.Errorf("read directory failed: %v", err)
|
||||
return err
|
||||
}
|
||||
for _, info := range dirs {
|
||||
if info.IsDir() {
|
||||
continue
|
||||
}
|
||||
if !strings.HasSuffix(info.Name(), ".md") {
|
||||
continue
|
||||
}
|
||||
page, err := o.readPage(fmt.Sprintf("%s/%s", dir, info.Name()))
|
||||
if err != nil {
|
||||
logger.Errorf("read page file failed: %v", err)
|
||||
continue
|
||||
}
|
||||
page.Link = strings.ReplaceAll(info.Name(), ".md", "")
|
||||
o.Pages = append(o.Pages, page)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (o *Outter) readTheme(themeName string) (err error) {
|
||||
o.Theme = new(Theme)
|
||||
var dir = fmt.Sprintf("%s/themes/%s", BaseDir, themeName)
|
||||
o.Theme.BaseLayout, err = os.ReadFile(fmt.Sprintf("%s/base.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read base.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.IndexLayout, err = os.ReadFile(fmt.Sprintf("%s/index.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read index.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.PageLayout, err = os.ReadFile(fmt.Sprintf("%s/page.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read page.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.PostLayout, err = os.ReadFile(fmt.Sprintf("%s/post.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read pose.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.ArchiveLayout, err = os.ReadFile(fmt.Sprintf("%s/archive.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read archive.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.TagLayout, err = os.ReadFile(fmt.Sprintf("%s/tag.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read tag.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.Theme.CategoryLayout, err = os.ReadFile(fmt.Sprintf("%s/category.html", dir))
|
||||
if err != nil {
|
||||
logger.Errorf("read category.html theme file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
|
||||
o.Theme.IndexLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.IndexLayout)
|
||||
o.Theme.PageLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.PageLayout)
|
||||
o.Theme.PostLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.PostLayout)
|
||||
o.Theme.ArchiveLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.ArchiveLayout)
|
||||
o.Theme.TagLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.TagLayout)
|
||||
o.Theme.CategoryLayout = bytes.ReplaceAll(o.Theme.BaseLayout, []byte("{{layout_placeholder}}"), o.Theme.TagLayout)
|
||||
return nil
|
||||
}
|
||||
|
||||
type Outter struct {
|
||||
SourceVersion string // 资源号 防缓存
|
||||
Config *Config // 全局配置
|
||||
SourceData map[string]interface{} // 记录source/data下的所有文件
|
||||
Posts []*Post // 记录文章
|
||||
DraftPosts []*Post // 不宜发布的草稿文章
|
||||
Pages []*Page // 记录页面
|
||||
Theme *Theme // 记录主题模板文件
|
||||
Now time.Time // 当前时间
|
||||
}
|
||||
|
||||
type Theme struct {
|
||||
Name string // 主题名
|
||||
BaseLayout []byte // 基本布局
|
||||
PostLayout []byte // 文章布局
|
||||
PageLayout []byte // 页面布局
|
||||
IndexLayout []byte // 首页布局
|
||||
ArchiveLayout []byte // 文章归档布局
|
||||
TagLayout []byte // 标签归档布局
|
||||
CategoryLayout []byte // 分类归档布局
|
||||
}
|
||||
|
||||
// Post 文章属性
|
||||
type Post struct {
|
||||
Title string // 文章标题
|
||||
FileBasename string // 文件名
|
||||
Link string // 链接
|
||||
Category string // 分类
|
||||
CategoryAlias string // 分类别名
|
||||
Tags []string // 标签
|
||||
CreatedAt time.Time // 创建时间
|
||||
CreatedAtFormat string // 创建时间格式化
|
||||
UpdatedAtFormat string // 更新时间
|
||||
MD string // 文章内容
|
||||
TOC string // 文章toc
|
||||
}
|
||||
|
||||
// Page 页面属性
|
||||
type Page struct {
|
||||
Title string // 展示名
|
||||
Link string // 链接名
|
||||
MD string // 页面内容
|
||||
}
|
||||
|
||||
func newOutter() *Outter {
|
||||
return &Outter{
|
||||
SourceVersion: randString(8),
|
||||
Config: new(Config),
|
||||
Now: time.Now(),
|
||||
}
|
||||
}
|
||||
|
||||
// loadConfig 加载配置文件
|
||||
func (o *Outter) loadConfig() error {
|
||||
if o.Config == nil {
|
||||
o.Config = new(Config)
|
||||
}
|
||||
return o.Config.load()
|
||||
}
|
||||
|
||||
// readDataSource 读取资源文件
|
||||
func (o *Outter) readDataSource() error {
|
||||
o.SourceData = make(map[string]interface{})
|
||||
var dataDir = BaseDir + "/data"
|
||||
dirs, err := os.ReadDir(dataDir)
|
||||
if err != nil {
|
||||
return nil
|
||||
}
|
||||
for _, dir := range dirs {
|
||||
if dir.IsDir() {
|
||||
continue
|
||||
}
|
||||
// 不是json文件
|
||||
if !strings.HasSuffix(dir.Name(), ".json") {
|
||||
continue
|
||||
}
|
||||
var body []byte
|
||||
body, err = os.ReadFile(fmt.Sprintf("%s/%s", dataDir, dir.Name()))
|
||||
if err != nil {
|
||||
logger.Errorf("read data source file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
var obj interface{}
|
||||
if err := json.Unmarshal(body, &obj); err != nil {
|
||||
logger.Errorf("unmarshal data source file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
o.SourceData[strings.ReplaceAll(dir.Name(), ".json", "")] = obj
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// generate 文件生成
|
||||
func (o *Outter) generate() {
|
||||
// 清理dist目录
|
||||
if err := o.clearDistDirectory(); err != nil {
|
||||
logger.Errorf("clear dist directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
if err := o.sourceCopy(); err != nil {
|
||||
logger.Errorf("copy theme source to dist failed: %v", err)
|
||||
return
|
||||
}
|
||||
_ = o.generateIndex()
|
||||
o.generatePost()
|
||||
o.generateDraftPost()
|
||||
o.generatePage()
|
||||
o.generateArchives()
|
||||
o.generateTags()
|
||||
o.generateCategories()
|
||||
}
|
||||
|
||||
// clearDistDirectory 清理dist目录
|
||||
func (o *Outter) clearDistDirectory() error {
|
||||
if err := os.RemoveAll(BaseDir + "/dist"); err != nil {
|
||||
logger.Errorf("remove old dist directory failed: %v", err)
|
||||
return err
|
||||
}
|
||||
|
||||
if err := mkdir(BaseDir + "/dist"); err != nil {
|
||||
logger.Errorf("create dist directory failed: %v", err)
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// sourceCopy 资源拷贝 将主题的资源拷贝到目标文件夹中
|
||||
func (o *Outter) sourceCopy() error {
|
||||
sourcePath := slash(fmt.Sprintf(BaseDir+"/themes/%s/", o.Config.Site.Theme))
|
||||
destPath := slash(BaseDir + "/dist/" + o.SourceVersion + "/")
|
||||
// mkdir
|
||||
cssDir := fmt.Sprintf("%s/css", destPath)
|
||||
jsDir := fmt.Sprintf("%s/js", destPath)
|
||||
imagesDir := fmt.Sprintf("%s/images", destPath)
|
||||
if err := mkdir(cssDir); err != nil {
|
||||
logger.Errorf("mkdir css directory failed: %v", err)
|
||||
return err
|
||||
}
|
||||
if err := mkdir(jsDir); err != nil {
|
||||
logger.Errorf("mkdir js directory failed: %v", err)
|
||||
}
|
||||
if err := mkdir(imagesDir); err != nil {
|
||||
logger.Errorf("mkdir images directory failed: %v", err)
|
||||
}
|
||||
cmd := exec.Command("cp", "-r", sourcePath+"css", destPath)
|
||||
if err := cmd.Run(); err != nil {
|
||||
logger.Infof(cmd.String())
|
||||
logger.Errorf("copy theme css source failed: %v", err)
|
||||
return err
|
||||
}
|
||||
cmd = exec.Command("cp", "-r", sourcePath+"js", destPath)
|
||||
if err := cmd.Run(); err != nil {
|
||||
logger.Infof(cmd.String())
|
||||
logger.Errorf("copy theme js source failed: %v", err)
|
||||
return err
|
||||
}
|
||||
cmd = exec.Command("cp", "-r", sourcePath+"images", destPath)
|
||||
if err := cmd.Run(); err != nil {
|
||||
logger.Infof(cmd.String())
|
||||
logger.Errorf("copy theme images source failed: %v", err)
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// generateIndex 首页生成
|
||||
func (o *Outter) generateIndex() error {
|
||||
indexTemplate := template.New("index")
|
||||
|
||||
funcMap["title"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
|
||||
indexTemplate.Funcs(funcMap)
|
||||
|
||||
indexTemplate, err := indexTemplate.Parse(string(o.Theme.IndexLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("parse index layout failed: %v", err)
|
||||
return err
|
||||
}
|
||||
|
||||
// 首页文件
|
||||
var buffer = bytes.Buffer{}
|
||||
|
||||
// 没开分页
|
||||
if !o.Config.PageConfig.Paginate {
|
||||
err = indexTemplate.Execute(&buffer, o)
|
||||
if err != nil {
|
||||
logger.Errorf("generate index page failed: %v", err)
|
||||
return err
|
||||
}
|
||||
var filename = BaseDir + "/dist/index.html"
|
||||
if err := os.WriteFile(filename, buffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write index file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
logger.Infof("index generate success...")
|
||||
return nil
|
||||
}
|
||||
|
||||
// 分页数据不规范
|
||||
if o.Config.PageConfig.Size < 1 {
|
||||
logger.Errorf("page size must > 1")
|
||||
return err
|
||||
}
|
||||
|
||||
var pageSize = int(math.Ceil(float64(len(o.Posts)) / float64(o.Config.PageConfig.Size)))
|
||||
o.Config.PageConfig.Total = pageSize
|
||||
var posts = make([]*Post, len(o.Posts))
|
||||
copy(posts, o.Posts)
|
||||
for i := 0; i < pageSize; i++ {
|
||||
rightBorder := int(o.Config.PageConfig.Size) * (i + 1)
|
||||
if rightBorder > len(posts) {
|
||||
rightBorder = len(posts)
|
||||
}
|
||||
leftBorder := int(o.Config.PageConfig.Size) * (i)
|
||||
o.Posts = make([]*Post, rightBorder-leftBorder)
|
||||
copy(o.Posts, posts[leftBorder:rightBorder])
|
||||
o.Config.PageConfig.CurrentSize = i + 1
|
||||
|
||||
err = indexTemplate.Execute(&buffer, o)
|
||||
if err != nil {
|
||||
logger.Errorf("generate index page failed: %v", err)
|
||||
return err
|
||||
}
|
||||
// 第一页 主页
|
||||
if i == 0 {
|
||||
// 第一次的时候创建目录
|
||||
dir := BaseDir + "/dist/page"
|
||||
if err := o.createDir(dir); err != nil {
|
||||
logger.Errorf("create index page failed: %v", err)
|
||||
return err
|
||||
}
|
||||
var filename = BaseDir + "/dist/index.html"
|
||||
if err := os.WriteFile(filename, buffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write index file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
}
|
||||
fPage := fmt.Sprintf(BaseDir+"/dist/page/%d.html", i+1)
|
||||
if err := os.WriteFile(fPage, buffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write index file failed: %v", err)
|
||||
return err
|
||||
}
|
||||
buffer.Reset()
|
||||
}
|
||||
o.Posts = make([]*Post, len(posts))
|
||||
copy(o.Posts, posts)
|
||||
return nil
|
||||
}
|
||||
|
||||
func (o *Outter) generatePost() {
|
||||
postTemplate := template.New("post")
|
||||
var postBuffer = new(bytes.Buffer)
|
||||
for _, post := range o.Posts {
|
||||
instance := PostOutter{
|
||||
Post: post,
|
||||
Config: o.Config,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return instance.Post.Title
|
||||
}
|
||||
funcMap["post_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
postTemplate.Funcs(funcMap)
|
||||
postTemplate, err := postTemplate.Parse(string(o.Theme.PostLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate post page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 入buffer
|
||||
if err := postTemplate.Execute(postBuffer, instance); err != nil {
|
||||
logger.Errorf("generate post page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// buffer写文件
|
||||
var catDir = fmt.Sprintf(BaseDir+"/dist/%s", post.Category)
|
||||
if !isExist(catDir) {
|
||||
if err := mkdir(catDir); err != nil {
|
||||
logger.Errorf("create directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
var filename = fmt.Sprintf("%s/%s.html", catDir, post.FileBasename)
|
||||
if err := os.WriteFile(filename, postBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write file failed: %v", err)
|
||||
return
|
||||
}
|
||||
postBuffer.Reset()
|
||||
}
|
||||
}
|
||||
|
||||
func (o *Outter) generateDraftPost() {
|
||||
postTemplate := template.New("post")
|
||||
var postBuffer = new(bytes.Buffer)
|
||||
for _, post := range o.DraftPosts {
|
||||
instance := PostOutter{
|
||||
Post: post,
|
||||
Config: o.Config,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return instance.Post.Title
|
||||
}
|
||||
funcMap["post_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
postTemplate.Funcs(funcMap)
|
||||
postTemplate, err := postTemplate.Parse(string(o.Theme.PostLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate post page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 入buffer
|
||||
if err := postTemplate.Execute(postBuffer, instance); err != nil {
|
||||
logger.Errorf("generate post page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// buffer写文件
|
||||
var catDir = BaseDir + "/dist/draft"
|
||||
if !isExist(catDir) {
|
||||
if err := mkdir(catDir); err != nil {
|
||||
logger.Errorf("create directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
var filename = fmt.Sprintf("%s/%s.html", catDir, post.FileBasename)
|
||||
if err := os.WriteFile(filename, postBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write file failed: %v", err)
|
||||
return
|
||||
}
|
||||
postBuffer.Reset()
|
||||
}
|
||||
}
|
||||
|
||||
func (o *Outter) generatePage() {
|
||||
pageTemplate := template.New("page")
|
||||
var pageBuffer = new(bytes.Buffer)
|
||||
for _, page := range o.Pages {
|
||||
instance := PageOutter{
|
||||
Page: page,
|
||||
Config: o.Config,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return page.Title
|
||||
}
|
||||
funcMap["page_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
pageTemplate.Funcs(funcMap)
|
||||
pageTemplate, err := pageTemplate.Parse(string(o.Theme.PageLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate page page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 入buffer
|
||||
if err := pageTemplate.Execute(pageBuffer, instance); err != nil {
|
||||
logger.Errorf("generate page page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// buffer写文件
|
||||
var filename = fmt.Sprintf(BaseDir+"/dist/%s.html", page.Link)
|
||||
if err := os.WriteFile(filename, pageBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write page content failed: %v", err)
|
||||
return
|
||||
}
|
||||
pageBuffer.Reset()
|
||||
}
|
||||
}
|
||||
|
||||
func (o *Outter) generateArchives() {
|
||||
archiveTemplate := template.New("archive")
|
||||
var archiveBuffer = new(bytes.Buffer)
|
||||
// 按时间归档
|
||||
var m = make(map[string][]*Post)
|
||||
for _, post := range o.Posts {
|
||||
newPost := post
|
||||
newPost.MD = ""
|
||||
m[int2String(post.CreatedAt.Year())] = append(m[int2String(post.CreatedAt.Year())], newPost)
|
||||
}
|
||||
|
||||
var postData []*PostData
|
||||
for year, posts := range m {
|
||||
postData = append(postData, &PostData{
|
||||
Key: year,
|
||||
Posts: posts,
|
||||
})
|
||||
}
|
||||
|
||||
sort.Slice(postData, func(i, j int) bool {
|
||||
return postData[i].Key > postData[j].Key
|
||||
})
|
||||
|
||||
var instance = PageOutter{
|
||||
Page: &Page{
|
||||
Title: "Archives",
|
||||
Link: "archives",
|
||||
},
|
||||
Config: o.Config,
|
||||
PostData: postData,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return instance.Page.Title
|
||||
}
|
||||
funcMap["page_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
archiveTemplate.Funcs(funcMap)
|
||||
archiveTemplate, err := archiveTemplate.Parse(string(o.Theme.ArchiveLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate archive page failed: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
// 入buffer
|
||||
if err := archiveTemplate.Execute(archiveBuffer, instance); err != nil {
|
||||
logger.Errorf("generate archive page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// buffer写文件
|
||||
var filename = fmt.Sprintf(BaseDir+"/dist/%s.html", instance.Page.Link)
|
||||
if err := os.WriteFile(filename, archiveBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write archive page failed: %v", err)
|
||||
return
|
||||
}
|
||||
archiveBuffer.Reset()
|
||||
}
|
||||
|
||||
func (o *Outter) generateTags() {
|
||||
tagsTemplate := template.New("tags")
|
||||
var tagsBuffer = new(bytes.Buffer)
|
||||
|
||||
// 按标签归档
|
||||
var mTag = make(map[string][]*Post)
|
||||
for _, post := range o.Posts {
|
||||
for _, tag := range post.Tags {
|
||||
newPost := post
|
||||
newPost.MD = ""
|
||||
mTag[tag] = append(mTag[tag], newPost)
|
||||
}
|
||||
}
|
||||
|
||||
for tag, posts := range mTag {
|
||||
// 按时间归档
|
||||
var m = make(map[string][]*Post)
|
||||
for _, post := range posts {
|
||||
newPost := post
|
||||
newPost.MD = ""
|
||||
m[int2String(post.CreatedAt.Year())] = append(m[int2String(post.CreatedAt.Year())], newPost)
|
||||
}
|
||||
|
||||
var postData []*PostData
|
||||
for year, _posts := range m {
|
||||
postData = append(postData, &PostData{
|
||||
Key: year,
|
||||
Posts: _posts,
|
||||
})
|
||||
}
|
||||
|
||||
sort.Slice(postData, func(i, j int) bool {
|
||||
return postData[i].Key > postData[j].Key
|
||||
})
|
||||
|
||||
var instance = PageOutter{
|
||||
Page: &Page{
|
||||
Title: tag,
|
||||
Link: tag,
|
||||
},
|
||||
Config: o.Config,
|
||||
PostData: postData,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return instance.Page.Title
|
||||
}
|
||||
funcMap["page_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
tagsTemplate.Funcs(funcMap)
|
||||
tagsTemplate, err := tagsTemplate.Parse(string(o.Theme.TagLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate tags page failed: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
// 入buffer
|
||||
if err := tagsTemplate.Execute(tagsBuffer, instance); err != nil {
|
||||
logger.Errorf("generate tags page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 创建tag文件夹
|
||||
var tagDir = BaseDir + "/dist/tags"
|
||||
if !isExist(tagDir) {
|
||||
if err := mkdir(tagDir); err != nil {
|
||||
logger.Errorf("create tag directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
// buffer写文件
|
||||
var filename = fmt.Sprintf("%s/%s.html", tagDir, instance.Page.Link)
|
||||
if err := os.WriteFile(filename, tagsBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write tag content failed: %v", err)
|
||||
return
|
||||
}
|
||||
tagsBuffer.Reset()
|
||||
}
|
||||
}
|
||||
|
||||
func (o *Outter) generateCategories() {
|
||||
tagsTemplate := template.New("categories")
|
||||
var tagsBuffer = new(bytes.Buffer)
|
||||
|
||||
// 按标签归档
|
||||
var cats = make(map[string][]*Post)
|
||||
for _, post := range o.Posts {
|
||||
newPost := post
|
||||
cats[post.Category] = append(cats[post.Category], newPost)
|
||||
}
|
||||
cats["draft"] = o.DraftPosts
|
||||
|
||||
for tag, posts := range cats {
|
||||
// 按时间归档
|
||||
var m = make(map[string][]*Post)
|
||||
for _, post := range posts {
|
||||
newPost := post
|
||||
newPost.MD = ""
|
||||
m[int2String(post.CreatedAt.Year())] = append(m[int2String(post.CreatedAt.Year())], newPost)
|
||||
}
|
||||
|
||||
var postData []*PostData
|
||||
for year, _posts := range m {
|
||||
postData = append(postData, &PostData{
|
||||
Key: year,
|
||||
Posts: _posts,
|
||||
})
|
||||
}
|
||||
|
||||
sort.Slice(postData, func(i, j int) bool {
|
||||
return postData[i].Key > postData[j].Key
|
||||
})
|
||||
|
||||
var instance = PageOutter{
|
||||
Page: &Page{
|
||||
Title: tag,
|
||||
Link: tag,
|
||||
},
|
||||
Config: o.Config,
|
||||
PostData: postData,
|
||||
}
|
||||
instance.Outter = o
|
||||
funcMap["title"] = func() string {
|
||||
return instance.Page.Title
|
||||
}
|
||||
funcMap["page_name"] = func() string {
|
||||
return o.Config.Site.Title
|
||||
}
|
||||
tagsTemplate.Funcs(funcMap)
|
||||
tagsTemplate, err := tagsTemplate.Parse(string(o.Theme.TagLayout))
|
||||
if err != nil {
|
||||
logger.Errorf("generate tags page failed: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
// 入buffer
|
||||
if err := tagsTemplate.Execute(tagsBuffer, instance); err != nil {
|
||||
logger.Errorf("generate tags page failed: %v", err)
|
||||
return
|
||||
}
|
||||
// 创建tag文件夹
|
||||
var tagDir = BaseDir + "/dist/category"
|
||||
if !isExist(tagDir) {
|
||||
if err := mkdir(tagDir); err != nil {
|
||||
logger.Errorf("create tag directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
// buffer写文件
|
||||
var filename = fmt.Sprintf("%s/%s.html", tagDir, instance.Page.Link)
|
||||
if err := os.WriteFile(filename, tagsBuffer.Bytes(), os.ModePerm); err != nil {
|
||||
logger.Errorf("write tag content failed: %v", err)
|
||||
return
|
||||
}
|
||||
tagsBuffer.Reset()
|
||||
}
|
||||
}
|
||||
|
||||
type PostOutter struct {
|
||||
Post *Post
|
||||
Config *Config
|
||||
*Outter
|
||||
}
|
||||
|
||||
type PageOutter struct {
|
||||
Page *Page
|
||||
Config *Config
|
||||
PostData []*PostData
|
||||
*Outter
|
||||
}
|
||||
|
||||
type PostData struct {
|
||||
Key string
|
||||
Posts []*Post
|
||||
}
|
||||
|
||||
func (o *Outter) createDir(fp string) error {
|
||||
if !isExist(fp) {
|
||||
if err := mkdir(fp); err != nil {
|
||||
logger.Errorf("create %s directory failed: %v", fp, err)
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
var funcMap = template.FuncMap{
|
||||
"getSource": getSource,
|
||||
"add": add,
|
||||
"sum": sum,
|
||||
}
|
||||
|
||||
func getSource(data interface{}, key string) interface{} {
|
||||
source, ok := data.(map[string]interface{})
|
||||
if !ok {
|
||||
return nil
|
||||
}
|
||||
return source[key]
|
||||
}
|
||||
|
||||
func add(a, b int) int {
|
||||
return a + b
|
||||
}
|
||||
|
||||
func sum(a, b int) int {
|
||||
return a - b
|
||||
}
|
51
go.mod
51
go.mod
@ -1,10 +1,49 @@
|
||||
module mder
|
||||
module gitter.top/mder/mder
|
||||
|
||||
go 1.18
|
||||
|
||||
require github.com/spf13/cobra v1.5.0
|
||||
go 1.20
|
||||
|
||||
require (
|
||||
github.com/inconshreveable/mousetrap v1.0.0 // indirect
|
||||
github.com/spf13/pflag v1.0.5 // indirect
|
||||
github.com/gin-gonic/gin v1.9.1
|
||||
github.com/guonaihong/gout v0.3.9
|
||||
github.com/radovskyb/watcher v1.0.7
|
||||
github.com/sirupsen/logrus v1.9.3
|
||||
github.com/spf13/cobra v1.8.0
|
||||
github.com/yuin/goldmark v1.6.0
|
||||
github.com/yuin/goldmark-emoji v1.0.2
|
||||
github.com/yuin/goldmark-meta v1.1.0
|
||||
go.abhg.dev/goldmark/mermaid v0.5.0
|
||||
go.abhg.dev/goldmark/toc v0.9.0
|
||||
gopkg.in/yaml.v3 v3.0.1
|
||||
)
|
||||
|
||||
require (
|
||||
github.com/andybalholm/brotli v1.1.0 // indirect
|
||||
github.com/bytedance/sonic v1.10.2 // indirect
|
||||
github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect
|
||||
github.com/chenzhuoyu/iasm v0.9.1 // indirect
|
||||
github.com/gabriel-vasile/mimetype v1.4.3 // indirect
|
||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
||||
github.com/go-playground/locales v0.14.1 // indirect
|
||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
||||
github.com/go-playground/validator/v10 v10.17.0 // indirect
|
||||
github.com/goccy/go-json v0.10.2 // indirect
|
||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||
github.com/json-iterator/go v1.1.12 // indirect
|
||||
github.com/klauspost/cpuid/v2 v2.2.6 // indirect
|
||||
github.com/leodido/go-urn v1.3.0 // indirect
|
||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||
github.com/pelletier/go-toml/v2 v2.1.1 // indirect
|
||||
github.com/pkg/errors v0.9.1 // indirect
|
||||
github.com/spf13/pflag v1.0.5 // indirect
|
||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||
github.com/ugorji/go/codec v1.2.12 // indirect
|
||||
golang.org/x/arch v0.7.0 // indirect
|
||||
golang.org/x/crypto v0.18.0 // indirect
|
||||
golang.org/x/net v0.20.0 // indirect
|
||||
golang.org/x/sys v0.16.0 // indirect
|
||||
golang.org/x/text v0.14.0 // indirect
|
||||
google.golang.org/protobuf v1.32.0 // indirect
|
||||
gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||
)
|
||||
|
25
init.go
25
init.go
@ -1,25 +0,0 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
func initCmd() *cobra.Command {
|
||||
var name string
|
||||
cmd := &cobra.Command{
|
||||
Use: "init",
|
||||
Short: "create a new mder folder",
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
if err := cloneTemplate(name); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
_, _ = fmt.Fprintf(os.Stdout, "create folder %s success.\n", name)
|
||||
},
|
||||
}
|
||||
|
||||
cmd.Flags().StringVar(&name, "name", "mder", "Name of the folder to create")
|
||||
return cmd
|
||||
}
|
39
init_command.go
Normal file
39
init_command.go
Normal file
@ -0,0 +1,39 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"regexp"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
func initCmd() *cobra.Command {
|
||||
var name string
|
||||
cmd := &cobra.Command{
|
||||
Use: "init",
|
||||
Short: "create a new mder folder",
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
if name == "" && len(args) == 0 {
|
||||
logger.Errorf("folder name empty")
|
||||
return
|
||||
}
|
||||
if name == "" && len(args) != 0 {
|
||||
name = args[0]
|
||||
}
|
||||
var rule = fmt.Sprintf(`[A-Za-z0-9_]{%d}`, len([]rune(name)))
|
||||
var reg = regexp.MustCompilePOSIX(rule)
|
||||
if !reg.MatchString(name) {
|
||||
logger.Errorf("folder name rule must be: %s", rule)
|
||||
return
|
||||
}
|
||||
if err := cloneTemplate(name); err != nil {
|
||||
logger.Errorf("clone template repository failed: %v", err)
|
||||
return
|
||||
}
|
||||
logger.Infof("create folder `%s` success", name)
|
||||
},
|
||||
}
|
||||
|
||||
cmd.Flags().StringVar(&name, "name", "", "name of the folder to create")
|
||||
return cmd
|
||||
}
|
36
logger.go
Normal file
36
logger.go
Normal file
@ -0,0 +1,36 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/sirupsen/logrus"
|
||||
)
|
||||
|
||||
var logger = logrus.New()
|
||||
|
||||
func init() {
|
||||
logger.SetReportCaller(true)
|
||||
logger.SetFormatter(&EasyFormatter{})
|
||||
}
|
||||
|
||||
type EasyFormatter struct{}
|
||||
|
||||
func (receiver *EasyFormatter) Format(entry *logrus.Entry) ([]byte, error) {
|
||||
var output bytes.Buffer
|
||||
output.WriteString(entry.Level.String()[:4])
|
||||
output.WriteString("|")
|
||||
var filenames = strings.Split(entry.Caller.File, "/")
|
||||
filename := filenames[len(filenames)-1]
|
||||
output.WriteString(fmt.Sprintf("%s:%d", filename, entry.Caller.Line))
|
||||
output.WriteString("|")
|
||||
output.WriteString(entry.Caller.Function)
|
||||
output.WriteString("|")
|
||||
for k, val := range entry.Data {
|
||||
output.WriteString(fmt.Sprintf("%s=%v|", k, val))
|
||||
}
|
||||
output.WriteString(" " + entry.Message)
|
||||
output.WriteRune('\n')
|
||||
return output.Bytes(), nil
|
||||
}
|
104
new_command.go
Normal file
104
new_command.go
Normal file
@ -0,0 +1,104 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/spf13/cobra"
|
||||
)
|
||||
|
||||
func newCmd() *cobra.Command {
|
||||
cmd := &cobra.Command{
|
||||
Use: "new",
|
||||
Short: "new a post or page",
|
||||
}
|
||||
|
||||
cmd.AddCommand(newPostCmd())
|
||||
cmd.AddCommand(newPageCmd())
|
||||
|
||||
return cmd
|
||||
}
|
||||
|
||||
func newPostCmd() *cobra.Command {
|
||||
var name string
|
||||
cmd := &cobra.Command{
|
||||
Use: "post",
|
||||
Short: "new a post",
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
if name == "" && len(args) != 0 {
|
||||
name = args[0]
|
||||
}
|
||||
var pureName = strings.ReplaceAll(name, ".md", "")
|
||||
pureName = strings.ReplaceAll(name, "-", " ")
|
||||
// 空格处理
|
||||
name = strings.ReplaceAll(name, " ", "-")
|
||||
if !strings.HasSuffix(name, ".md") {
|
||||
name = name + ".md"
|
||||
}
|
||||
filename := fmt.Sprintf("posts/%s", name)
|
||||
// 检测文件夹是否存在
|
||||
dir := filepath.Dir(filename)
|
||||
if !isExist(dir) {
|
||||
if err := mkdir(dir); err != nil {
|
||||
logger.Errorf("make directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, os.ModePerm)
|
||||
if err != nil {
|
||||
logger.Errorf("create file failed: %v", err)
|
||||
return
|
||||
}
|
||||
defer f.Close()
|
||||
var data = fmt.Sprintf("---\ntitle: %s\ndate: %s\ncatagories:\ntags:\n---", pureName, time.Now().Format("2006-01-02 15:04:05"))
|
||||
if _, err := f.Write([]byte(data)); err != nil {
|
||||
logger.Errorf("create file failed: %v", err)
|
||||
return
|
||||
}
|
||||
},
|
||||
}
|
||||
cmd.Flags().StringVar(&name, "name", "uname.md", "Name of the post file to create")
|
||||
return cmd
|
||||
}
|
||||
|
||||
func newPageCmd() *cobra.Command {
|
||||
var name string
|
||||
cmd := &cobra.Command{
|
||||
Use: "page",
|
||||
Short: "new a page",
|
||||
Run: func(cmd *cobra.Command, args []string) {
|
||||
if name == "" && len(args) != 0 {
|
||||
name = args[0]
|
||||
}
|
||||
var pureName = strings.ReplaceAll(name, ".md", "")
|
||||
if !strings.HasSuffix(name, ".md") {
|
||||
name = name + ".md"
|
||||
}
|
||||
filename := fmt.Sprintf("pages/%s", name)
|
||||
// 检测文件夹是否存在
|
||||
dir := filepath.Dir(filename)
|
||||
if !isExist(dir) {
|
||||
if err := mkdir(dir); err != nil {
|
||||
logger.Errorf("make directory failed: %v", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE, os.ModePerm)
|
||||
if err != nil {
|
||||
logger.Errorf("create file failed: %v", err)
|
||||
return
|
||||
}
|
||||
defer f.Close()
|
||||
var data = fmt.Sprintf("---\ntitle: %s\ndate: %s\n---", pureName, time.Now().Format("2006-01-02 15:04:05"))
|
||||
if _, err := f.Write([]byte(data)); err != nil {
|
||||
logger.Errorf("create file failed: %v", err)
|
||||
return
|
||||
}
|
||||
},
|
||||
}
|
||||
cmd.Flags().StringVar(&name, "name", "uname.md", "Name of the page file to create")
|
||||
return cmd
|
||||
}
|
85
serve_command.go
Normal file
85
serve_command.go
Normal file