初探富文本之OT协同实例
初探富文本之OT协同实例
在前边初探富文本之OT协同算法一文中我们探讨了为什么需要协同、为什么仅有原子化的操作并不能实现协同、为什么要有操作变换、如何进行操作变换、什么时候能够应用操作、服务端如何进行协同调度等等,这些属于完成协同所需要了解的基础知识,实际上当前有很多成熟的协同实现,例如ot.js
、ShareDB
、ot-json
、EasySync
等等,本文就是以ShareDB
为OT
协同框架来实现协同的实例。
描述
接入协同框架实际上并不是一件简单的事情,尤其是对于OT
实现的协同算法而言,OT
的英文全称是Operational Transformation
,也就是说实现OT
的基础就是对内容的描述与操作是Operational
原子化的。在富文本领域,最经典的Operation
有quill
的delta
模型,通过retain
、insert
、delete
三个操作完成整篇文档的描述与操作,还有slate
的JSON
模型,通过insert_text
、split_node
、remove_text
等等操作来完成整篇文档的描述与操作。有了这个协同实现的基础之后,还需要对所有Op
具体实现变换Transformation
,这就是个比较麻烦的工作了,而且也是必不可少的实现。同样是以quill
与slate
两款开源编辑器为例,在quill
中已经实现了对于其数据结构delta
的所有Transformation
,可以直接调用官方的quill-delta
包即可;对于slate
而言,官方只提供了原子化的操作API
,并没有Transformation
的具体实现,但是有社区维护的slate-ot
包实现了其JSON
数据的Transformation
,也可以直接调用即可。
OT
协同的实现在富文本领域有比较多的实现可供参考,特别是在开源的富文本引擎上,其实现方案还是比较成熟的,但是引申一下,在其他领域可能并没有具体的实现,那么就需要参考接入的文档自己实现了。例如我们有一个自研的思维导图功能需要实现协同,而保存的数据结构都是自定义的,没有直接可以调用的实现方案,那么这就需要自己实现操作变换了,对于一个思维导图而言我们实现原子化的操作还是比较容易的,所以我们主要关注于变换的实现。假如这个思维导图功能我们是通过JSON
的数据结构保存的数据,那么我们就可以参考json0
或者slate-ot
的实现,特别是通过阅读单元测试可以比较容易地理解具体的功能,通过参考其实现来自行实现一份OT
的变换,或者直接依照其实现维护一个中间层的数据结构,依照于这个中间层进行数据转换。再假如我们的思维导图维护的是一个线性的类文本结构,那么就可以参考rich-text
与quill-delta
的实现,只不过这样的话实现原子化的操作可能就麻烦一些了,当然同样我们也可以维护一个中间层的数据结构来完成OT
。实际上有比较多的参考之后,接入OT
协同就主要是理解并且实现的问题了,这样就有一个大体的实现方向了,而不是毫无头绪不知道应该从哪里开始做协同。另外还是那个宗旨,合适的才是最好的,要考虑到实现的成本问题,没有必要硬套数据结构的实现,就比如上边说的实现思维导图使用线性的文本来表示还是有点牵强的,当然并不是不可能的,比如Google Docs
的Table
就是完全的线性结构,要知道其是可以实现表格中嵌套表格的,相当于每一个单元格都是一篇文档,内部可以嵌入任何的富文本结构,而在实现上就是通过线性的结构完成的。
或许上边的json0
和rich-text
等概念可能一时间让人难以理解,所以下面的Counter
与Quill
两个实例就是介绍了如何使用sharedb
实现协同,以及json0
和rich-text
究竟完成了什么样的工作,当然具体的API
调用还是还是需要看sharedb
的文档,本文只涉及到最基本的协同操作,所有的代码都在https://github.com/WindrunnerMax/Collab
中,注意这是个pnpm
的workspace monorepo
项目,要注意使用pnpm
安装依赖。
Counter
首先我们运行一个基础的协同实例Counter
,实现的主要功能是在多个客户端可以+1
的情况下我们可以维护同一份计数器总数,该实例的地址是https://github.com/WindrunnerMax/Collab/tree/master/packages/ot-counter
,首先简单看一下目录结构(tree --dirsfirst -I node_modules
):
ot-counter
├── public
│ ├── favicon.ico
│ └── index.html
├── server
│ ├── index.ts
│ └── types.d.ts
├── src
│ ├── client.ts
│ ├── counter.tsx
│ └── index.tsx
├── babel.config.js
├── package.json
├── rollup.config.js
├── rollup.server.js
└── tsconfig.json
先简略说明下各个文件夹和文件的作用,public
存储了静态资源文件,在客户端打包时将会把内容移动到build
文件夹,server
文件夹中存储了OT
服务端的实现,在运行时同样会编译为js
文件放置于build
文件夹下,src
文件夹是客户端的代码,主要是视图与OT
客户端的实现,babel.config.js
是babel
的配置信息,rollup.config.js
是打包客户端的配置文件,rollup.server.js
是打包服务端的配置文件,package.json
与tsconfig.json
大家都懂,就不赘述了。
首先我们需要了解一下json0
,乍眼一看json0
确实不容易知道这是个啥,实际上这是sharedb
默认携带的类型,sharedb
提供了很多处理操作的机制,例如我们前边提到的服务端对于Op
原子操作的调度,但没有提供转换操作的实际实现,因为业务的复杂性,必然会导致将要操作的数据结构的复杂性,于是转换和处理操作实际上是委托到业务自行实现的,在sharedb
中称为OT Types
。
OT Types
实际上相当于定义了一系列的接口,而要在sharedb
中注册类型必须实现这些接口,而这些实现就是我们需要实现的OT
操作变换,例如需要实现的transform
函数transform(op1, op2, side) -> op1'
,则必须满足apply(apply(snapshot, op1), transform(op2, op1, 'left')) == apply(apply(snapshot, op2), transform(op1, op2, 'right'))
,由此来保证变换的最终一致性,再比如compose
函数compose(op1, op2) -> op
,就必须满足apply(apply(snapshot, op1), op2) == apply(snapshot, compose(op1, op2))
,具体的文档与要求可以参考https://github.com/ottypes/docs
。
上边的这个实现看起来就很麻烦,乍眼一看还有公式,看起来对于数学上还有些要求。实现操作变换虽然本质上就是索引的转换,通过转换索引位置以确保收敛,但是要自己写还是需要些时间的,所幸在开源社区已经有很多的实现可以提供参考,在sharedb
中也附带一个了默认类型json0
,通过json0
这个JSON OT
类型可用于编辑任意JSON
文档,实际上不光是JSON
文档,我们的计数器也就是使用json0
来实现的,毕竟在这里计数器也是只需要通过借助JSON
的一个字段就可以实现的。回到json0
支持以下操作:
- 在列表中插入/删除/移动/替换项目。
- 对象插入/删除/替换。
- 原子数值加法运算。
- 嵌入任意子类型。
- 嵌入式字符串编辑,使用
text0 OT
类型作为子类型。
json0
也是一种可逆类型,也就是说所有的操作都有一个逆操作,可以撤销原来的操作。但其并不完美,其不能实现对象移动,设置为NULL
,在列表中高效地插入多个项目。此外也可以看一下json1
的实现,其实现了json0
的超集,解决了json0
的一些问题。其实看是否可以支持某些操作,直接看其文档中是不是有定义的操作就可以了,比如本例子中需要实现的计数器,就需要{p:[path], na:x}
这个Op
,将x
添加到[path]
处的数字,具体的文档可以参考https://github.com/ottypes/json0
。
接下来我们来看看服务端的实现,主要实现是实例化ShareDB
并且通过通过collection
与id
获取文档实例,在文档就绪之后触发回调启动HTTP
服务器,在这里如果不存在的文档就需要初始化,注意在这里初始化的数据就是客户端订阅时获得的数据。实例中具体的API
就不介绍了,可以参考https://share.github.io/sharedb/api/
,在这里主要是描述一下其功能。当然在这里只是非常简单的实现,真正的生产环境肯定是需要接入路由、数据库等功能的。
const backend = new ShareDB(); // `ShareDB`服务端实例
function start(callback: () => void) {
const connection = backend.connect(); // 连接到`ShareDB`
const doc = connection.get("ot-example", "counter"); // 通过`collection`与`id`获取`Doc`实例
doc.fetch(err => {
if (err) throw err;
if (doc.type === null) { // 如果不存在
doc.create({ num: 0 }, callback); // 创建初始文档然后触发回调
return;
}
callback(); // 触发回调
});
}
function server() {
const app = express(); // 实例化`express`
app.use(express.static("build")); // 客户端打包过后的静态资源路径
const server = http.createServer(app); // 创建`HTTP`服务器
const wss = new WebSocket.Server({ server: server });
wss.on("connection", ws => {
const stream = new WebSocketJSONStream(ws);
backend.listen(stream); // `ShareDB`后端需要`Stream`实例
});
server.listen(3000);
console.log("Listening on http://localhost:3000");
}
start(server);
在客户端方面主要是定义了一个定义了一个共用的链接,通过collection
与id
来获取的获取了文档的实例,也就是上面我们在服务端创建的那个文档,之后我们通过订阅文档的快照以及监听Op
事件,来操作数据,在这里我们没有直接操作数据,而是所有的操作都走的client
,这种方式就不需要考虑原子化操作的问题了,如果类似于我们下边的Quill
的实例的话,就需要监听文档的变化来实现了,在完整的实现了原子化操作的情况下,这种方案更加合适。
export type ClientCallback = (num: number) => void;
class Connection {
private connection: sharedb.Connection;
constructor() {
// 通过`WebSocket`连接到`ShareDB`
const socket = new ReconnectingWebSocket("ws://localhost:3000");
this.connection = new sharedb.Connection(socket as Socket);
}
bind(cb: ClientCallback) {
const doc = this.connection.get("ot-example", "counter"); // 通过`collection`与`id`获取`Doc`实例
const onSubscribe = () => cb(doc.data.num); // 初始化数据的回调
doc.subscribe(onSubscribe); // 订阅初始化数据
const onOpExec = () => cb(doc.data.num); // 触发`Op`的回调
doc.on("op", onOpExec); // 订阅`Op`事件 // 客户端或服务器的`Op`都会触发
return {
increase: () => {
doc.submitOp([{ p: ["num"], na: 1 }]); // `json0`的`Op`操作 // 此处为`{ num: 0 }`增加了`1`
},
unbind: () => {
doc.off("op", onOpExec); // 取消事件监听
doc.unsubscribe(onSubscribe); // 取消订阅
doc.destroy(); // 销毁文档
},
};
}
destroy() {
this.connection.close(); // 关闭链接
}
}
Quill
接下来我们运行一个富文本的实例Quill
,实现的主要功能是在quill
富文本编辑器中接入协同,并支持编辑光标的同步,该实例的地址是https://github.com/WindrunnerMax/Collab/tree/master/packages/ot-quill
,首先简单看一下目录结构(tree --dirsfirst -I node_modules
):
ot-quill
├── public
│ └── favicon.ico
├── server
│ ├── index.ts
│ └── types.d.ts
├── src
│ ├── client.ts
│ ├── index.css
│ ├── index.ts
│ ├── quill.ts
│ └── types.d.ts
├── package.json
├── rollup.config.js
├── rollup.server.js
└── tsconfig.json
依旧简略说明下各个文件夹和文件的作用,public
存储了静态资源文件,在客户端打包时将会把内容移动到build
文件夹,server
文件夹中存储了OT
服务端的实现,在运行时同样会编译为js
文件放置于build
文件夹下,src
文件夹是客户端的代码,主要是视图与OT
客户端的实现,rollup.config.js
是打包客户端的配置文件,rollup.server.js
是打包服务端的配置文件,package.json
与tsconfig.json
大家都懂,就不赘述了。
quill
的数据结构并不是JSON
而是Delta
,Delta
是通过retain
、insert
、delete
三个操作完成整篇文档的描述与操作,那么这样我们就不能使用json0
来对数据结构进行描述了,我们需要使用新的OT
类型rich-text
,rich-text
的具体的实现是在官方的quill-delta
中实现的,具体可以参考https://www.npmjs.com/package/rich-text
与https://www.npmjs.com/package/quill-delta
。
ShareDB.types.register(richText.type); // 注册`rich-text`类型
const backend = new ShareDB({ presence: true, doNotForwardSendPresenceErrorsToClient: true }); // `ShareDB`服务端实例
function start(callback: () => void) {
const connection = backend.connect(); // 连接到`ShareDB`
const doc = connection.get("ot-example", "quill"); // 通过`collection`与`id`获取`Doc`实例
doc.fetch(err => {
if (err) throw err;
if (doc.type === null) { // 如果不存在
doc.create([{ insert: "OT Quill" }], "rich-text", callback); // 创建初始文档然后触发回调
return;
}
callback();
});
}
function server() {
const app = express(); // 实例化`express`
app.use(express.static("build")); // 客户端打包过后的静态资源路径
app.use(express.static("node_modules/quill/dist")); // `quill`的静态资源路径
const server = http.createServer(app); // 创建`HTTP`服务器
const wss = new WebSocket.Server({ server: server });
wss.on("connection", function (ws) {
const stream = new WebSocketJSONStream(ws);
backend.listen(stream); // `ShareDB`后端需要`Stream`实例
});
server.listen(3000);
console.log("Listening on http://localhost:3000");
}
start(server);
在客户端主要分为了三部分,分别是实例化quill
的实例,实例化ShareDB
的客户端实例,以及quill
与ShareDB
客户端通信的实现。在quill
的实现中主要是将quill
实例化,注册光标的插件,随机生成id
的方法,以及通过id
获取随机颜色的方法。在ShareDB
的客户端操作中主要是注册了rich-text OT
类型,并且实例化了客户端与服务端的ws
链接。在quill
与ShareDB
客户端通信的实现中,主要是完成了对于quill
与doc
的事件监听,主要是Op
与Cursor
相关的实现。
Quill.register("modules/cursors", QuillCursors); // 注册光标插件
export default new Quill("#editor", { // 实例化`quill`
theme: "snow",
modules: { cursors: true },
});
const COLOR_MAP: Record<string, string> = {}; // `id => color`
export const getRandomId = () => Math.floor(Math.random() * 10000).toString(); // 随机生成用户`id`
export const getCursorColor = (id: string) => { // 根据`id`获取颜色
COLOR_MAP[id] = COLOR_MAP[id] || tinyColor.random().toHexString();
return COLOR_MAP[id];
};
const collection = "ot-example";
const id = "quill";
class Connection {
public doc: sharedb.Doc<Delta>;
private connection: sharedb.Connection;
constructor() {
sharedb.types.register(richText.type); // 注册`rich-text`类型
// 通过`WebSocket`连接到`ShareDB`
const socket = new ReconnectingWebSocket("ws://localhost:3000");
this.connection = new sharedb.Connection(socket as Socket);
this.doc = this.connection.get(collection, id); // 通过`collection`与`id`获取`Doc`实例
}
getDocPresence() {
// 订阅来自其他客户端的在线状态信息
return this.connection.getDocPresence(collection, id);
}
destroy() {
this.doc.destroy(); // 销毁文档
this.connection.close(); // 关闭链接
}
const presenceId = getRandomId(); // 生成随机`id`
const doc = client.doc; // 获取`doc`实例
const userNode = document.getElementById("user") as HTMLInputElement;
userNode && (userNode.value = "User: " + presenceId); // 显示当前用户
doc.subscribe(err => { // 订阅`doc`的初始化
if (err) {
console.log("DOC SUBSCRIBE ERROR", err);
return;
}
const cursors = quill.getModule("cursors"); // 获取光标模块
quill.setContents(doc.data); // 初始化`doc`数据
quill.on("text-change", (delta, oldDelta, source) => { // 订阅编辑器变化
if (source !== "user") return; // 非当前用户操作不提交
doc.submitOp(delta); // 提交操作
});
doc.on("op", (op, source) => { // 订阅`Op`变化
if (source) return; // 当前用户操作则返回
quill.updateContents(op as unknown as Delta); // 服务端的`Op`更新本地内容
});
const presence = client.getDocPresence(); // 订阅其他客户端状态
presence.subscribe(error => { // 订阅错误信息
if (error) console.log("PRESENCE SUBSCRIBE ERROR", err);
});
const localPresence = presence.create(presenceId); // 创建本地的状态
quill.on("selection-change", (range, oldRange, source) => { // 选区发生变化
if (source !== "user") return; // 不是当前用户则返回
if (!range) return; // 没有`Range`则返回
localPresence.submit(range, error => { // 本地的状态来提交选区`Range`
if (error) console.log("LOCAL PRESENCE SUBSCRIBE ERROR", err);
});
});
presence.on("receive", (id, range) => { // 订阅收到状态的回调
const color = getCursorColor(id); // 获取颜色
const name = "User: " + id; // 拼装名字
cursors.createCursor(id, name, color); // 创建光标
cursors.moveCursor(id, range); // 移动光标
});
});
每日一题
https://github.com/WindrunnerMax/EveryDay
参考
https://github.com/ottypes/docs
https://share.github.io/sharedb/
https://github.com/share/sharedb
https://www.npmjs.com/package/ot-json0
https://www.npmjs.com/package/ot-json1
https://zhuanlan.zhihu.com/p/481370601
https://zhuanlan.zhihu.com/p/425265438
https://www.npmjs.com/package/rich-text
https://www.npmjs.com/package/quill-delta
初探富文本之OT协同实例的更多相关文章
- React Native之TextInput的介绍与使用(富文本封装与使用实例,常用输入框封装与使用实例)
React Native之TextInput的介绍与使用(富文本封装与使用实例,常用输入框封装与使用实例) TextInput组件介绍 TextInput是一个允许用户在应用中通过键盘输入文本的基本组 ...
- Django的media配置与富文本编辑器使用的实例
效果预览 文章列表 添加文章 编辑文章|文章详情|删除文章 项目的基本文件 项目的Model from django.db import models # 导入富文本编辑器相关的模块 from cke ...
- vue-quill-editor 富文本集成quill-image-extend-module插件实例,以及UglifyJsPlugin打包抱错问题处理
官网 vue-quill-editor Toolbar Module - Quill vue-quill-image-upload 图片支持上传服务器并调整大小 1.在 package.json 中加 ...
- UILabel的富文本显示选项
UILabel的富文本格式设置 1.实例化方法和使用方法 实例化方法: 使用字符串初始化 - (id)initWithString:(NSString *)str; 例: NSMutableAttri ...
- 常用的富文本框插件FreeTextBox、CuteEditor、CKEditor、FCKEditor、TinyMCE、KindEditor ;和CKEditor实例
http://www.cnblogs.com/cxd4321/archive/2013/01/30/2883078.html 目前市面上用的比较多的富文本编辑器有: FreeTextBox 一个有很多 ...
- 关于富文本编辑器ueditor(jsp版)上传文件到阿里云OSS的简单实例,适合新手
关于富文本编辑器ueditor(jsp版)上传文件到阿里云OSS的简单实例,适合新手 本人菜鸟一枚,最近公司有需求要用到富文本编辑器,我选择的是百度的ueditor富文本编辑器,闲话不多说,进入正 ...
- UEditor 百度富文本编辑器 .Net实例
转自 http://download.csdn.net/download/hdsslxl/6740605 1.UEditor 百度富文本编辑器完整版 .Net实例 已解决上传图片问题. 2.内附完整d ...
- vue集成ckeditor富文本框,怎么获取CKEditor实例?
CKEDITOR 版本5 ,vue集成形式 vue集成ckeditor富文本框,由于不是通过js创建的富文本对象,所以,无法取得实例对象,官方说明 官方在builds-->Getting and ...
- 现代富文本编辑器Quill的模块化机制
DevUI是一支兼具设计视角和工程视角的团队,服务于华为云DevCloud平台和华为内部数个中后台系统,服务于设计师和前端工程师.官方网站:devui.designNg组件库:ng-devui(欢迎S ...
- 现代富文本编辑器Quill的内容渲染机制
DevUI是一支兼具设计视角和工程视角的团队,服务于华为云DevCloud平台和华为内部数个中后台系统,服务于设计师和前端工程师.官方网站:devui.designNg组件库:ng-devui(欢迎S ...
随机推荐
- 记录一次sshd服务启动失败
记录一次sshd服务启动失败 问题描述: 服务器开机之后发现无法通过远程连接服务器终端,但是服务器并未宕机,于是考虑到sshd服务出现异常 解决思路: 查看服务器sshd服务运行情况 [root@ha ...
- 19_Vue如何监测到对象类型数据发生改变的?
数据更新 关于监视 我们之前讲过,我们在data当中配置的属性,最终会挂载在vue实例身上,而data这个配置项,最终也会在vue身上成为一个新的属性 == _data 当我们在页面DOM当中,去使用 ...
- kubernetes之kubectl与YAML详解1
k8s集群的日志,带有组件的信息,多看日志. kubectl命令汇总 kubectl命令汇总 kubectl命令帮助信息 [root@mcwk8s04 ~]# kubectl -h kubectl c ...
- c++ 关于引用变量你不知道的东西
引用变量延迟绑定 我们知道引用变量定义时要立刻赋值,告诉编译器他是谁的引用.如果不赋值,编译会失败. 如果引用变量是单个定义的,对他赋值还比较简单. struct test_T { int data; ...
- 搭建K8S集群前置条件
搭建K8S集群 搭建k8s环境平台规划 单master集群 单个master节点,然后管理多个node节点 多master集群 多个master节点,管理多个node节点,同时中间多了一个负载均衡的过 ...
- 【OpenStack云平台】SecureCRT 连接 CentOS虚拟机
1.安装SecureCRT SecureCRT是一款支持SSH等协议的终端仿真软件,可以在windows下登录Linux服务器,这样大大方便了开发工作.安装SecureCRT可以通过网上的各种教程安装 ...
- kubernetes_CoreDNS全解析
一.前言 kubernetes CoreDNS 是 kube-system 命令空间里面的一个Pod,用于域名解析. kubernetes自带三个命名空间(用kubeadm安装的Kubernetes集 ...
- JqGrid 编辑单元格内容时提示url未设定错误 2018-08-06
感谢大佬的资料https://blog.csdn.net/Easy_____/article/details/30218421 虽然没实例,但也给了一些信息.我以为cellsubmit属性是添加到co ...
- day20-web开发会话技术02
WEB开发会话技术02 6.Cookie的生命周期 默认情况下,Cookie只在浏览器的内存中存活,也就是说,当你关闭浏览器后,Cookie就会消失.但是也可以通过方法设置cookie的生存时间. c ...
- kettel
下载教程:(目前最高版本7.1) 1.网址:https://community.hitachivantara.com/docs/DOC-1009855 2.