如何在Laravel中实现模型修订?

Say*_*jee 4 php mysql laravel eloquent revisionable

这个问题适用于我用PHP编写的pastebin应用程序.

我做了一些研究,虽然我找不到符合我需求的解决方案.我有一个这种结构的表:

+-----------+------------------+------+-----+---------+----------------+
| Field     | Type             | Null | Key | Default | Extra          |
+-----------+------------------+------+-----+---------+----------------+
| id        | int(12) unsigned | NO   | PRI | NULL    | auto_increment |
| author    | varchar(50)      | YES  |     |         |                |
| authorid  | int(12) unsigned | YES  |     | NULL    |                |
| project   | varchar(50)      | YES  |     |         |                |
| timestamp | int(11) unsigned | NO   |     | NULL    |                |
| expire    | int(11) unsigned | NO   |     | NULL    |                |
| title     | varchar(25)      | YES  |     |         |                |
| data      | longtext         | NO   |     | NULL    |                |
| language  | varchar(50)      | NO   |     | php     |                |
| password  | varchar(60)      | NO   |     | NULL    |                |
| salt      | varchar(5)       | NO   |     | NULL    |                |
| private   | tinyint(1)       | NO   |     | 0       |                |
| hash      | varchar(12)      | NO   |     | NULL    |                |
| ip        | varchar(50)      | NO   |     | NULL    |                |
| urlkey    | varchar(8)       | YES  | MUL |         |                |
| hits      | int(11)          | NO   |     | 0       |                |
+-----------+------------------+------+-----+---------+----------------+
Run Code Online (Sandbox Code Playgroud)

这适用于pastebin应用程序.我基本上想要粘贴修改,这样如果你打开粘贴#1234,它会显示该粘贴的所有过去修订版.

我想到了三种方式:

方法1

有一个带有id和old_id或其他东西的修订表,对于每个ID,我会插入所有旧版本,所以如果我的结构如下所示:

rev3: 1234
rev2: 1233
rev1: 1232
Run Code Online (Sandbox Code Playgroud)

该表将包含以下数据:

+-------+----------+
| id    | old_id   |
+-------+----------+
| 1234  | 1233     |
| 1234  | 1232     |
| 1233  | 1232     |
+-------+----------+
Run Code Online (Sandbox Code Playgroud)

我遇到的问题是它引入了大量重复数据.修正得到的越多,它不仅有更多的数据,而且我需要为修订表中的每个新粘贴执行N次插入,这对于大N来说并不是很好.

方法2

我可以将child_id添加到顶部的粘贴表中,然后更新它.然后,在获取粘贴时,我将继续查询每个child_id及其child_id的数据库等等......但问题是,每次打开包含许多修订的粘贴时,都会引入太多数据库读取.

方法3

还涉及单独的修订表,但对于与方法1相同的方案,它将存储如下数据:

+-------+-----------------+
| id    | old_id          |
+-------+-----------------+
| 1234  | 1233,1232       |
| 1233  | 1232            |
+-------+-----------------+
Run Code Online (Sandbox Code Playgroud)

当有人打开粘贴1234时,我将使用IN子句来获取那里的所有子粘贴数据.

哪种方法最好?还是有更好的方法?我正在使用具有Eloquent ORM的Laravel 4框架.

编辑:我可以使用oneToMany关系进行方法1吗?我知道我可以使用Eager Loading来获取所有修订,但是如何插入它们而不必进行肮脏的黑客攻击?

编辑:我想出了如何处理上述问题.我将添加一个答案来关闭这个问题.

Abi*_*hek 8

如果您使用的是Laravel 4,请尝试使用Revisionable.这可能满足您的需求

  • 好吧,虽然这是一个很好的解决方案,但它将数据存储为old_value和new_value,因此我将始终拥有两个new_value副本,一个在我的主表中,一个在revisions表中.我想我最终可能会编写自己的版本控制系统:( (2认同)

Say*_*jee 2

所以这就是我正在做的事情:

假设这是修订流程:

1232 -> 1233 -> 1234
1232 -> 1235
Run Code Online (Sandbox Code Playgroud)

这是我的修订表的样子:

+----+--------+--------+
| id | new_id | old_id |
+----+--------+--------+
| 1  | 1233   | 1232   |
| 2  | 1234   | 1233   |
| 3  | 1234   | 1232   |
| 4  | 1235   | 1232   |
+----+--------+--------+
Run Code Online (Sandbox Code Playgroud)

ID 2 和 3 显示,当我打开 1234 时,它应该在列表中显示 1233 和 1232 作为修订版本。

现在是实现部分:我将使 Paste 模型与 Revision 模型建立一对多的关系。

  • 当我为现有粘贴创建新修订版时,我将运行批量插入,不仅添加当前的 new_id 和 old_id 对,还将当前的 new_id 与与 old_id 关联的所有修订版配对。
  • 当我打开粘贴时 - 我将通过查询 new_id 来执行此操作,我基本上会获取修订表中的所有关联行(使用粘贴模型中定义 hasMany('Revision', 'new_id') 的函数)并将显示用户。

我还在考虑在“查看粘贴”页面的“修订历史记录”部分中显示每个修订的作者,所以我想我还会在修订表中添加一个作者列,这样我就不需要去返回并查询主粘贴表以获取作者。

就是这样!