From e87fa4cbb5d851caf5978c5c5c3aacc977b77e2d Mon Sep 17 00:00:00 2001 From: caixw Date: Wed, 18 Dec 2019 00:20:57 +0800 Subject: [PATCH] =?UTF-8?q?=E6=9B=B4=E6=96=B0=E8=87=B3=20v5.2.1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit favicon 和 logo 固定为 SVG 格式,Firefox 支持 SVG 图片作为 favicon, 而 chrome 将在 80 版本支持 SVG,safari 也支持使用 mask-icon 作为图片。 修正 static.Pack 可能的错误,不再允许打包目标文件夹中的二进制文件, 将 favicon 允许的格式改为 SVG,也将唯一的进制文件 icon.png 去掉了。 --- .gitignore | 3 +- CHANGELOG.md | 10 +- apidoc.go | 2 +- config.go | 2 +- config_test.go | 94 +++++++++++++++---- docs/example/.apidoc.yaml | 2 +- docs/example/index.xml | 2 +- docs/icon.png | Bin 2953 -> 0 bytes docs/index.xml | 2 +- docs/index.xsl | 3 +- docs/index.zh-hant.xml | 2 +- docs/v5/apidoc.xsl | 5 +- internal/vars/version.go | 2 +- static/pack.go | 61 ++++-------- static/pack_test.go | 34 +++---- static/static.go | 10 +- static/testdir/testdir1/{file1 => file1.xml} | 0 static/testdir/testdir2/{file2 => file2.xsl} | 0 18 files changed, 139 insertions(+), 95 deletions(-) delete mode 100644 docs/icon.png rename static/testdir/testdir1/{file1 => file1.xml} (100%) rename static/testdir/testdir2/{file2 => file2.xsl} (100%) diff --git a/.gitignore b/.gitignore index 2c17577e..d754ef7f 100644 --- a/.gitignore +++ b/.gitignore @@ -19,10 +19,11 @@ dist # project .apidoc.yaml .apidoc.yml -/.vscode apidoc.json apidoc.yaml apidoc.xml .vscode .idea + +.testdata diff --git a/CHANGELOG.md b/CHANGELOG.md index 850941a7..123c170c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,14 @@ # CHANGELOG -## [Unreleased] +## [v5.2.1] + +### Changed + +- favicon 现在只支持 SVG 格式的图片; + +### Fixed + +- 修正 Pack 可能将二进制等文件进行打包的错误; ## [v5.2.0] diff --git a/apidoc.go b/apidoc.go index b1b5f6b2..794e01c3 100644 --- a/apidoc.go +++ b/apidoc.go @@ -106,7 +106,7 @@ func Pack(h *message.Handler, url string, contentType, pkgName, varName, path st contentType = http.DetectContentType(data) } - return static.Pack("./docs", pkgName, varName, path, t, nil, &static.FileInfo{ + return static.Pack("./docs", pkgName, varName, path, t, &static.FileInfo{ Name: url, Content: data, ContentType: contentType, diff --git a/config.go b/config.go index e42615d3..025a39cb 100644 --- a/config.go +++ b/config.go @@ -48,7 +48,7 @@ type Config struct { // LoadConfig 加载指定目录下的配置文件 // -// 所有的错误信息会输出到 h +// 所有的错误信息会输出到 h,在出错时,会返回 nil func LoadConfig(h *message.Handler, wd string) *Config { for _, filename := range vars.AllowConfigFilenames { p := filepath.Join(wd, filename) diff --git a/config_test.go b/config_test.go index 8f7855b0..60907474 100644 --- a/config_test.go +++ b/config_test.go @@ -5,7 +5,6 @@ package apidoc import ( "bytes" "path/filepath" - "strconv" "testing" "time" @@ -14,15 +13,40 @@ import ( "github.com/caixw/apidoc/v5/input" "github.com/caixw/apidoc/v5/internal/vars" "github.com/caixw/apidoc/v5/message" + "github.com/caixw/apidoc/v5/static" ) -func buildMessageHandle() (*bytes.Buffer, message.HandlerFunc) { - buf := new(bytes.Buffer) - - return buf, func(msg *message.Message) { - buf.WriteString(strconv.Itoa(int(msg.Type))) - buf.WriteString(msg.Message) +func buildMessageHandle() (erro, succ *bytes.Buffer, h *message.Handler) { + erro = new(bytes.Buffer) + succ = new(bytes.Buffer) + + f := func(msg *message.Message) { + switch msg.Type { + case message.Erro: + erro.WriteString(msg.Message) + default: + succ.WriteString(msg.Message) + } } + + return erro, succ, message.NewHandler(f) +} + +func TestLoadConfig(t *testing.T) { + a := assert.New(t) + + erro, succ, h := buildMessageHandle() + cfg := LoadConfig(h, "./") + a.NotNil(cfg). + Empty(erro.String()) + + erro, succ, h = buildMessageHandle() + cfg = LoadConfig(h, "./docs") // 不存在 apidoc 的配置文件 + + h.Stop() + a.Nil(cfg). + NotEmpty(erro.String()). + Empty(succ.String()) } func TestLoadFile(t *testing.T) { @@ -42,9 +66,9 @@ func TestDetect_Load(t *testing.T) { a.NotError(err).NotEmpty(wd) a.NotError(Detect(wd, true)) - out, f := buildMessageHandle() - cfg := LoadConfig(message.NewHandler(f), wd) - a.Empty(out.String()).NotNil(cfg) + erro, _, h := buildMessageHandle() + cfg := LoadConfig(h, wd) + a.Empty(erro.String()).NotNil(cfg) a.Equal(cfg.Version, vars.Version()). Equal(cfg.Inputs[0].Lang, "go") @@ -78,21 +102,55 @@ func TestConfig_sanitize(t *testing.T) { Equal(err.Field, "output") } +func TestConfig_Test(t *testing.T) { + a := assert.New(t) + + erro, succ, h := buildMessageHandle() + cfg := LoadConfig(h, "./docs/example") + a.NotNil(cfg) + cfg.Test() + + h.Stop() + a.Empty(erro.String()). + NotEmpty(succ.String()) // 有成功提示 +} + +func TestConfig_Pack(t *testing.T) { + a := assert.New(t) + + erro, succ, h := buildMessageHandle() + cfg := LoadConfig(h, "./docs/example") + a.NotNil(cfg) + cfg.Pack("testdata", "Data", "./.testdata", "apidoc.xml", "application/xml", static.TypeAll) + + h.Stop() + a.Empty(erro.String()). + Empty(succ.String()) +} + func TestConfig_Do(t *testing.T) { a := assert.New(t) - out, f := buildMessageHandle() - h := message.NewHandler(f) - LoadConfig(h, "./docs/example").Do(time.Now()) - a.Empty(out.String()) + erro, succ, h := buildMessageHandle() + cfg := LoadConfig(h, "./docs/example") + a.NotNil(cfg) + cfg.Do(time.Now()) + + h.Stop() + a.NotEmpty(succ.String()). // 有成功提示 + Empty(erro.String()) } func TestConfig_Buffer(t *testing.T) { a := assert.New(t) - out, f := buildMessageHandle() - h := message.NewHandler(f) - buf := LoadConfig(h, "./docs/example").Buffer() - a.Empty(out.String()). + erro, succ, h := buildMessageHandle() + cfg := LoadConfig(h, "./docs/example") + a.NotNil(cfg) + + buf := cfg.Buffer() + h.Stop() + a.Empty(erro.String()). + Empty(succ.String()). True(buf.Len() > 0) } diff --git a/docs/example/.apidoc.yaml b/docs/example/.apidoc.yaml index f4ec2640..5460b4a2 100755 --- a/docs/example/.apidoc.yaml +++ b/docs/example/.apidoc.yaml @@ -1,4 +1,4 @@ -version: 5.0.0 +version: 5.2.0 inputs: - lang: c++ dir: . diff --git a/docs/example/index.xml b/docs/example/index.xml index e16a5bac..d90e3947 100755 --- a/docs/example/index.xml +++ b/docs/example/index.xml @@ -1,7 +1,7 @@ - + 示例文档 5SsEn;Lf2>}7I zS&MWvO*KJ9)5vA!3^gH`2Ei}QqOL9*BgA?u%$X>r2zXhECS3%rRw!O#T2o^pQAzDe zR}0>_iArFGIlCX;edf#!n9DiS|CjT=d%tHtmwkKo^E^AKF=B%c$jroKOvD6?#u<2j z`s`gC!~yKW%YvPP9bug}Ysp`zMWb&&UNIa8uZc#{`=~v*)V9U?)^f0dc8O zj%vapoY$%9J;76=pDFilowsnnUhjdkoQBcbjm3F_iMT*C1E+Or>sO-n+BJu}y;BXo z2d#j(*qQ2Fi?7OuHoPv{BzOov4MlAOYP(=KK8Z_lB|ea<4&XL*ZVhW47!3adRzM_Z zvPg6j#-zgSg4M=*G^}!LFpTw}x5_ZP%@r)itW^A#=z1;IhkUy98v{~6T~CBBRzPfYiQsPeD*l?_`q1nj``b=6 z5Sx5XbSFN9cSMEI;=ca0(Pst3RWf_5#@AEP{YEH++a2l;!D=HGoQ@XEOGRrGUDwws z(I*AOdFq|v9>IJZ6MQYc9zk5SRG4l1Fg1V56g!8*a8>rf8DjJe*VhKE#5@1fxags_e7f(I&IfK6i!( zwGM?KgF!5qC%RWqf1FPa<5e8!vIfKj&TxteqRFDGRrcF5_ToO7=CIC-)n|R+NS5d# zT#fU)RsS5D1<#9K={(hYT?ErU{~S&i{k7n?_%f!HKfw=0H-;AfUK5G>i1-y_v{{Cw z@~s*63+_X!=vut3&IMtu|LXq#9tud?-BXx``$N$c9d%7UCTft|`|nG|TeQ1&AV$bw z5EnQ_y_Hz5a;n@Wy4^ur%2PEKJb-zECpDYZeUa~}fPx13zPiTusX%XtOHZ<483H?G3TP9GN>+4qcB6z8RYRq!(UK$4eL|S{Pc~)49Iu8jv4sc43U* ziqP!-6Ec!3;wCIiZSBJRQ1l-Ilr z+?2NTvFa`BQrAHsUqyd|(K0jo(r_0Bmm1-y)$*wU_iMGF8dD-Rx=PT3Q}Kd2mxr}p z=~o*~riq@xe#iV?*`H7u0Y?<%=lN~@6&}Lvj)tP8qA%iYTyD6nX$}q;vB_TvHpqo< z*6LIJ6&}+1umfirrcf@gC?Hsrp4JS8;n3pyqUrcw!E9`g*HkR8z9QyE+D)Q0`Kx82 zXvv^PRIoZdEi4yhBp)7=@P_do9Wa41ofbPC^GW%-aK2+6kBgnzuSN=P#0svYdu8V0zcItRHpf-ncAjpCO)eLF2T`yn6m6*{Pf8xj z3FR_paf!3>Q5?3f`etodXE)|zpWyR$TY0>1V}nVeyW}17jnM4gYV+fupU6+WQ{qBn zGbOVS1W(rLq!)@_R&p8I1vkV77k2+9E^vy>W}GJ49ExtK2`>+7P1|8G%*;$VaFP4% zYliS$PvBb^CbPN6qFC=%d<-u+qOq2|kSx{UqnSKhMd<(D+HZ`v3QysK>fBf^js-K& zh$AwWg$*fSQY&Oh)?i{L_y}mXtENPyRXAFVqiBo`J{4CPhC2{Me>Wt4hEh^MG$9k6 zm0n(}IbnnryV5G3sdIGL=4;iAnWa3uop_(o zK2d811k=;gwoYZ`=N`x8s-w*`Q zU?2~~d?dYeKYr&5Db;W{+U-x|IS?34m(sG@*&&GHP)S;<3FeGKm~LErABwp>;dlRP$>g&EkdO-s4FLjfsPmIoh# zzo`WiDslYYd=E7{lf>NVdqaBt`_6gB-J`>|%F|Eh#HJe9<}nEudhtDk#8j-o2xxK@ zX3C>NT_Q|f6cE-rAi6#=vSzS>%^nCHk*M|_b>*@w9g(YqxWE~rn-l#Nq1bi$^>m1| zSTA}S;{?TOG9#v*554#vvefgXD!by3tKC<>Oc~i(dSIDW!Lp1=zCKtd1z)7UxZu5X4khRJCAqT)KmGkf2NML1t`>Y- zzUaUX!D{1dJyEh%Z1gR`O5BMJ@}a+5(T9%Ld$C}+=(94*#SUe^Ay}c=h9SK+*M+fs z4lcvj=*aqbQ?MC7Q17SVc1OGXr_uR>=doY(fM5yMgko1;L|kfwHq(?`g83?n-iKRp zTX);Ozo_2T*+oVgu2Hl|*ZabUMNbQUCZwygnK!iiQTpy3`O5vLq60D$l}t>o>*;h! zt$Pu78>S`vLv<|1+FYJ4xIi#JrJ_w41f7yR+47%7n?j49SB=Spnkb+%;zDE9yHGS) z?hu_IpDLr3?-~ccO&Kctb-q3y?dTLcs0scLss@As$IO`400000NkvXXu0mjfV(gv` diff --git a/docs/index.xml b/docs/index.xml index 63cc0bd0..e60f1fb4 100644 --- a/docs/index.xml +++ b/docs/index.xml @@ -121,7 +121,7 @@

用于描述整个文档的相关内容,只能出现一次。

文档的版本 内容的本地化 ID,比如 zh-hans 等。 - 图标 + 图标,默认采用官网的 https://apidoc.tools/icon.svg,同时作用于 favicon 和 logo,只支持 SVG 格式。 文档的生成时间 文档的标题 文档的整体介绍,可以是使用 HTML 内容。 diff --git a/docs/index.xsl b/docs/index.xsl index f6fd1df0..7c3fa42d 100644 --- a/docs/index.xsl +++ b/docs/index.xsl @@ -32,7 +32,8 @@ - + + diff --git a/docs/index.zh-hant.xml b/docs/index.zh-hant.xml index 72af14ec..da5f0b88 100644 --- a/docs/index.zh-hant.xml +++ b/docs/index.zh-hant.xml @@ -121,7 +121,7 @@

用於描述整個文檔的相關內容,只能出現壹次。

文檔的版本 內容的本地化 ID,比如 zh-hans 等。 - 圖標 + 圖標,默認采用官網的 https://apidoc.tools/icon.svg,同時作用於 favicon 和 logo,只支持 SVG 格式。 文檔的生成時間 文檔的標題 文檔的整體介紹,可以是使用 HTML 內容。 diff --git a/docs/v5/apidoc.xsl b/docs/v5/apidoc.xsl index 3db15837..0cb7499f 100644 --- a/docs/v5/apidoc.xsl +++ b/docs/v5/apidoc.xsl @@ -17,7 +17,8 @@ - + + @@ -509,7 +510,7 @@ - + diff --git a/internal/vars/version.go b/internal/vars/version.go index 52264a19..6d3903d6 100644 --- a/internal/vars/version.go +++ b/internal/vars/version.go @@ -8,7 +8,7 @@ import "strings" // // 遵守 https://semver.org/lang/zh-CN/ 规则。 // 程序不兼容或是文档格式不兼容时,需要提升主版本号。 -const version = "5.2.0" +const version = "5.2.1" var ( fullVersion = version diff --git a/static/pack.go b/static/pack.go index 215cd863..865bb3b4 100644 --- a/static/pack.go +++ b/static/pack.go @@ -3,25 +3,24 @@ package static import ( - "bufio" "bytes" "io/ioutil" "net/http" "os" "path/filepath" - "strings" - "unicode" "github.com/issue9/utils" - - "github.com/caixw/apidoc/v5/internal/locale" - "github.com/caixw/apidoc/v5/message" ) -const goModPath = "../go.mod" +const modulePath = "github.com/caixw/apidoc/v5/static" const header = "// 当前文件由工具自动生成,请勿手动修改!\n\n" +var allowPackExts = []string{ + ".xml", ".xsl", ".svg", + ".css", ".js", ".html", ".htm", +} + // FileInfo 被打包文件的信息 type FileInfo struct { // 相对于打包根目录的地址,同时也会被作为路由地址 @@ -39,6 +38,8 @@ type FileInfo struct { // path 内容保存的文件名; // t 打包的文件类型,如果为 TypeNone,则只打包 addTo 的内容; // addTo 追加的打包内容; +// +// NOTE: 隐藏文件不会被打包 func Pack(root, pkgName, varName, path string, t Type, addTo ...*FileInfo) error { fis, err := getFileInfos(root, t) if err != nil { @@ -58,11 +59,7 @@ func Pack(root, pkgName, varName, path string, t Type, addTo ...*FileInfo) error ws("package ", pkgName, "\n\n") - goMod, err := getPkgPath(goModPath) - if err != nil { - return err - } - ws("import \"", goMod+"/static", "\"\n\n") + ws("import \"", modulePath, "\"\n\n") ws("var ", varName, "= []*static.FileInfo{") for _, info := range fis { @@ -82,24 +79,25 @@ func getFileInfos(root string, t Type) ([]*FileInfo, error) { return nil, nil } - paths := []string{} + var paths []string walk := func(path string, info os.FileInfo, err error) error { if err != nil { return err } - if info.IsDir() { + // 过滤各类未知的隐藏文件 + if info.IsDir() || !isAllowPackFile(filepath.Ext(info.Name())) { return nil } - relpath, err := filepath.Rel(root, path) + relPath, err := filepath.Rel(root, path) if err != nil { return err } - if t != TypeStylesheet || isStylesheetFile(relpath) { - paths = append(paths, relpath) + if t != TypeStylesheet || isStylesheetFile(relPath) { + paths = append(paths, relPath) } return nil @@ -144,30 +142,11 @@ func dump(buf *bytes.Buffer, file *FileInfo) (err error) { return err } -const modulePrefix = "module" - -// 分析 go.mod 文件,获取其中的 module 值 -func getPkgPath(path string) (string, error) { - file, err := os.Open(path) - if err != nil { - return "", err - } - - s := bufio.NewScanner(bufio.NewReader(file)) - s.Split(bufio.ScanLines) - for s.Scan() { - line := strings.TrimSpace(s.Text()) - if !strings.HasPrefix(line, modulePrefix) { - continue +func isAllowPackFile(ext string) bool { + for _, e := range allowPackExts { + if e == ext { + return true } - - line = line[len(modulePrefix):] - if line == "" || !unicode.IsSpace(rune(line[0])) { - continue - } - - return strings.TrimSpace(line), nil } - - return "", message.NewLocaleError(goModPath, "", 0, locale.ErrInvalidFormat) + return false } diff --git a/static/pack_test.go b/static/pack_test.go index 8ded8bf1..724956a1 100644 --- a/static/pack_test.go +++ b/static/pack_test.go @@ -3,12 +3,24 @@ package static import ( + "path" "path/filepath" + "strings" "testing" "github.com/issue9/assert" + + "github.com/caixw/apidoc/v5/internal/vars" ) +// 保证 modulePath 的值正确性 +func TestModulePath(t *testing.T) { + a := assert.New(t) + + suffix := path.Join(vars.DocVersion(), "static") + a.True(strings.HasSuffix(modulePath, suffix)) +} + func TestPack(t *testing.T) { a := assert.New(t) a.NotError(Pack("./testdir", "testdata", "Data", "./testdata/testdata.go", TypeAll)) @@ -19,14 +31,14 @@ func TestGetFileInfos(t *testing.T) { info, err := getFileInfos("./testdir", TypeAll) a.NotError(err).NotNil(info) - a.Equal(6, len(info)) + a.Equal(3, len(info)) - // 采有绝对路径 + // 采用绝对路径 dir, err := filepath.Abs("./testdir") a.NotError(err).NotEmpty(dir) info, err = getFileInfos(dir, TypeAll) a.NotError(err).NotNil(info) - a.Equal(6, len(info)) + a.Equal(3, len(info)) } func TestGetFileInfos_TypeStylesheet(t *testing.T) { @@ -45,19 +57,3 @@ func TestGetFileInfos_TypeNone(t *testing.T) { a.NotError(err).Nil(info) a.Equal(0, len(info)) } - -func TestGetPkgPath(t *testing.T) { - a := assert.New(t) - - p, err := getPkgPath("") - a.Error(err).Empty(p) - - p, err = getPkgPath("./testdir/go.mod1") - a.NotError(err).Equal(p, "test/v6") - - p, err = getPkgPath("./testdir/go.mod2") - a.Error(err).Empty(p) - - p, err = getPkgPath("./testdir/go.mod3") - a.Error(err).Empty(p) -} diff --git a/static/static.go b/static/static.go index 3a5e2cff..38fabf61 100644 --- a/static/static.go +++ b/static/static.go @@ -13,20 +13,20 @@ import ( "github.com/caixw/apidoc/v5/internal/vars" ) -// Type 表示文件的类型 +// Type 表示对打包文件的分类 type Type int8 // 几种文件类型的定义 const ( - TypeNone = iota - TypeAll - TypeStylesheet + TypeNone Type = iota // 不包含任何文件 + TypeAll // 所有文件 + TypeStylesheet // 仅与 xsl 相关的文件 ) // 默认页面 const indexPage = "index.xml" -// 指定了 xml 文档所需的 xsl 内容。 +// 指定在 TypeStylesheet 下需要的文件列表。 // // 可以以前缀的方式指定,比如:v5/ 表示以 v5/ 开头的所有文件。 var styles = []string{ diff --git a/static/testdir/testdir1/file1 b/static/testdir/testdir1/file1.xml similarity index 100% rename from static/testdir/testdir1/file1 rename to static/testdir/testdir1/file1.xml diff --git a/static/testdir/testdir2/file2 b/static/testdir/testdir2/file2.xsl similarity index 100% rename from static/testdir/testdir2/file2 rename to static/testdir/testdir2/file2.xsl