GitBook 本地使用排雷,及导出基本可用的 PDF 版本

原文地址:https://alphahinex.github.io/2022/09/18/gitbook-export-fine-enough-pdf/

cover

description: "总结了离线环境使用 GitBook 以及将 GitBook 编写的文档导出成 PDF 时可能会遇到的问题和解决办法"
date: 2022.09.18 10:34
categories:
- Book
tags: [GitBook]
keywords: GitBook, PDF, graceful-fs, ebook-convert


GitBook 简介

GitBook 是一个现代的文档平台,提供了基于 Markdown 等方式的在线协作编辑文档方式,并可以方便的通过浏览器阅读文档内容。

有大量的公司、开源项目等,都在使用,如 GitBook 自己的文档Fluent Bit 等。

GitBook 团队曾经还提供过一个离线的命令行工具和 Node.js 类库 - gitbook,用来在本地离线环境使用 Markdown 或 AsciiDoc 构建一个电子书,遗憾的是目前这个项目已经被弃用了。

虽然 GitbookIO 的这个 gitbook 工具已经不再维护了,但其仍是目前通过 Markdown 编写可导出的电子书的比较好的工具,且简单易用。

本文针对使用 gitbook 工具编写、发布 HTML 版本和导出成 PDF 版本所遇到的一些问题进行总结,并给出解决方案,使我们可以利用 gitbook 获得一个还算够好的 PDF 版本电子书。

本地使用 gitbook

本地使用 GitBook 需要 Node.js 运行环境,可全局安装命令行工具:

$ npm install gitbook-cli -g

gitbook-cli 是一个安装和使用多个版本 GitBook(Node.js 类库)的工具,会自动安装所需要的 GitBook 版本。可通过如下命令查看 CLI 版本及自动安装 GitBook 最后发布版,一切顺利的话会看到:

$ gitbook --version
CLI version: 2.3.2
Installing GitBook 3.2.3
...
GitBook version: 3.2.3

如果不顺利,可以查看下面的 问题1 ,及解决方案。

常用命令:

# 初始化一个模板
$ gitbook init
# 本地启动一个 HTTP 服务进行预览
$ gitbook serve
# 编译静态网站,可将生成的 _book 路径中内容发布到 HTTP 服务中
$ gitbook build

如果不想全局安装,也可以仅安装至当前路径,并按如下方式使用:

$ npm i gitbook-cli
$ node node_modules/gitbook-cli/bin/gitbook.js fetch 3.2.3
$ node node_modules/gitbook-cli/bin/gitbook.js --version
CLI version: 2.3.2
GitBook version: 3.2.3

问题1:gitbook-cli 报错

在使用 gitbook-cli 时,可能会遇到如下问题:

$ gitbook --version
CLI version: 2.3.2
Installing GitBook 3.2.3
/Users/alphahinex/.nvm/versions/node/v12.22.12/lib/node_modules/gitbook-cli/node_modules/npm/node_modules/graceful-fs/polyfills.js:287
      if (cb) cb.apply(this, arguments)
                 ^

TypeError: cb.apply is not a function
    at /Users/alphahinex/.nvm/versions/node/v12.22.12/lib/node_modules/gitbook-cli/node_modules/npm/node_modules/graceful-fs/polyfills.js:287:18
    at FSReqCallback.oncomplete (fs.js:169:5)

原因是依赖的 graceful-fs 版本有问题,可按照 Gitbook build stopped to work in node 12.18.3 #110 中提供的方式,修改 gitbook-cli 所依赖的 graceful-fs 版本解决。

全局安装的 CLI 安装在 npm 配置的全局安装路径,如 ~/.nvm/versions/node/v12.22.12/lib/node_modules/gitbook-cli

$ cd ~/.nvm/versions/node/v12.22.12/lib/node_modules/gitbook-cli
$ npm i graceful-fs@4.1.4 --save
$ cd ~/.nvm/versions/node/v12.22.12/lib/node_modules/gitbook-cli/node_modules/npm
$ npm i graceful-fs@4.1.4 --save

注意:有两处版本需要修改:

  1. gitbook-cli 自身依赖的 graceful-fs
  2. gitbook-cli 依赖的 npm 中依赖的 graceful-fs

除上面方法外,也可以直接将报错位置代码注掉,或降低 gitbook-cli 版本解决此问题:

$ npm install -g gitbook-cli@2.2.0
$ gitbook fetch 3.2.3

问题2:如何解除对 README.md 文件的依赖

使用 gitbook init 生成模板工程后,会自动生成 SUMMARY.mdREADME.md 两个文件。SUMMARY.md 中内容为 GitBook 的目录,README.md 会被初始化为 Introduction 章节。这两个文件默认是必须存在的,即使在 SUMMARY.md 中去掉对 README.md 文件的依赖,使用 gitbook serve 等命令时也会提示找不到 README.md 文件。

可以按照官方文档中关于配置文件的说明,在项目根路径放置一个 book.json 文件,通过 structure.* 参数调整默认文件,以解除对 README.md 文件的依赖。

Variable Description
structure.readme Readme file name (defaults to README.md)
structure.summary Summary file name (defaults to SUMMARY.md)
structure.glossary Glossary file name (defaults to GLOSSARY.md)
structure.languages Languages file name (defaults to LANGS.md)

book.json 示例:

{
  "structure": {
    "readme": "rdm.md"
  }
}

问题3:如何自动生成 SUMMARY.md 的内容

目录内容及层级较多时,可通过一些工具自动生成 SUMMARY.md 文件,如 gitbook-summary

$ npm install -g gitbook-summary
$ book sm

问题4:如何添加插件

GitBook 可通过插件对功能进行扩展,如添加页面内导航、目录折叠等。

不过由于项目的弃用,如今已无法方便的搜索可用插件了。

可利用搜索引擎,搜索介绍 GitBook 插件的文章,实现曲线救国,如 Gitbook常用插件合集Gitbook 插件 等。

设置插件时,同样是在 book.json 中配置,如:

{
    "plugins": [
        "-lunr",
        "-search",
        "search-pro"
    ]
}

在添加插件后,需通过 gitbook install 命令将插件安装(下载)至本地。

注意:有些插件(如页面内导航等)会影响导出成 PDF 时生成的目录,建议在导出 PDF 文件时,尽量减少不必要的插件。

问题5:gitbook build 报缺少插件中的文件

在 CI/CD 环境执行 gitbook build 时,即使已通过 gitbook install 完成了插件的安装,也会有较大概率出现随机缺少插件中的某个文件的报错(报错信息如下),导致构建失败。

Error: ENOENT: no such file or directory, stat '/opt/buildagent/work/173f165a668e1392/_book/gitbook/gitbook-plugin-code/plugin.css'

遇到这个问题,需要修改 GitBook 中的 copyPluginAssets.js 文件中的内容。

GitBook 被安装在 ~/.gitbook/versions/ 路径的版本号文件夹下,如:~/.gitbook/versions/3.2.3

copyPluginAssets.js 文件的路径则为 ~/.gitbook/versions/3.2.3/lib/output/website/copyPluginAssets.js,需将此文件中的 confirm: true 替换为 confirm: false

# linux
$ sed -i "s#confirm: true#confirm: false#g" copyPluginAssets.js
# macOS
$ sed -i '' 's/confirm: true/confirm false/g' copyPluginAssets.js

GitBook 导出 PDF

之所以在 gitbook 项目被弃用了之后依然使用其编写文档,主要看重的是 GitBook 能够导出成电子书的能力。

GitBook 导出电子书,需要安装 Calibre 应用 ,并使命令行中可以使用 ebook-convert 工具。

Mac 上将 calibre.app 移至应用文件夹后,还需参照 官方文档 或如下内容,创建一个软链接,以便可以在终端中直接使用 ebook-convert

$ ln -s /Applications/calibre.app/Contents/MacOS/ebook-convert /usr/local/bin/ebook-convert
$ ebook-convert --version
ebook-convert (calibre 6.3.0)
Created by: Kovid Goyal <kovid@kovidgoyal.net>

Windows 上安装最新版 Calibre 后,可直接使用 ebook-convert,如果不行,可以在安装路径找到 ebook-convert.exe,并将其路径配置到环境变量的 Path 中。

ebook-convert 可用后,即可使用如下命令,将使用 GitBook 编写的文档,输出成 PDF 格式了:

$ gitbook pdf

一切看上去都很美,然而打开生成的 PDF 文件之后,会发现理想和现实还是有差距的。

wl-awesome 仓库中的 GitBook 为例,直接按上述方式生成的 PDF 电子书,打开后会发现存在不少问题:

  • 所有页的页眉都是第一个章节的名称
  • 点击目录书签只能跳转到文档中的目录页
  • 点击文档中的目录页则是打开浏览器跳转到一个无法访问的链接
  • 页边距过大,显得每页内容过少
  • 以电子书的角度来看,缺少封面
  • 生成的 PDF 文件名为 book.pdf
  • 字体过小
  • ……

这些问题,使得生成的 PDF 文件距离一本基本令人满意的电子书还有一定的距离。接下来让我们看看如何解决或缓解这些问题。

问题1:页眉问题

在 GitBook 的 文档 中,并未找到设置页眉相关的参数。但从依赖关系我们可以知道生成 PDF 使用的是 ebook-convert

通过查看 gitbook 的源码,在 ~/.gitbook/versions/3.2.3/lib/output/ebook/onFinish.jsrunEbookConvert 方法中,可以看到使用 ebook-convert 转换成 PDF 时所执行的命令:

var cmd = [
    'ebook-convert',
    path.resolve(outputFolder, SUMMARY_FILE),
    path.resolve(outputFolder, 'index.' + format),
    command.optionsToShellArgs(options)
].join(' ');

return command.exec(cmd)

其中的 options 是传给 ebook-convert 的参数。完整的可用参数列表,可以在 ebook-convert使用手册 中查阅。

继续查看源码,在 lib/output/ebook/getConvertOptions.js 中可以找到生成 PDF 文件所使用的参数。

经过试验,去掉如下四行代码的内容,即可使页眉正常显示所属章节的标题:

// '--chapter':                    'descendant-or-self::*[contains(concat(\' \', normalize-space(@class), \' \'), \' book-chapter \')]',
// '--level1-toc':                 'descendant-or-self::*[contains(concat(\' \', normalize-space(@class), \' \'), \' book-chapter-1 \')]',
// '--level2-toc':                 'descendant-or-self::*[contains(concat(\' \', normalize-space(@class), \' \'), \' book-chapter-2 \')]',
// '--level3-toc':                 'descendant-or-self::*[contains(concat(\' \', normalize-space(@class), \' \'), \' book-chapter-3 \')]',

问题2:目录书签跳转问题

在按照 问题1 的解决方式,去掉四个参数后,生成的 PDF 文件目录书签也发生了变化,从折叠的层级关系变成了平铺的,但点击书签跳转到对应页面的功能可以正常使用了,牺牲了一点美观性,换来了实用性,也还勉强可以接受。

问题3:目录跳转问题

点击 PDF 内目录页中链接,跳转到的地址(如:https://calibre-pdf-anchor.n/%2309Guan%20Bi%20Tu%20Xing%20Hua.html )转义后可以发现,跳转地址是由 https://calibre-pdf-anchor.n/ + #09Guan Bi Tu Xing Hua.html 两部分组成的。其中第二部分 # 后的内容,正是我们在 SUMMARY.md 关联的 Markdown 文件名(转换为了 html 格式)。

而关联的 Markdown 文件名如果是英文的,则可以在 PDF 中正常跳转,如 1.8网络 下的 03-wireshark 章节。

故要解决目录跳转问题,在 SUMMARY.md 中关联的 Markdown 文件,文件名中不能有中文(路径中可以有中文)。

问题4:边距太大

关于 PDF 文件的边距,在 PDF Options 文档中有说明,即可以通过在 book.json 添加配置进行调整。

然而即使将边距都设置为 0,依然会感觉页边距有点大。

"pdf": {
  "margin": {
    "left": 0,
    "right": 0,
    "top": 0,
    "bottom": 0
  }
}

此时可以继续修改 getConvertOptions.js 文件,在 getConvertOptions 方法最后返回的 options 对象使用的 extend 方法的第二个参数中,增加 ebook-convert 中所支持的 --pdf-page-margin-* 参数。

例如,将 --pdf-page-margin-left--pdf-page-margin-right 默认的 72px 调整为 50px

return options = extend(options, {
    '--chapter-mark':           String(pdfOptions.chapterMark),
    '--page-breaks-before':     String(pdfOptions.pageBreaksBefore),
    '--margin-left':            String(pdfOptions.margin.left),
    '--margin-right':           String(pdfOptions.margin.right),
    '--margin-top':             String(pdfOptions.margin.top),
    '--margin-bottom':          String(pdfOptions.margin.bottom),
    '--pdf-default-font-size':  String(pdfOptions.fontSize),
    '--pdf-mono-font-size':     String(pdfOptions.fontSize),
    '--paper-size':             String(pdfOptions.paperSize),
    '--pdf-page-numbers':       Boolean(pdfOptions.pageNumbers),
    '--pdf-sans-family':        String(pdfOptions.fontFamily),
    '--pdf-header-template':    headerTpl,
    '--pdf-footer-template':    footerTpl,
    '--pdf-page-margin-left':   '50',
    '--pdf-page-margin-right':  '50',
});

问题5:如何添加封面

为电子书添加封面,只需按照 Cover 中说明,在根路径放置一个符合要求的 cover.jpg 图片即可。

问题6:指定 PDF 文件名

在导出 PDF 命令中,可以指定生成 PDF 的文件名,如:

$ gitbook pdf . 文档集.pdf

问题7:如何调整字体大小

book.json 中可设定字体大小(默认为 12):

  "pdf": {
    "fontSize": 15
  }

问题8:目录(平铺)内容较多时,有不完整的情况

当按 问题1 的方式去掉了识别目录层级的 XPath 表达式后,目录书签不再按照层级的方式展现,而是平铺了出来。此时如果目录内容较多,会出现仅有部分目录显示在书签里的情况。

原因是 ebook-convert--max-toc-links 参数默认值是 50,意为最多只插入 50 个链接至目录书签中。

同样可以修改 getConvertOptions.js 文件,在 options 中加入

'--max-toc-links':              '0'

以禁用最大数量的限制。

最终效果

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352

推荐阅读更多精彩内容