如何备份 Outline 导出的 Markdown 文件

在这里插入图片描述

前面,我撰写了两篇文章,介绍了:

  • 《如何在本地环境安装 Outline》
  • 《使用 Outline 搭建企业、个人知识库面临的问题》

今天,我们继续这个话题。使用 Outline 搭建知识库,如何备份自己知识库内的资料。

Outline 底层使用数据库 PostgreSQL 和 Redis 作为异步队列的承载。其数据默认存储在 PostgreSQL 里,这和一般的 Markdown 文档系统不太一样,虽然底层本质上也是 Markdown 文档,但是却不是用纯文本格式来存储。这对备份和恢复不太友好。

前文已经介绍过,导入、导出的兼容性可能不太好,不过单纯使用导出的话,实测下来,我觉得还是可以依赖的。意思就是,你单方面在 Outline 编辑文档,然后定期导出后,进行备份,不再将导出的文档导入回 Outline 系统,站内的相对链接回失效,Outline 不能正确处理。

如何导出

在这里插入图片描述
在管理员界面,点击导出,选择 Markdown 格式,可以生成异步导出的压缩包,然后再点击下载,可以得到所有的导出文档。

此前已经说过,在 Outline 里,嵌入的图片,链接都是指向自己的 Server,而不是指向图片所在的真实地址,当你导出后,你会发现,图片什么的都正常导出了,而文章中的链接也替换成了导出目录的相对链接,这一点到让我挺满意的,超出我的预期。

不过,因为 Outline 系统自己会根据每个文档集的情况,在网站左侧建立导航栏,所以,你导出的全部文档,是一个一个的文档集的目录。没有所有文档的索引,如果你要把导出的文档,都上传的 Git 去备份的话,查看这些文档是比较让人困扰的。

如何自助建立索引

不过这难不倒程序员,我们可以用 Python 扫描所有的目录,然后,按照目录的结构,手动建立一个索引,比如,我利用 Gitlab 的 Wiki 来备份我导出的所有文档,所以,我需要一个 home.md 和 _sidebar.md 来展示我整个文档集的索引。

import os

def build_tree(path):
    tree = {'name': os.path.basename(path), 'children': []}
    if os.path.isdir(path):
        all_children = os.listdir(path)
        for child in sorted(all_children):
            child_path = os.path.join(path, child)
            if child in ['home.md', '_sidebar.md', '.git', 'public', 'index.py']:
                continue
            if os.path.isdir(child_path):
                if child in [os.path.splitext(x)[0] for x in all_children if x.endswith('.md')]:
                    sub_tree = build_tree(child_path)
                    sub_tree['link'] = os.path.join(path, child)
                    tree['children'].append(sub_tree)
                else:
                    tree['children'].append(build_tree(child_path))
            else:
                if os.path.splitext(child)[0] not in [x for x in all_children if not x.endswith('.md')]:
                    tree['children'].append({'name': child, 'children': []})
    return tree

def write_line(file, level, name, link=None, isTitle=False):
    indent = '  ' * (level - 1)
    linkStr = ''
    if link is None:
        linkStr = f"{name}"
    else:
        linkStr = f"[{name}]({link})"
    if isTitle:
        file.write(f"{indent}* **{linkStr}**\n")
    else:
        file.write(f"{indent}* {linkStr}\n")

def write_tree(tree, home, sidebar, path='', level=1):
    if 'children' in tree:
        if tree['name'] != '.':
            if len(tree['children']) > 0:
                write_line(home, level, tree['name'], tree.get('link'), isTitle=True)
                write_line(sidebar, level, tree['name'], tree.get('link'))
            else:
                filename = os.path.splitext(tree['name'])[0]
                write_line(home, level, filename, os.path.join(path, filename))
                write_line(sidebar, level, filename, os.path.join(path, filename))
        for child in tree['children']:
            write_tree(child, home, sidebar, os.path.join(path, tree['name']), level + 1)

tree = build_tree('.')
with open("home.md", "w") as home, open("_sidebar.md", "w") as sidebar:
    home.write("<!--本文件使用 index.py 生成,请勿修改本文件-->\n")
    sidebar.write("<!--本文件使用 index.py 生成,请勿修改本文件-->\n")
    write_tree(tree, home, sidebar)

这段代码是我用 ChatGPT 生成的,其作用主要是:

  • build_tree 扫描所有的目录和文件,按照原有的目录结构,建立一个目录树;
  • write_tree 递归打印目录树,按照原有目录层级,处理缩进,生成一个美观的索引文件;
  • 将索引写入 home.md 和 _sidebar.md

这样我们就可以为自己导出的文档,生成一个美观的索引目录:

在这里插入图片描述
这是我将所有的文档,提交到 Gitlab 后,渲染出来的 home.md 文件。这样 Outline 也可以当成是一个文档集的编辑器来用。

总结

本文介绍了,如何从 Outline 知识库中导出文档,以及导出的文档缺乏索引的情况下,如何自己编写代码来生成一个索引文件 home.md 和 _sidebard.md。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/512979.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

VTK| VTK可视化流程+圆锥示例

要想入门vtk&#xff0c;了解vtk的可视化流程是非常有必要的。 VTK可视化流程 VTK可视化流程主要分为数据处理和渲染两个过程&#xff0c;有一张不错的可视化流程图把这个过程理解为一个舞台剧。 VTKVS运行圆锥示例 先来运行一个简单的示例代码来理解VTK运作的过程&#xff…

薅熊链Berachain测试网空投

Berachain 是 Layer1 的一条公链。 Berachain 的经济模型引入了三种代币 BERA&#xff08;原生 Gas 代币&#xff09;&#xff1a;Bearchain 的Gas TokenBGT: Berachain 的治理 Token。 在权益证明&#xff08;Proof-of-Stake&#xff09;区块链中&#xff0c;治理通证通常用于…

一套C#自主版权+应用案例的手麻系统源码

手术麻醉信息管理系统源码&#xff0c;自主版权应用案例的手麻系统源码 手术麻醉信息管理系统包含了患者从预约申请手术到术前、术中、术后的流程控制。手术麻醉信息管理系统主要是由监护设备数据采集子系统和麻醉临床系统两个子部分组成。包括从手术申请到手术分配&#xff0c…

LeetCode-热题100:142. 环形链表 II

题目描述 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内…

CURL 实例用法参考

文章目录 1. 基础使用2. 指定请求Header3. 指定Http请求方法4. 发送POST请求,添加请求体5. 发送POST请求时&#xff0c;对请求体进行编码6. 设置请求来源7. 上传二进制文件8. 构造URL查询字段9. 新增请求头标头10. 参数打印服务器响应的标头11. 跳过SSL检测12. 模拟慢网络环境&…

GEE问题——在使用sentienl数据云掩膜的时候发现出现中间连贯性的“条带”问题,如何解决?

简介 在使用sentienl+landsat数据掩膜的时候发现出现了中间连贯性的条带问题,如何解决?这里我们使用GEE出品的Landsat和sentinel数据的过程中,当我们进行云掩膜的时候出现了条带的问题。 问题 您注意到这个问题了吗? 我该如何消除它们(例如,在镶嵌前遮蔽瓦片最外层的 …

【零基础学数据结构】顺序表

目录 1.了解数据结构 什么是数据结构&#xff1f; 为什么要进行数据管理&#xff1f; 2.顺序表 顺序表概要解析&#xff1a; ​编辑顺序表的分类&#xff1a; 差别和使用优先度&#xff1a; 1.创建顺序表 1.1顺序表分为静态顺序表和动态顺序表 1.2顺序表的初始化…

【考研数学】打基础,张宇《30讲》还是武忠祥《基础篇》?

如果基础不好&#xff0c;并且已经听过了汤家凤老师的零基础课程&#xff0c;我建议再去听一听张宇30讲 因为张宇30讲讲的要比汤家凤的零基础更加进阶&#xff0c;主要是引导学生思考&#xff0c;主要是讲题比较多。武忠祥老师的课程其实也不错&#xff0c;张宇和武忠祥的主要…

Java入门学习Day04

本篇文章主要介绍了&#xff1a;如何输入数据、字符串拼接、自增自减运算符、类型转换&#xff08;int&#xff0c;double等&#xff09; CSDN&#xff1a;码银 公众号&#xff1a;码银学编程 一、键盘输入练习 Scanner是Java中的一个类&#xff0c;用于从控制台或文件中读…

如何搭建自动化测试平台

“自动化测试”有何优势&#xff1f; 具有一致性和重复性的特点&#xff0c;而且测试更客观&#xff0c;提高了软件测试的准确度、精确度和可信任度。 可将任务自动化&#xff0c;能够解放人力去做更重要的工作。 自动化测试只需要部署好相应的场景&#xff0c;如高度复杂的使…

【CKA模拟题】StorageClass实战案例分析

Useful Resources: Storage Classes , Persistent Volumes Claim , Pods 题干 For this question, please set this context (In exam, diff cluster name) kubectl config use-context kubernetes-adminkubernetes Create a Storage Class named fast-storage with a provis…

用于无人机小型化设计的高精度温补晶振

用于无人机小型化设计的高精度温补晶振:TG2016SMN和TG2520SMN。无人机的发展可以说是非常的迅速&#xff0c;在安防&#xff0c;农业&#xff0c;交通&#xff0c;电力&#xff0c;直播等领域经常能看到无人机大显身手。无人机的应用场最是非常的广泛&#xff0c;功能更强&…

EVM Layer2 主流解决方案

深度解析主流 EVM Layer 2 解决方案&#xff1a;zk Rollups 和 Optimistic Rollups 随着以太坊网络的不断演进和 DeFi 生态系统的迅速增长&#xff0c;以太坊 Layer 2 解决方案日益受到关注。 其中&#xff0c;zk Rollups 和 Optimistic Rollups 作为两种备受瞩目的主流 EVM&…

【学习】成为优秀的软件测试工程师需要学哪些知识

成为软件测试工程师&#xff0c;需要学习的内容非常的多&#xff0c;但是无非是这几大类&#xff0c;今天就和小编一起来看看这些知识&#xff0c;你是否都已经掌握。 01、测试环境的搭建 本部分主要是学习从操作系统开始&#xff0c;有关的计算机基础知识、软件和硬件知识、…

Python基于深度学习的人脸识别项目源码+演示视频,利用OpenCV进行人脸检测与识别 preview

​ 一、原理介绍 该人脸识别实例是一个基于深度学习和计算机视觉技术的应用&#xff0c;主要利用OpenCV和Python作为开发工具。系统采用了一系列算法和技术&#xff0c;其中包括以下几个关键步骤&#xff1a; 图像预处理&#xff1a;首先&#xff0c;对输入图像进行预处理&am…

[Leetcode笔记] 动态规划相关

前言 写题目写到了一些和动态规划相关的内容&#xff0c;所以在这里记录一下 LCR 089 题解思路 总的来说&#xff0c;就是用一个数组去存储当前的最优解&#xff0c;然后从0开始一路向上顺推过去&#xff0c;求得最后一位的最优解。 class Solution { public:int rob(vect…

CAD绘制A1图框的技巧

CAD如何绘制A1图框&#xff1f;这里给大家介绍下&#xff1a; 输入REC&#xff0c;绘制矩形第一点 输入D并输入841,594 文章源自四五设计网-https://www.45te.com/44546.html 输入O&#xff0c;框选图框&#xff0c;将其偏移10文章源自四五设计网-https://www.45te.com/44546…

mysql 判断一张表是否存在的方法

查询表是否存在 使用 SHOW TABLES SHOW TABLES LIKE %tbl_tabl%;结果: 查询 INFORMATION_SCHEMA // like 匹配 SELECT TABLE_NAME FROM INFORMATION_SCHEMA.TABLES where TABLE_SCHEMA test AND TABLE_NAME like %tbl%; // 完全匹配 SELECT TABLE_NAME FROM INFORMATION_SC…

OSPF实验1

1,配置IP地址 [R1]dis ip interface brief Interface IP Address/Mask Physical Protocol GigabitEthernet0/0/0 200.1.1.1/24 up up GigabitEthernet0/0/1 10.1.1.1/24 up …

车载通信与DDS标准解读系列(4):DDSI-RTPS协议

▎什么是RTPS 在DDS协议中&#xff0c;主要描述了实现数据分发服务的DCPS模型和QoS策略&#xff0c;但是我们还不清楚数据怎样在网络中传输&#xff0c;想要了解这些内容&#xff0c;就需要请出咱们的数据搬运工——RTPS。 RTPS全称是Real-Time Publish-Subscribe Protocol&a…