LOTS of changes to properly handle post/topic revisions
FIX: history revision can now properly be hidden FIX: PostRevision serializer is now entirely dynamic to properly handle hidden revisions FIX: default history modal to "side by side" view on mobile FIX: properly hiden which revision has been hidden UX: inline category/user/wiki/post_type changes with the revision details FEATURE: new '/posts/:post_id/revisions/latest' endpoint to retrieve latest revision UX: do not show the hide/show revision button on mobile (no room for them) UX: remove CSS transitions on the buttons in the history modal FIX: PostRevisor now handles all the changes that might create new revisions FIX: PostRevision.ensure_consistency! was wrong due to off by 1 mistake... refactored topic's callbacks for better readability extracted 'PostRevisionGuardian'
This commit is contained in:
parent
caf31dde1e
commit
e7f251c105
|
@ -12,8 +12,8 @@ import ObjectController from 'discourse/controllers/object';
|
||||||
@module Discourse
|
@module Discourse
|
||||||
**/
|
**/
|
||||||
export default ObjectController.extend(ModalFunctionality, {
|
export default ObjectController.extend(ModalFunctionality, {
|
||||||
loading: false,
|
loading: true,
|
||||||
viewMode: "side_by_side",
|
viewMode: Discourse.Mobile.mobileView ? "inline" : "side_by_side",
|
||||||
revisionsTextKey: "post.revisions.controls.comparing_previous_to_current_out_of_total",
|
revisionsTextKey: "post.revisions.controls.comparing_previous_to_current_out_of_total",
|
||||||
|
|
||||||
refresh: function(postId, postVersion) {
|
refresh: function(postId, postVersion) {
|
||||||
|
@ -41,87 +41,72 @@ export default ObjectController.extend(ModalFunctionality, {
|
||||||
|
|
||||||
createdAtDate: function() { return moment(this.get("created_at")).format("LLLL"); }.property("created_at"),
|
createdAtDate: function() { return moment(this.get("created_at")).format("LLLL"); }.property("created_at"),
|
||||||
|
|
||||||
previousVersion: function() { return this.get("version") - 1; }.property("version"),
|
previousVersion: function() { return this.get("current_version") - 1; }.property("current_version"),
|
||||||
|
|
||||||
displayGoToFirst: Em.computed.gt("version", 3),
|
displayGoToFirst: function() { return this.get("current_revision") > this.get("first_revision"); }.property("current_revision", "first_revision"),
|
||||||
displayGoToPrevious: Em.computed.gt("version", 2),
|
displayGoToPrevious: function() { return this.get("previous_revision") && this.get("current_revision") > this.get("previous_revision"); }.property("current_revision", "previous_revision"),
|
||||||
displayRevisions: Em.computed.gt("revisions_count", 2),
|
displayRevisions: Em.computed.gt("version_count", 2),
|
||||||
displayGoToNext: function() { return this.get("version") < this.get("revisions_count"); }.property("version", "revisions_count"),
|
displayGoToNext: function() { return this.get("next_revision") && this.get("current_revision") < this.get("next_revision"); }.property("current_revision", "next_revision"),
|
||||||
displayGoToLast: function() { return this.get("version") < this.get("revisions_count") - 1; }.property("version", "revisions_count"),
|
displayGoToLast: function() { return this.get("current_revision") < this.get("last_revision"); }.property("current_revision", "last_revision"),
|
||||||
|
|
||||||
displayShow: function() { return this.get("hidden") && Discourse.User.currentProp('staff'); }.property("hidden"),
|
displayShow: function() { return !Discourse.Mobile.mobileView && this.get("previous_hidden") && Discourse.User.currentProp('staff'); }.property("previous_hidden"),
|
||||||
displayHide: function() { return !this.get("hidden") && Discourse.User.currentProp('staff'); }.property("hidden"),
|
displayHide: function() { return !Discourse.Mobile.mobileView && !this.get("previous_hidden") && Discourse.User.currentProp('staff'); }.property("previous_hidden"),
|
||||||
|
|
||||||
|
isEitherRevisionHidden: Em.computed.or("previous_hidden", "current_hidden"),
|
||||||
|
|
||||||
|
hiddenClasses: function() {
|
||||||
|
if (this.get("displayingInline")) {
|
||||||
|
return this.get("isEitherRevisionHidden") ? "hidden-revision-either" : null;
|
||||||
|
} else {
|
||||||
|
var result = [];
|
||||||
|
if (this.get("previous_hidden")) { result.push("hidden-revision-previous"); }
|
||||||
|
if (this.get("current_hidden")) { result.push("hidden-revision-current"); }
|
||||||
|
return result.join(" ");
|
||||||
|
}
|
||||||
|
}.property("previous_hidden", "current_hidden", "displayingInline"),
|
||||||
|
|
||||||
displayingInline: Em.computed.equal("viewMode", "inline"),
|
displayingInline: Em.computed.equal("viewMode", "inline"),
|
||||||
displayingSideBySide: Em.computed.equal("viewMode", "side_by_side"),
|
displayingSideBySide: Em.computed.equal("viewMode", "side_by_side"),
|
||||||
displayingSideBySideMarkdown: Em.computed.equal("viewMode", "side_by_side_markdown"),
|
displayingSideBySideMarkdown: Em.computed.equal("viewMode", "side_by_side_markdown"),
|
||||||
|
|
||||||
category_diff: function() {
|
previousCategory: function() {
|
||||||
var viewMode = this.get("viewMode");
|
|
||||||
var changes = this.get("category_changes");
|
var changes = this.get("category_changes");
|
||||||
|
if (changes) {
|
||||||
if (changes === null) { return; }
|
var category = Discourse.Category.findById(changes["previous"]);
|
||||||
|
return Discourse.HTML.categoryBadge(category, { allowUncategorized: true });
|
||||||
var prevCategory = Discourse.Category.findById(changes.previous_category_id);
|
|
||||||
var curCategory = Discourse.Category.findById(changes.current_category_id);
|
|
||||||
|
|
||||||
var raw = "";
|
|
||||||
var opts = { allowUncategorized: true };
|
|
||||||
prevCategory = Discourse.HTML.categoryBadge(prevCategory, opts);
|
|
||||||
curCategory = Discourse.HTML.categoryBadge(curCategory, opts);
|
|
||||||
|
|
||||||
if(viewMode === "side_by_side_markdown" || viewMode === "side_by_side") {
|
|
||||||
raw = "<div class='span8'>" + prevCategory + "</div> <div class='span8 offset1'>" + curCategory + "</div>";
|
|
||||||
} else {
|
|
||||||
var diff = "<del>" + prevCategory + "</del> " + "<ins>" + curCategory + "</ins>";
|
|
||||||
raw = "<div class='inline-diff'>" + diff + "</div>";
|
|
||||||
}
|
}
|
||||||
|
}.property("category_changes"),
|
||||||
|
|
||||||
return raw;
|
currentCategory: function() {
|
||||||
|
var changes = this.get("category_changes");
|
||||||
}.property("viewMode", "category_changes"),
|
if (changes) {
|
||||||
|
var category = Discourse.Category.findById(changes["current"]);
|
||||||
|
return Discourse.HTML.categoryBadge(category, { allowUncategorized: true });
|
||||||
|
}
|
||||||
|
}.property("category_changes"),
|
||||||
|
|
||||||
wiki_diff: function() {
|
wiki_diff: function() {
|
||||||
var viewMode = this.get("viewMode");
|
var changes = this.get("wiki_changes")
|
||||||
var changes = this.get("wiki_changes");
|
if (changes) {
|
||||||
if (changes === null) { return; }
|
return changes["current"] ?
|
||||||
|
'<span class="fa-stack"><i class="fa fa-pencil-square-o fa-stack-2x"></i></span>' :
|
||||||
if (viewMode === "inline") {
|
'<span class="fa-stack"><i class="fa fa-pencil-square-o fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
||||||
var diff = changes["current_wiki"] ? '<i class="fa fa-pencil-square-o fa-2x"></i>' : '<span class="fa-stack"><i class="fa fa-pencil-square-o fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
|
||||||
return "<div class='inline-diff'>" + diff + "</div>";
|
|
||||||
} else {
|
|
||||||
var prev = changes["previous_wiki"] ? '<i class="fa fa-pencil-square-o fa-2x"></i>' : " ";
|
|
||||||
var curr = changes["current_wiki"] ? '<i class="fa fa-pencil-square-o fa-2x"></i>' : '<span class="fa-stack"><i class="fa fa-pencil-square-o fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
|
||||||
return "<div class='span8'>" + prev + "</div><div class='span8 offset1'>" + curr + "</div>";
|
|
||||||
}
|
}
|
||||||
}.property("viewMode", "wiki_changes"),
|
}.property("wiki_changes"),
|
||||||
|
|
||||||
post_type_diff: function () {
|
post_type_diff: function () {
|
||||||
var viewMode = this.get("viewMode");
|
|
||||||
var changes = this.get("post_type_changes");
|
|
||||||
if (changes === null) { return; }
|
|
||||||
|
|
||||||
var moderator = Discourse.Site.currentProp('post_types.moderator_action');
|
var moderator = Discourse.Site.currentProp('post_types.moderator_action');
|
||||||
|
var changes = this.get("post_type_changes");
|
||||||
if (viewMode === "inline") {
|
if (changes) {
|
||||||
var diff = changes["current_post_type"] === moderator ?
|
return changes["current"] == moderator ?
|
||||||
'<i class="fa fa-shield fa-2x"></i>' :
|
'<span class="fa-stack"><i class="fa fa-shield fa-stack-2x"></i></span>' :
|
||||||
'<span class="fa-stack"><i class="fa fa-shield fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
'<span class="fa-stack"><i class="fa fa-shield fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
||||||
return "<div class='inline-diff'>" + diff + "</div>";
|
|
||||||
} else {
|
|
||||||
var prev = changes["previous_post_type"] === moderator ? '<i class="fa fa-shield fa-2x"></i>' : " ";
|
|
||||||
var curr = changes["current_post_type"] === moderator ?
|
|
||||||
'<i class="fa fa-shield fa-2x"></i>' :
|
|
||||||
'<span class="fa-stack"><i class="fa fa-shield fa-stack-2x"></i><i class="fa fa-ban fa-stack-2x"></i></span>';
|
|
||||||
return "<div class='span8'>" + prev + "</div><div class='span8 offset1'>" + curr + "</div>";
|
|
||||||
}
|
}
|
||||||
}.property("viewMode", "post_type_changes"),
|
}.property("post_type_changes"),
|
||||||
|
|
||||||
title_diff: function() {
|
title_diff: function() {
|
||||||
var viewMode = this.get("viewMode");
|
var viewMode = this.get("viewMode");
|
||||||
if(viewMode === "side_by_side_markdown") {
|
if (viewMode === "side_by_side_markdown") { viewMode = "side_by_side"; }
|
||||||
viewMode = "side_by_side";
|
|
||||||
}
|
|
||||||
return this.get("title_changes." + viewMode);
|
return this.get("title_changes." + viewMode);
|
||||||
}.property("viewMode", "title_changes"),
|
}.property("viewMode", "title_changes"),
|
||||||
|
|
||||||
|
@ -130,13 +115,13 @@ export default ObjectController.extend(ModalFunctionality, {
|
||||||
}.property("viewMode", "body_changes"),
|
}.property("viewMode", "body_changes"),
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
loadFirstVersion: function() { this.refresh(this.get("post_id"), 2); },
|
loadFirstVersion: function() { this.refresh(this.get("post_id"), this.get("first_revision")); },
|
||||||
loadPreviousVersion: function() { this.refresh(this.get("post_id"), this.get("version") - 1); },
|
loadPreviousVersion: function() { this.refresh(this.get("post_id"), this.get("previous_revision")); },
|
||||||
loadNextVersion: function() { this.refresh(this.get("post_id"), this.get("version") + 1); },
|
loadNextVersion: function() { this.refresh(this.get("post_id"), this.get("next_revision")); },
|
||||||
loadLastVersion: function() { this.refresh(this.get("post_id"), this.get("revisions_count")); },
|
loadLastVersion: function() { this.refresh(this.get("post_id"), this.get("last_revision")); },
|
||||||
|
|
||||||
hideVersion: function() { this.hide(this.get("post_id"), this.get("version")); },
|
hideVersion: function() { this.hide(this.get("post_id"), this.get("current_revision")); },
|
||||||
showVersion: function() { this.show(this.get("post_id"), this.get("version")); },
|
showVersion: function() { this.show(this.get("post_id"), this.get("current_revision")); },
|
||||||
|
|
||||||
displayInline: function() { this.set("viewMode", "inline"); },
|
displayInline: function() { this.set("viewMode", "inline"); },
|
||||||
displaySideBySide: function() { this.set("viewMode", "side_by_side"); },
|
displaySideBySide: function() { this.set("viewMode", "side_by_side"); },
|
||||||
|
|
|
@ -111,9 +111,7 @@ Discourse.Post = Discourse.Model.extend({
|
||||||
}.property('link_counts.@each.internal'),
|
}.property('link_counts.@each.internal'),
|
||||||
|
|
||||||
// Edits are the version - 1, so version 2 = 1 edit
|
// Edits are the version - 1, so version 2 = 1 edit
|
||||||
editCount: function() {
|
editCount: function() { return this.get('version') - 1; }.property('version'),
|
||||||
return this.get('version') - 1;
|
|
||||||
}.property('version'),
|
|
||||||
|
|
||||||
flagsAvailable: function() {
|
flagsAvailable: function() {
|
||||||
var post = this;
|
var post = this;
|
||||||
|
|
|
@ -82,7 +82,7 @@ Discourse.TopicRoute = Discourse.Route.extend({
|
||||||
|
|
||||||
showHistory: function(post) {
|
showHistory: function(post) {
|
||||||
Discourse.Route.showModal(this, 'history', post);
|
Discourse.Route.showModal(this, 'history', post);
|
||||||
this.controllerFor('history').refresh(post.get("id"), post.get("version"));
|
this.controllerFor('history').refresh(post.get("id"), "latest");
|
||||||
this.controllerFor('modal').set('modalClass', 'history-modal');
|
this.controllerFor('modal').set('modalClass', 'history-modal');
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -7,13 +7,13 @@
|
||||||
{{#if loading}}
|
{{#if loading}}
|
||||||
<div id='revision-loading'><i class='fa fa-spinner fa-spin'></i>{{i18n loading}}</div>
|
<div id='revision-loading'><i class='fa fa-spinner fa-spin'></i>{{i18n loading}}</div>
|
||||||
{{else}}
|
{{else}}
|
||||||
{{boundI18n revisionsTextKey previousBinding="previousVersion" currentBinding="version" totalBinding="revisions_count"}}
|
{{boundI18n revisionsTextKey previousBinding="previousVersion" currentBinding="current_version" totalBinding="version_count"}}
|
||||||
{{/if}}
|
{{/if}}
|
||||||
</div>
|
</div>
|
||||||
<button title="{{i18n post.revisions.controls.next}}" {{bind-attr class=":btn :standard :no-text displayGoToNext::invisible" disabled=loading}} {{action "loadNextVersion"}}><i class="fa fa-forward"></i></button>
|
<button title="{{i18n post.revisions.controls.next}}" {{bind-attr class=":btn :standard :no-text displayGoToNext::invisible" disabled=loading}} {{action "loadNextVersion"}}><i class="fa fa-forward"></i></button>
|
||||||
<button title="{{i18n post.revisions.controls.last}}" {{bind-attr class=":btn :standard :no-text displayGoToLast::invisible" disabled=loading}} {{action "loadLastVersion"}}><i class="fa fa-fast-forward"></i></button>
|
<button title="{{i18n post.revisions.controls.last}}" {{bind-attr class=":btn :standard :no-text displayGoToLast::invisible" disabled=loading}} {{action "loadLastVersion"}}><i class="fa fa-fast-forward"></i></button>
|
||||||
{{#if displayHide}}
|
{{#if displayHide}}
|
||||||
<button title="{{i18n post.revisions.controls.hide}}" {{bind-attr class=":btn :standard :no-text" disabled=loading}} {{action "hideVersion"}}><i class="fa fa-trash-o"></i></button>
|
<button title="{{i18n post.revisions.controls.hide}}" {{bind-attr class=":btn :standard :no-text :btn-danger" disabled=loading}} {{action "hideVersion"}}><i class="fa fa-trash-o"></i></button>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{#if displayShow}}
|
{{#if displayShow}}
|
||||||
<button title="{{i18n post.revisions.controls.show}}" {{bind-attr class=":btn :standard :no-text" disabled=loading}} {{action "showVersion"}}><i class="fa fa-undo"></i></button>
|
<button title="{{i18n post.revisions.controls.show}}" {{bind-attr class=":btn :standard :no-text" disabled=loading}} {{action "showVersion"}}><i class="fa fa-undo"></i></button>
|
||||||
|
@ -28,22 +28,41 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="revision-details">
|
<div id="revision-details">
|
||||||
{{i18n post.revisions.details.edited_by}} {{#link-to 'user' username}}{{bound-avatar-template content.avatar_template "small"}} {{username}}{{/link-to}} <span class="date">{{bound-date created_at}}</span> {{#if edit_reason}} — <span class="edit-reason">{{edit_reason}}</span>{{/if}}
|
{{i18n post.revisions.details.edited_by}}
|
||||||
|
{{#link-to 'user' username}}
|
||||||
|
{{bound-avatar-template content.avatar_template "small"}} {{username}}
|
||||||
|
{{/link-to}}
|
||||||
|
<span class="date">{{bound-date created_at}}</span>
|
||||||
|
{{#if edit_reason}}
|
||||||
|
— <span class="edit-reason">{{edit_reason}}</span>
|
||||||
|
{{/if}}
|
||||||
|
{{#unless site.mobileView}}
|
||||||
|
{{#if user_changes}}
|
||||||
|
— {{bound-avatar-template user_changes.previous.avatar_template "small"}} {{user_changes.previous.username}}
|
||||||
|
→ {{bound-avatar-template user_changes.current.avatar_template "small"}} {{user_changes.current.username}}
|
||||||
|
{{/if}}
|
||||||
|
{{#if wiki_changes}}
|
||||||
|
— {{{wiki_diff}}}
|
||||||
|
{{/if}}
|
||||||
|
{{#if post_type_changes}}
|
||||||
|
— {{{post_type_diff}}}
|
||||||
|
{{/if}}
|
||||||
|
{{#if category_changes}}
|
||||||
|
— {{{previousCategory}}} → {{{currentCategory}}}
|
||||||
|
{{/if}}
|
||||||
|
{{/unless}}
|
||||||
</div>
|
</div>
|
||||||
<div id="revisions" {{bind-attr class="hidden:hidden-revision"}}>
|
<div id="revisions" {{bind-attr class="hiddenClasses"}}>
|
||||||
{{#if title_changes}}
|
{{#if title_changes}}
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<h2>{{{title_diff}}}</h2>
|
<h2>{{{title_diff}}}</h2>
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{#if category_changes}}
|
{{#if site.mobileView}}
|
||||||
<div class="row">
|
|
||||||
{{{category_diff}}}
|
|
||||||
</div>
|
|
||||||
{{/if}}
|
|
||||||
{{#if user_changes}}
|
{{#if user_changes}}
|
||||||
<div class="row">
|
<div class="row">
|
||||||
{{bound-avatar-template user_changes.previous.avatar_template "small"}} {{user_changes.previous.username}} → {{bound-avatar-template user_changes.current.avatar_template "small"}} {{user_changes.current.username}}
|
{{bound-avatar-template user_changes.previous.avatar_template "small"}} {{user_changes.previous.username}}
|
||||||
|
→ {{bound-avatar-template user_changes.current.avatar_template "small"}} {{user_changes.current.username}}
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
{{#if wiki_changes}}
|
{{#if wiki_changes}}
|
||||||
|
@ -56,6 +75,14 @@
|
||||||
{{{post_type_diff}}}
|
{{{post_type_diff}}}
|
||||||
</div>
|
</div>
|
||||||
{{/if}}
|
{{/if}}
|
||||||
|
{{#if category_changes}}
|
||||||
|
<div class="row">
|
||||||
|
{{{previousCategory}}} → {{{currentCategory}}}
|
||||||
|
</div>
|
||||||
|
{{/if}}
|
||||||
|
{{/if}}
|
||||||
|
<div class="row">
|
||||||
{{{body_diff}}}
|
{{{body_diff}}}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -11,6 +11,9 @@
|
||||||
margin-right: 7px;
|
margin-right: 7px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#revisions .row:first-of-type {
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
ins, .diff-ins {
|
ins, .diff-ins {
|
||||||
code, img {
|
code, img {
|
||||||
border: 2px solid $success;
|
border: 2px solid $success;
|
||||||
|
@ -75,7 +78,17 @@
|
||||||
.fa-ban {
|
.fa-ban {
|
||||||
color: #f00;
|
color: #f00;
|
||||||
}
|
}
|
||||||
.hidden-revision {
|
.hidden-revision-either {
|
||||||
opacity: 0.5;
|
opacity: .5;
|
||||||
|
}
|
||||||
|
.hidden-revision-previous .row {
|
||||||
|
div:nth-of-type(1), td:nth-of-type(1) {
|
||||||
|
opacity: .5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.hidden-revision-current .row {
|
||||||
|
div:nth-of-type(2), td:nth-of-type(2) {
|
||||||
|
opacity: .5;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,10 @@
|
||||||
// styles that apply to the popup that appears when you show the edit history of a post
|
// styles that apply to the popup that appears when you show the edit history of a post
|
||||||
|
|
||||||
.modal.history-modal {
|
.modal.history-modal {
|
||||||
|
.btn {
|
||||||
|
// remove transitions on the buttons in the history modal
|
||||||
|
transition: none;
|
||||||
|
}
|
||||||
.modal-inner-container {
|
.modal-inner-container {
|
||||||
min-width: 960px;
|
min-width: 960px;
|
||||||
min-height: 500px;
|
min-height: 500px;
|
||||||
|
@ -18,13 +22,13 @@
|
||||||
background-color: scale-color-diff();
|
background-color: scale-color-diff();
|
||||||
padding: 5px;
|
padding: 5px;
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
|
line-height: 2em;
|
||||||
|
height: 30px;
|
||||||
}
|
}
|
||||||
#revisions {
|
#revisions {
|
||||||
word-wrap: break-word;
|
word-wrap: break-word;
|
||||||
.row, table {
|
table {
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
padding-bottom: 10px;
|
|
||||||
border-bottom: 1px dotted;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
img {
|
img {
|
||||||
|
|
|
@ -35,7 +35,7 @@
|
||||||
.modal-body {
|
.modal-body {
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
max-height: 400px;
|
max-height: 400px;
|
||||||
padding: 10px 0 10px 15px;
|
padding: 10px 0 10px 10px;
|
||||||
width: 95%;
|
width: 95%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ require_dependency 'distributed_memoizer'
|
||||||
class PostsController < ApplicationController
|
class PostsController < ApplicationController
|
||||||
|
|
||||||
# Need to be logged in for all actions here
|
# Need to be logged in for all actions here
|
||||||
before_filter :ensure_logged_in, except: [:show, :replies, :by_number, :short_link, :reply_history, :revisions, :expand_embed, :markdown, :raw, :cooked]
|
before_filter :ensure_logged_in, except: [:show, :replies, :by_number, :short_link, :reply_history, :revisions, :latest_revision, :expand_embed, :markdown, :raw, :cooked]
|
||||||
|
|
||||||
skip_before_filter :check_xhr, only: [:markdown_id, :markdown_num, :short_link]
|
skip_before_filter :check_xhr, only: [:markdown_id, :markdown_num, :short_link]
|
||||||
|
|
||||||
|
@ -99,39 +99,30 @@ class PostsController < ApplicationController
|
||||||
post.image_sizes = params[:image_sizes] if params[:image_sizes].present?
|
post.image_sizes = params[:image_sizes] if params[:image_sizes].present?
|
||||||
|
|
||||||
if too_late_to(:edit, post)
|
if too_late_to(:edit, post)
|
||||||
render json: {errors: [I18n.t('too_late_to_edit')]}, status: 422
|
return render json: { errors: [I18n.t('too_late_to_edit')] }, status: 422
|
||||||
return
|
|
||||||
end
|
end
|
||||||
|
|
||||||
guardian.ensure_can_edit!(post)
|
guardian.ensure_can_edit!(post)
|
||||||
|
|
||||||
# to stay consistent with the create api,
|
changes = {
|
||||||
# we should allow for title changes and category changes here
|
raw: params[:post][:raw],
|
||||||
# we should also move all of this to a post updater.
|
edit_reason: params[:post][:edit_reason]
|
||||||
if post.post_number == 1 && (params[:title] || params[:post][:category_id])
|
}
|
||||||
post.topic.acting_user = current_user
|
|
||||||
post.topic.title = params[:title] if params[:title]
|
|
||||||
Topic.transaction do
|
|
||||||
post.topic.change_category_to_id(params[:post][:category_id].to_i)
|
|
||||||
post.topic.save
|
|
||||||
end
|
|
||||||
|
|
||||||
if post.topic.errors.present?
|
# to stay consistent with the create api, we allow for title & category changes here
|
||||||
render_json_error(post.topic)
|
if post.post_number == 1
|
||||||
return
|
changes[:title] = params[:title] if params[:title]
|
||||||
end
|
changes[:category_id] = params[:post][:category_id] if params[:post][:category_id]
|
||||||
end
|
end
|
||||||
|
|
||||||
revisor = PostRevisor.new(post)
|
revisor = PostRevisor.new(post)
|
||||||
if revisor.revise!(current_user, params[:post][:raw], edit_reason: params[:post][:edit_reason])
|
if revisor.revise!(current_user, changes)
|
||||||
TopicLink.extract_from(post)
|
TopicLink.extract_from(post)
|
||||||
QuotedPost.extract_from(post)
|
QuotedPost.extract_from(post)
|
||||||
end
|
end
|
||||||
|
|
||||||
if post.errors.present?
|
return render_json_error(post) if post.errors.present?
|
||||||
render_json_error(post)
|
return render_json_error(post.topic) if post.topic.errors.present?
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
|
post_serializer = PostSerializer.new(post, scope: guardian, root: false)
|
||||||
post_serializer.draft_sequence = DraftSequence.current(current_user, post.topic.draft_key)
|
post_serializer.draft_sequence = DraftSequence.current(current_user, post.topic.draft_key)
|
||||||
|
@ -194,7 +185,6 @@ class PostsController < ApplicationController
|
||||||
end
|
end
|
||||||
|
|
||||||
def destroy_many
|
def destroy_many
|
||||||
|
|
||||||
params.require(:post_ids)
|
params.require(:post_ids)
|
||||||
|
|
||||||
posts = Post.where(id: post_ids_including_replies)
|
posts = Post.where(id: post_ids_including_replies)
|
||||||
|
@ -222,17 +212,35 @@ class PostsController < ApplicationController
|
||||||
render_json_dump(post_revision_serializer)
|
render_json_dump(post_revision_serializer)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def latest_revision
|
||||||
|
post_revision = find_latest_post_revision_from_params
|
||||||
|
post_revision_serializer = PostRevisionSerializer.new(post_revision, scope: guardian, root: false)
|
||||||
|
render_json_dump(post_revision_serializer)
|
||||||
|
end
|
||||||
|
|
||||||
def hide_revision
|
def hide_revision
|
||||||
post_revision = find_post_revision_from_params
|
post_revision = find_post_revision_from_params
|
||||||
guardian.ensure_can_hide_post_revision! post_revision
|
guardian.ensure_can_hide_post_revision!(post_revision)
|
||||||
|
|
||||||
post_revision.hide!
|
post_revision.hide!
|
||||||
|
|
||||||
|
post = find_post_from_params
|
||||||
|
post.public_version -= 1
|
||||||
|
post.save
|
||||||
|
|
||||||
render nothing: true
|
render nothing: true
|
||||||
end
|
end
|
||||||
|
|
||||||
def show_revision
|
def show_revision
|
||||||
post_revision = find_post_revision_from_params
|
post_revision = find_post_revision_from_params
|
||||||
guardian.ensure_can_show_post_revision! post_revision
|
guardian.ensure_can_show_post_revision!(post_revision)
|
||||||
|
|
||||||
post_revision.show!
|
post_revision.show!
|
||||||
|
|
||||||
|
post = find_post_from_params
|
||||||
|
post.public_version += 1
|
||||||
|
post.save
|
||||||
|
|
||||||
render nothing: true
|
render nothing: true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -252,9 +260,7 @@ class PostsController < ApplicationController
|
||||||
guardian.ensure_can_wiki!
|
guardian.ensure_can_wiki!
|
||||||
|
|
||||||
post = find_post_from_params
|
post = find_post_from_params
|
||||||
post.wiki = params[:wiki]
|
post.revise(current_user, { wiki: params[:wiki] })
|
||||||
post.version += 1
|
|
||||||
post.save
|
|
||||||
|
|
||||||
render nothing: true
|
render nothing: true
|
||||||
end
|
end
|
||||||
|
@ -263,9 +269,7 @@ class PostsController < ApplicationController
|
||||||
guardian.ensure_can_change_post_type!
|
guardian.ensure_can_change_post_type!
|
||||||
|
|
||||||
post = find_post_from_params
|
post = find_post_from_params
|
||||||
post.post_type = params[:post_type].to_i
|
post.revise(current_user, { post_type: params[:post_type].to_i })
|
||||||
post.version += 1
|
|
||||||
post.save
|
|
||||||
|
|
||||||
render nothing: true
|
render nothing: true
|
||||||
end
|
end
|
||||||
|
@ -329,9 +333,26 @@ class PostsController < ApplicationController
|
||||||
raise Discourse::InvalidParameters.new(:revision) if revision < 2
|
raise Discourse::InvalidParameters.new(:revision) if revision < 2
|
||||||
|
|
||||||
post_revision = PostRevision.find_by(post_id: post_id, number: revision)
|
post_revision = PostRevision.find_by(post_id: post_id, number: revision)
|
||||||
post_revision.post = find_post_from_params
|
raise Discourse::NotFound unless post_revision
|
||||||
|
|
||||||
|
post_revision.post = find_post_from_params
|
||||||
guardian.ensure_can_see!(post_revision)
|
guardian.ensure_can_see!(post_revision)
|
||||||
|
|
||||||
|
post_revision
|
||||||
|
end
|
||||||
|
|
||||||
|
def find_latest_post_revision_from_params
|
||||||
|
post_id = params[:id] || params[:post_id]
|
||||||
|
|
||||||
|
finder = PostRevision.where(post_id: post_id).order(:number)
|
||||||
|
finder = finder.where(hidden: false) unless guardian.is_staff?
|
||||||
|
post_revision = finder.last
|
||||||
|
|
||||||
|
raise Discourse::NotFound unless post_revision
|
||||||
|
|
||||||
|
post_revision.post = find_post_from_params
|
||||||
|
guardian.ensure_can_see!(post_revision)
|
||||||
|
|
||||||
post_revision
|
post_revision
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -122,14 +122,15 @@ class TopicsController < ApplicationController
|
||||||
topic = Topic.find_by(id: params[:topic_id])
|
topic = Topic.find_by(id: params[:topic_id])
|
||||||
guardian.ensure_can_edit!(topic)
|
guardian.ensure_can_edit!(topic)
|
||||||
|
|
||||||
topic.title = params[:title] if params[:title].present?
|
changes = {}
|
||||||
topic.acting_user = current_user
|
changes[:title] = params[:title] if params[:title]
|
||||||
|
changes[:category_id] = params[:category_id] if params[:category_id]
|
||||||
|
|
||||||
success = false
|
success = true
|
||||||
Topic.transaction do
|
|
||||||
success = topic.save
|
if changes.length > 0
|
||||||
success &= topic.change_category_to_id(params[:category_id].to_i) unless topic.private_message?
|
first_post = topic.ordered_posts.first
|
||||||
EditRateLimiter.new(current_user).performed!
|
success = PostRevisor.new(first_post, topic).revise!(current_user, changes)
|
||||||
end
|
end
|
||||||
|
|
||||||
# this is used to return the title to the client as it may have been changed by "TextCleaner"
|
# this is used to return the title to the client as it may have been changed by "TextCleaner"
|
||||||
|
@ -308,21 +309,17 @@ class TopicsController < ApplicationController
|
||||||
|
|
||||||
guardian.ensure_can_change_post_owner!
|
guardian.ensure_can_change_post_owner!
|
||||||
|
|
||||||
topic = Topic.find(params[:topic_id].to_i)
|
post_ids = params[:post_ids].to_a
|
||||||
new_user = User.find_by_username(params[:username])
|
topic = Topic.find_by(id: params[:topic_id].to_i)
|
||||||
ids = params[:post_ids].to_a
|
new_user = User.find_by(username: params[:username])
|
||||||
|
|
||||||
unless new_user && topic && ids
|
return render json: failed_json, status: 422 unless post_ids && topic && new_user
|
||||||
render json: failed_json, status: 422
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
ActiveRecord::Base.transaction do
|
ActiveRecord::Base.transaction do
|
||||||
ids.each do |id|
|
post_ids.each do |post_id|
|
||||||
post = Post.find(id)
|
post = Post.find(post_id)
|
||||||
if post.is_first_post?
|
# update topic owner (first avatar)
|
||||||
topic.user = new_user # Update topic owner (first avatar)
|
topic.user = new_user if post.is_first_post?
|
||||||
end
|
|
||||||
post.set_owner(new_user, current_user)
|
post.set_owner(new_user, current_user)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -84,11 +84,10 @@ module Jobs
|
||||||
delay = SiteSetting.ninja_edit_window * args[:backoff]
|
delay = SiteSetting.ninja_edit_window * args[:backoff]
|
||||||
Jobs.enqueue_in(delay.seconds.to_i, :pull_hotlinked_images, args.merge!(backoff: backoff))
|
Jobs.enqueue_in(delay.seconds.to_i, :pull_hotlinked_images, args.merge!(backoff: backoff))
|
||||||
elsif raw != post.raw
|
elsif raw != post.raw
|
||||||
options = {
|
changes = { raw: raw, edit_reason: I18n.t("upload.edit_reason") }
|
||||||
edit_reason: I18n.t("upload.edit_reason"),
|
# we never want that job to bump the topic
|
||||||
bypass_bump: true # we never want that job to bump the topic
|
options = { bypass_bump: true }
|
||||||
}
|
post.revise(Discourse.system_user, changes, options)
|
||||||
post.revise(Discourse.system_user, raw, options)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -324,8 +324,8 @@ class Post < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def revise(updated_by, new_raw, opts = {})
|
def revise(updated_by, changes={}, opts={})
|
||||||
PostRevisor.new(self).revise!(updated_by, new_raw, opts)
|
PostRevisor.new(self).revise!(updated_by, changes, opts)
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.rebake_old(limit)
|
def self.rebake_old(limit)
|
||||||
|
@ -364,13 +364,14 @@ class Post < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
def set_owner(new_user, actor)
|
def set_owner(new_user, actor)
|
||||||
revise(actor, self.raw, {
|
return if user_id == new_user.id
|
||||||
new_user: new_user,
|
|
||||||
changed_owner: true,
|
edit_reason = I18n.t('change_owner.post_revision_text',
|
||||||
edit_reason: I18n.t('change_owner.post_revision_text',
|
|
||||||
old_user: self.user.username_lower,
|
old_user: self.user.username_lower,
|
||||||
new_user: new_user.username_lower)
|
new_user: new_user.username_lower
|
||||||
})
|
)
|
||||||
|
|
||||||
|
revise(actor, { raw: self.raw, user_id: new_user.id, edit_reason: edit_reason })
|
||||||
end
|
end
|
||||||
|
|
||||||
before_create do
|
before_create do
|
||||||
|
@ -414,14 +415,6 @@ class Post < ActiveRecord::Base
|
||||||
self.baked_version = BAKED_VERSION
|
self.baked_version = BAKED_VERSION
|
||||||
end
|
end
|
||||||
|
|
||||||
after_save do
|
|
||||||
save_revision if self.version_changed?
|
|
||||||
end
|
|
||||||
|
|
||||||
after_update do
|
|
||||||
update_revision if self.changed?
|
|
||||||
end
|
|
||||||
|
|
||||||
def advance_draft_sequence
|
def advance_draft_sequence
|
||||||
return if topic.blank? # could be deleted
|
return if topic.blank? # could be deleted
|
||||||
DraftSequence.next!(last_editor_id, topic.draft_key)
|
DraftSequence.next!(last_editor_id, topic.draft_key)
|
||||||
|
@ -537,33 +530,6 @@ class Post < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def save_revision
|
|
||||||
modifications = changes.extract!(:raw, :cooked, :edit_reason, :user_id, :wiki, :post_type)
|
|
||||||
# make sure cooked is always present (oneboxes might not change the cooked post)
|
|
||||||
modifications["cooked"] = [self.cooked, self.cooked] unless modifications["cooked"].present?
|
|
||||||
PostRevision.create!(
|
|
||||||
user_id: last_editor_id,
|
|
||||||
post_id: id,
|
|
||||||
number: version,
|
|
||||||
modifications: modifications
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
def update_revision
|
|
||||||
revision = PostRevision.find_by(post_id: id, number: version)
|
|
||||||
return unless revision
|
|
||||||
revision.user_id = last_editor_id
|
|
||||||
modifications = changes.extract!(:raw, :cooked, :edit_reason)
|
|
||||||
[:raw, :cooked, :edit_reason].each do |field|
|
|
||||||
if modifications[field].present?
|
|
||||||
old_value = revision.modifications[field].try(:[], 0) || ""
|
|
||||||
new_value = modifications[field][1]
|
|
||||||
revision.modifications[field] = [old_value, new_value]
|
|
||||||
end
|
|
||||||
end
|
|
||||||
revision.save
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# == Schema Information
|
# == Schema Information
|
||||||
|
|
|
@ -163,9 +163,7 @@ class PostAction < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
def moderator_already_replied?(topic, moderator)
|
def moderator_already_replied?(topic, moderator)
|
||||||
topic.posts
|
topic.posts.where("user_id = :user_id OR post_type = :post_type", user_id: moderator.id, post_type: Post.types[:moderator_action]).exists?
|
||||||
.where("user_id = :user_id OR post_type = :post_type", user_id: moderator.id, post_type: Post.types[:moderator_action])
|
|
||||||
.exists?
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.create_message_for_post_action(user, post, post_action_type_id, opts)
|
def self.create_message_for_post_action(user, post, post_action_type_id, opts)
|
||||||
|
|
|
@ -30,11 +30,13 @@ class PostAlertObserver < ActiveRecord::Observer
|
||||||
post = post_action.post
|
post = post_action.post
|
||||||
return if post_action.user.blank?
|
return if post_action.user.blank?
|
||||||
|
|
||||||
alerter.create_notification(post.user,
|
alerter.create_notification(
|
||||||
|
post.user,
|
||||||
Notification.types[:liked],
|
Notification.types[:liked],
|
||||||
post,
|
post,
|
||||||
display_username: post_action.user.username,
|
display_username: post_action.user.username,
|
||||||
post_action_id: post_action.id)
|
post_action_id: post_action.id
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def after_create_post_revision(post_revision)
|
def after_create_post_revision(post_revision)
|
||||||
|
@ -51,11 +53,10 @@ class PostAlertObserver < ActiveRecord::Observer
|
||||||
Notification.types[:edited],
|
Notification.types[:edited],
|
||||||
post,
|
post,
|
||||||
display_username: post_revision.user.username,
|
display_username: post_revision.user.username,
|
||||||
post_revision: post_revision
|
acting_user_id: post_revision.try(:user_id)
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
def callback_for(action, model)
|
def callback_for(action, model)
|
||||||
|
|
|
@ -8,185 +8,30 @@ class PostRevision < ActiveRecord::Base
|
||||||
|
|
||||||
def self.ensure_consistency!
|
def self.ensure_consistency!
|
||||||
# 1 - fix the numbers
|
# 1 - fix the numbers
|
||||||
sql = <<-SQL
|
PostRevision.exec_sql <<-SQL
|
||||||
UPDATE post_revisions
|
UPDATE post_revisions
|
||||||
SET number = pr.rank
|
SET number = pr.rank
|
||||||
FROM (SELECT id, ROW_NUMBER() OVER (PARTITION BY post_id ORDER BY number, created_at, updated_at) AS rank FROM post_revisions) AS pr
|
FROM (SELECT id, 1 + ROW_NUMBER() OVER (PARTITION BY post_id ORDER BY number, created_at, updated_at) AS rank FROM post_revisions) AS pr
|
||||||
WHERE post_revisions.id = pr.id
|
WHERE post_revisions.id = pr.id
|
||||||
AND post_revisions.number <> pr.rank
|
AND post_revisions.number <> pr.rank
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
PostRevision.exec_sql(sql)
|
|
||||||
|
|
||||||
# 2 - fix the versions on the posts
|
# 2 - fix the versions on the posts
|
||||||
sql = <<-SQL
|
PostRevision.exec_sql <<-SQL
|
||||||
UPDATE posts
|
UPDATE posts
|
||||||
SET version = pv.version
|
SET version = 1 + (SELECT COUNT(*) FROM post_revisions WHERE post_id = posts.id),
|
||||||
FROM (SELECT post_id, MAX(number) AS version FROM post_revisions GROUP BY post_id) AS pv
|
public_version = 1 + (SELECT COUNT(*) FROM post_revisions pr WHERE post_id = posts.id AND pr.hidden = 'f')
|
||||||
WHERE posts.id = pv.post_id
|
WHERE version <> 1 + (SELECT COUNT(*) FROM post_revisions WHERE post_id = posts.id)
|
||||||
AND posts.version <> pv.version
|
OR public_version <> 1 + (SELECT COUNT(*) FROM post_revisions pr WHERE post_id = posts.id AND pr.hidden = 'f')
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
PostRevision.exec_sql(sql)
|
|
||||||
end
|
|
||||||
|
|
||||||
def body_changes
|
|
||||||
cooked_diff = DiscourseDiff.new(previous("cooked"), current("cooked"))
|
|
||||||
raw_diff = DiscourseDiff.new(previous("raw"), current("raw"))
|
|
||||||
|
|
||||||
{
|
|
||||||
inline: cooked_diff.inline_html,
|
|
||||||
side_by_side: cooked_diff.side_by_side_html,
|
|
||||||
side_by_side_markdown: raw_diff.side_by_side_markdown
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def category_changes
|
|
||||||
prev = previous("category_id")
|
|
||||||
cur = current("category_id")
|
|
||||||
return if prev == cur
|
|
||||||
|
|
||||||
{
|
|
||||||
previous_category_id: prev,
|
|
||||||
current_category_id: cur,
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def wiki_changes
|
|
||||||
prev = previous("wiki")
|
|
||||||
cur = current("wiki")
|
|
||||||
return if prev == cur
|
|
||||||
|
|
||||||
{
|
|
||||||
previous_wiki: prev,
|
|
||||||
current_wiki: cur,
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def post_type_changes
|
|
||||||
prev = previous("post_type")
|
|
||||||
cur = current("post_type")
|
|
||||||
return if prev == cur
|
|
||||||
|
|
||||||
{
|
|
||||||
previous_post_type: prev,
|
|
||||||
current_post_type: cur,
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def title_changes
|
|
||||||
prev = "<div>#{CGI::escapeHTML(previous("title"))}</div>"
|
|
||||||
cur = "<div>#{CGI::escapeHTML(current("title"))}</div>"
|
|
||||||
return if prev == cur
|
|
||||||
|
|
||||||
diff = DiscourseDiff.new(prev, cur)
|
|
||||||
|
|
||||||
{
|
|
||||||
inline: diff.inline_html,
|
|
||||||
side_by_side: diff.side_by_side_html
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def user_changes
|
|
||||||
prev = previous("user_id")
|
|
||||||
cur = current("user_id")
|
|
||||||
return if prev == cur
|
|
||||||
|
|
||||||
{
|
|
||||||
previous_user: User.find_by(id: prev),
|
|
||||||
current_user: User.find_by(id: cur)
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def previous(field)
|
|
||||||
val = lookup(field)
|
|
||||||
if val.nil?
|
|
||||||
val = lookup_in_previous_revisions(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
if val.nil?
|
|
||||||
val = lookup_in_post(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
val
|
|
||||||
end
|
|
||||||
|
|
||||||
def current(field)
|
|
||||||
val = lookup_in_next_revision(field)
|
|
||||||
if val.nil?
|
|
||||||
val = lookup_in_post(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
if val.nil?
|
|
||||||
val = lookup(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
if val.nil?
|
|
||||||
val = lookup_in_previous_revisions(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
return val
|
|
||||||
end
|
|
||||||
|
|
||||||
def previous_revisions
|
|
||||||
@previous_revs ||= PostRevision.where("post_id = ? AND number < ? AND hidden = ?", post_id, number, false)
|
|
||||||
.order("number desc")
|
|
||||||
.to_a
|
|
||||||
end
|
|
||||||
|
|
||||||
def next_revision
|
|
||||||
@next_revision ||= PostRevision.where("post_id = ? AND number > ? AND hidden = ?", post_id, number, false)
|
|
||||||
.order("number asc")
|
|
||||||
.to_a.first
|
|
||||||
end
|
|
||||||
|
|
||||||
def has_topic_data?
|
|
||||||
post && post.post_number == 1
|
|
||||||
end
|
|
||||||
|
|
||||||
def lookup_in_previous_revisions(field)
|
|
||||||
previous_revisions.each do |v|
|
|
||||||
val = v.lookup(field)
|
|
||||||
return val unless val.nil?
|
|
||||||
end
|
|
||||||
|
|
||||||
nil
|
|
||||||
end
|
|
||||||
|
|
||||||
def lookup_in_next_revision(field)
|
|
||||||
if next_revision
|
|
||||||
return next_revision.lookup(field)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def lookup_in_post(field)
|
|
||||||
if !post
|
|
||||||
return
|
|
||||||
elsif ["cooked", "raw"].include?(field)
|
|
||||||
val = post.send(field)
|
|
||||||
elsif ["title", "category_id"].include?(field)
|
|
||||||
val = post.topic.send(field)
|
|
||||||
end
|
|
||||||
|
|
||||||
val
|
|
||||||
end
|
|
||||||
|
|
||||||
def lookup(field)
|
|
||||||
return nil if hidden
|
|
||||||
mod = modifications[field]
|
|
||||||
unless mod.nil?
|
|
||||||
mod[0]
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def hide!
|
def hide!
|
||||||
self.hidden = true
|
update_column(:hidden, true)
|
||||||
self.save!
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def show!
|
def show!
|
||||||
self.hidden = false
|
update_column(:hidden, false)
|
||||||
self.save!
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -99,7 +99,6 @@ class Topic < ActiveRecord::Base
|
||||||
has_many :topic_invites
|
has_many :topic_invites
|
||||||
has_many :invites, through: :topic_invites, source: :invite
|
has_many :invites, through: :topic_invites, source: :invite
|
||||||
|
|
||||||
has_many :revisions, foreign_key: :topic_id, class_name: 'TopicRevision'
|
|
||||||
has_one :warning
|
has_one :warning
|
||||||
|
|
||||||
# When we want to temporarily attach some data to a forum topic (usually before serialization)
|
# When we want to temporarily attach some data to a forum topic (usually before serialization)
|
||||||
|
@ -148,71 +147,72 @@ class Topic < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
attr_accessor :ignore_category_auto_close
|
attr_accessor :ignore_category_auto_close
|
||||||
|
attr_accessor :skip_callbacks
|
||||||
|
|
||||||
before_create do
|
before_create do
|
||||||
|
initialize_default_values
|
||||||
|
inherit_auto_close_from_category
|
||||||
|
end
|
||||||
|
|
||||||
|
after_create do
|
||||||
|
unless skip_callbacks
|
||||||
|
changed_to_category(category)
|
||||||
|
advance_draft_sequence
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
before_save do
|
||||||
|
unless skip_callbacks
|
||||||
|
cancel_auto_close_job
|
||||||
|
ensure_topic_has_a_category
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
after_save do
|
||||||
|
unless skip_callbacks
|
||||||
|
schedule_auto_close_job
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def initialize_default_values
|
||||||
self.bumped_at ||= Time.now
|
self.bumped_at ||= Time.now
|
||||||
self.last_post_user_id ||= user_id
|
self.last_post_user_id ||= user_id
|
||||||
|
end
|
||||||
|
|
||||||
if !@ignore_category_auto_close and self.category and self.category.auto_close_hours and self.auto_close_at.nil?
|
def inherit_auto_close_from_category
|
||||||
|
if !@ignore_category_auto_close && self.category && self.category.auto_close_hours && self.auto_close_at.nil?
|
||||||
self.auto_close_based_on_last_post = self.category.auto_close_based_on_last_post
|
self.auto_close_based_on_last_post = self.category.auto_close_based_on_last_post
|
||||||
set_auto_close(self.category.auto_close_hours)
|
set_auto_close(self.category.auto_close_hours)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
attr_accessor :skip_callbacks
|
def advance_draft_sequence
|
||||||
|
|
||||||
after_create do
|
|
||||||
unless skip_callbacks
|
|
||||||
changed_to_category(category)
|
|
||||||
if archetype == Archetype.private_message
|
if archetype == Archetype.private_message
|
||||||
DraftSequence.next!(user, Draft::NEW_PRIVATE_MESSAGE)
|
DraftSequence.next!(user, Draft::NEW_PRIVATE_MESSAGE)
|
||||||
else
|
else
|
||||||
DraftSequence.next!(user, Draft::NEW_TOPIC)
|
DraftSequence.next!(user, Draft::NEW_TOPIC)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def cancel_auto_close_job
|
||||||
|
if (auto_close_at_changed? && !auto_close_at_was.nil?) || (auto_close_user_id_changed? && auto_close_at)
|
||||||
|
self.auto_close_started_at ||= Time.zone.now if auto_close_at
|
||||||
|
Jobs.cancel_scheduled_job(:close_topic, { topic_id: id })
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
before_save do
|
def schedule_auto_close_job
|
||||||
unless skip_callbacks
|
if auto_close_at && (auto_close_at_changed? || auto_close_user_id_changed?)
|
||||||
if (auto_close_at_changed? and !auto_close_at_was.nil?) or (auto_close_user_id_changed? and auto_close_at)
|
options = { topic_id: id, user_id: auto_close_user_id || user_id }
|
||||||
self.auto_close_started_at ||= Time.zone.now if auto_close_at
|
Jobs.enqueue_at(auto_close_at, :close_topic, options)
|
||||||
Jobs.cancel_scheduled_job(:close_topic, {topic_id: id})
|
|
||||||
end
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def ensure_topic_has_a_category
|
||||||
if category_id.nil? && (archetype.nil? || archetype == Archetype.default)
|
if category_id.nil? && (archetype.nil? || archetype == Archetype.default)
|
||||||
self.category_id = SiteSetting.uncategorized_category_id
|
self.category_id = SiteSetting.uncategorized_category_id
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
|
||||||
|
|
||||||
after_save do
|
|
||||||
save_revision if should_create_new_version?
|
|
||||||
|
|
||||||
unless skip_callbacks
|
|
||||||
if auto_close_at and (auto_close_at_changed? or auto_close_user_id_changed?)
|
|
||||||
Jobs.enqueue_at(auto_close_at, :close_topic, {topic_id: id, user_id: auto_close_user_id || user_id})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
# TODO move into PostRevisor or TopicRevisor
|
|
||||||
def save_revision
|
|
||||||
if first_post_id = posts.where(post_number: 1).pluck(:id).first
|
|
||||||
|
|
||||||
number = PostRevision.where(post_id: first_post_id).count + 2
|
|
||||||
PostRevision.create!(
|
|
||||||
user_id: acting_user.id,
|
|
||||||
post_id: first_post_id,
|
|
||||||
number: number,
|
|
||||||
modifications: changes.extract!(:category_id, :title)
|
|
||||||
)
|
|
||||||
|
|
||||||
Post.where(id: first_post_id).update_all(version: number)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def should_create_new_version?
|
|
||||||
!new_record? && (category_id_changed? || title_changed?)
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.top_viewed(max = 10)
|
def self.top_viewed(max = 10)
|
||||||
Topic.listable_topics.visible.secured.order('views desc').limit(max)
|
Topic.listable_topics.visible.secured.order('views desc').limit(max)
|
||||||
|
@ -266,10 +266,6 @@ class Topic < ActiveRecord::Base
|
||||||
Redcarpet::Render::SmartyPants.render(sanitized_title)
|
Redcarpet::Render::SmartyPants.render(sanitized_title)
|
||||||
end
|
end
|
||||||
|
|
||||||
def new_version_required?
|
|
||||||
title_changed? || category_id_changed?
|
|
||||||
end
|
|
||||||
|
|
||||||
# Returns hot topics since a date for display in email digest.
|
# Returns hot topics since a date for display in email digest.
|
||||||
def self.for_digest(user, since, opts=nil)
|
def self.for_digest(user, since, opts=nil)
|
||||||
opts = opts || {}
|
opts = opts || {}
|
||||||
|
@ -383,10 +379,8 @@ class Topic < ActiveRecord::Base
|
||||||
|
|
||||||
candidate_ids = candidates.pluck(:id)
|
candidate_ids = candidates.pluck(:id)
|
||||||
|
|
||||||
|
|
||||||
return [] unless candidate_ids.present?
|
return [] unless candidate_ids.present?
|
||||||
|
|
||||||
|
|
||||||
similar = Topic.select(sanitize_sql_array(["topics.*, similarity(topics.title, :title) + similarity(topics.title, :raw) AS similarity", title: title, raw: raw]))
|
similar = Topic.select(sanitize_sql_array(["topics.*, similarity(topics.title, :title) + similarity(topics.title, :raw) AS similarity", title: title, raw: raw]))
|
||||||
.joins("JOIN posts AS p ON p.topic_id = topics.id AND p.post_number = 1")
|
.joins("JOIN posts AS p ON p.topic_id = topics.id AND p.post_number = 1")
|
||||||
.limit(SiteSetting.max_similar_results)
|
.limit(SiteSetting.max_similar_results)
|
||||||
|
@ -451,37 +445,30 @@ class Topic < ActiveRecord::Base
|
||||||
(topics.avg_time <> x.gmean OR topics.avg_time IS NULL)")
|
(topics.avg_time <> x.gmean OR topics.avg_time IS NULL)")
|
||||||
|
|
||||||
if min_topic_age
|
if min_topic_age
|
||||||
builder.where("topics.bumped_at > :bumped_at",
|
builder.where("topics.bumped_at > :bumped_at", bumped_at: min_topic_age)
|
||||||
bumped_at: min_topic_age)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
builder.exec
|
builder.exec
|
||||||
end
|
end
|
||||||
|
|
||||||
def changed_to_category(cat)
|
def changed_to_category(new_category)
|
||||||
return true if cat.blank? || Category.find_by(topic_id: id).present?
|
return true if new_category.blank? || Category.find_by(topic_id: id).present?
|
||||||
|
return false if new_category.id == SiteSetting.uncategorized_category_id && !SiteSetting.allow_uncategorized_topics
|
||||||
|
|
||||||
Topic.transaction do
|
Topic.transaction do
|
||||||
old_category = category
|
old_category = category
|
||||||
|
|
||||||
if category_id.present? && category_id != cat.id
|
if self.category_id != new_category.id
|
||||||
Category.where(['id = ?', category_id]).update_all 'topic_count = topic_count - 1'
|
self.category_id = new_category.id
|
||||||
|
self.update_column(:category_id, new_category.id)
|
||||||
|
Category.where(id: old_category.id).update_all("topic_count = topic_count - 1") if old_category
|
||||||
end
|
end
|
||||||
|
|
||||||
success = true
|
Category.where(id: new_category.id).update_all("topic_count = topic_count + 1")
|
||||||
if self.category_id != cat.id
|
|
||||||
self.category_id = cat.id
|
|
||||||
success = save
|
|
||||||
end
|
|
||||||
|
|
||||||
if success
|
|
||||||
CategoryFeaturedTopic.feature_topics_for(old_category) unless @import_mode
|
CategoryFeaturedTopic.feature_topics_for(old_category) unless @import_mode
|
||||||
Category.where(id: cat.id).update_all 'topic_count = topic_count + 1'
|
CategoryFeaturedTopic.feature_topics_for(new_category) unless @import_mode || old_category.id == new_category.id
|
||||||
CategoryFeaturedTopic.feature_topics_for(cat) unless @import_mode || old_category.try(:id) == cat.try(:id)
|
|
||||||
else
|
|
||||||
return false
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -513,15 +500,15 @@ class Topic < ActiveRecord::Base
|
||||||
def change_category_to_id(category_id)
|
def change_category_to_id(category_id)
|
||||||
return false if private_message?
|
return false if private_message?
|
||||||
|
|
||||||
# If the category name is blank, reset the attribute
|
new_category_id = category_id.to_i
|
||||||
if (category_id.nil? || category_id.to_i == 0)
|
# if the category name is blank, reset the attribute
|
||||||
cat = Category.find_by(id: SiteSetting.uncategorized_category_id)
|
new_category_id = SiteSetting.uncategorized_category_id if new_category_id == 0
|
||||||
else
|
|
||||||
cat = Category.where(id: category_id).first
|
|
||||||
end
|
|
||||||
|
|
||||||
return true if cat == category
|
return true if self.category_id == new_category_id
|
||||||
|
|
||||||
|
cat = Category.find_by(id: new_category_id)
|
||||||
return false unless cat
|
return false unless cat
|
||||||
|
|
||||||
changed_to_category(cat)
|
changed_to_category(cat)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -50,8 +50,7 @@ class TopicEmbed < ActiveRecord::Base
|
||||||
post = embed.post
|
post = embed.post
|
||||||
# Update the topic if it changed
|
# Update the topic if it changed
|
||||||
if post && post.topic && content_sha1 != embed.content_sha1
|
if post && post.topic && content_sha1 != embed.content_sha1
|
||||||
revisor = PostRevisor.new(post)
|
post.revise(user, { raw: absolutize_urls(url, contents) }, skip_validations: true, bypass_rate_limiter: true)
|
||||||
revisor.revise!(user, absolutize_urls(url, contents), skip_validations: true, bypass_rate_limiter: true)
|
|
||||||
embed.update_column(:content_sha1, content_sha1)
|
embed.update_column(:content_sha1, content_sha1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -33,7 +33,7 @@ class UserActionObserver < ActiveRecord::Observer
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.log_notification(post, user, notification_type, acting_user_id = nil)
|
def self.log_notification(post, user, notification_type, acting_user_id=nil)
|
||||||
action =
|
action =
|
||||||
case notification_type
|
case notification_type
|
||||||
when Notification.types[:quoted]
|
when Notification.types[:quoted]
|
||||||
|
|
|
@ -1,38 +1,70 @@
|
||||||
class PostRevisionSerializer < ApplicationSerializer
|
class PostRevisionSerializer < ApplicationSerializer
|
||||||
attributes :post_id,
|
|
||||||
:version,
|
attributes :created_at,
|
||||||
:revisions_count,
|
:post_id,
|
||||||
|
# which revision is hidden
|
||||||
|
:previous_hidden,
|
||||||
|
:current_hidden,
|
||||||
|
# dynamic & based on the current scope
|
||||||
|
:first_revision,
|
||||||
|
:previous_revision,
|
||||||
|
:current_revision,
|
||||||
|
:next_revision,
|
||||||
|
:last_revision,
|
||||||
|
# used for display
|
||||||
|
:current_version,
|
||||||
|
:version_count,
|
||||||
|
# from the user
|
||||||
:username,
|
:username,
|
||||||
:display_username,
|
:display_username,
|
||||||
:avatar_template,
|
:avatar_template,
|
||||||
:created_at,
|
# all the changes
|
||||||
:edit_reason,
|
:edit_reason,
|
||||||
:body_changes,
|
:body_changes,
|
||||||
:title_changes,
|
:title_changes,
|
||||||
:category_changes,
|
:category_changes,
|
||||||
:user_changes,
|
:user_changes,
|
||||||
:wiki_changes,
|
:wiki_changes,
|
||||||
:post_type_changes,
|
:post_type_changes
|
||||||
:hidden
|
|
||||||
|
|
||||||
def include_title_changes?
|
def previous_hidden
|
||||||
object.has_topic_data?
|
previous["hidden"]
|
||||||
end
|
end
|
||||||
|
|
||||||
def include_category_changes?
|
def current_hidden
|
||||||
object.has_topic_data?
|
current["hidden"]
|
||||||
end
|
end
|
||||||
|
|
||||||
def hidden
|
def first_revision
|
||||||
object.hidden
|
revisions.first["revision"]
|
||||||
end
|
end
|
||||||
|
|
||||||
def version
|
def previous_revision
|
||||||
|
@previous_revision ||= revisions.select { |r| r["revision"] >= first_revision }
|
||||||
|
.select { |r| r["revision"] < current_revision }
|
||||||
|
.last.try(:[], "revision")
|
||||||
|
end
|
||||||
|
|
||||||
|
def current_revision
|
||||||
object.number
|
object.number
|
||||||
end
|
end
|
||||||
|
|
||||||
def revisions_count
|
def next_revision
|
||||||
object.post.version
|
@next_revision ||= revisions.select { |r| r["revision"] <= last_revision }
|
||||||
|
.select { |r| r["revision"] > current_revision }
|
||||||
|
.first.try(:[], "revision")
|
||||||
|
end
|
||||||
|
|
||||||
|
def last_revision
|
||||||
|
@last_revision ||= revisions.select { |r| r["revision"] <= post.version }.last["revision"]
|
||||||
|
end
|
||||||
|
|
||||||
|
def current_version
|
||||||
|
@current_version ||= revisions.select { |r| r["revision"] <= current_revision }.count + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
def version_count
|
||||||
|
revisions.count
|
||||||
end
|
end
|
||||||
|
|
||||||
def username
|
def username
|
||||||
|
@ -48,29 +80,160 @@ class PostRevisionSerializer < ApplicationSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def edit_reason
|
def edit_reason
|
||||||
object.current("edit_reason")
|
# only show 'edit_reason' when revisions are consecutive
|
||||||
|
current["edit_reason"] if scope.can_view_hidden_post_revisions? ||
|
||||||
|
current["revision"] == previous["revision"] + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
def body_changes
|
||||||
|
cooked_diff = DiscourseDiff.new(previous["cooked"], current["cooked"])
|
||||||
|
raw_diff = DiscourseDiff.new(previous["raw"], current["raw"])
|
||||||
|
|
||||||
|
{
|
||||||
|
inline: cooked_diff.inline_html,
|
||||||
|
side_by_side: cooked_diff.side_by_side_html,
|
||||||
|
side_by_side_markdown: raw_diff.side_by_side_markdown
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def title_changes
|
||||||
|
prev = "<div>#{CGI::escapeHTML(previous["title"])}</div>"
|
||||||
|
cur = "<div>#{CGI::escapeHTML(current["title"])}</div>"
|
||||||
|
return if prev == cur
|
||||||
|
|
||||||
|
diff = DiscourseDiff.new(prev, cur)
|
||||||
|
|
||||||
|
{
|
||||||
|
inline: diff.inline_html,
|
||||||
|
side_by_side: diff.side_by_side_html
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def category_changes
|
||||||
|
prev = previous["category_id"]
|
||||||
|
cur = current["category_id"]
|
||||||
|
return if prev == cur
|
||||||
|
|
||||||
|
{
|
||||||
|
previous: prev,
|
||||||
|
current: cur,
|
||||||
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def user_changes
|
def user_changes
|
||||||
obj = object.user_changes
|
prev = previous["user_id"]
|
||||||
return unless obj
|
cur = current["user_id"]
|
||||||
# same as below - if stuff is messed up, default to system
|
return if prev == cur
|
||||||
prev = obj[:previous_user] || Discourse.system_user
|
|
||||||
new = obj[:current_user] || Discourse.system_user
|
# if stuff is messed up, default to system
|
||||||
|
previous = User.find_by(id: prev) || Discourse.system_user
|
||||||
|
current = User.find_by(id: cur) || Discourse.system_user
|
||||||
|
|
||||||
{
|
{
|
||||||
previous: {
|
previous: {
|
||||||
username: prev.username_lower,
|
username: previous.username_lower,
|
||||||
display_username: prev.username,
|
display_username: previous.username,
|
||||||
avatar_template: prev.avatar_template
|
avatar_template: previous.avatar_template
|
||||||
},
|
},
|
||||||
current: {
|
current: {
|
||||||
username: new.username_lower,
|
username: current.username_lower,
|
||||||
display_username: new.username,
|
display_username: current.username,
|
||||||
avatar_template: new.avatar_template
|
avatar_template: current.avatar_template
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def wiki_changes
|
||||||
|
prev = previous["wiki"]
|
||||||
|
cur = current["wiki"]
|
||||||
|
return if prev == cur
|
||||||
|
|
||||||
|
{
|
||||||
|
previous: prev,
|
||||||
|
current: cur,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def post_type_changes
|
||||||
|
prev = previous["post_type"]
|
||||||
|
cur = current["post_type"]
|
||||||
|
return if prev == cur
|
||||||
|
|
||||||
|
{
|
||||||
|
previous: prev,
|
||||||
|
current: cur,
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
protected
|
||||||
|
|
||||||
|
def post
|
||||||
|
@post ||= object.post
|
||||||
|
end
|
||||||
|
|
||||||
|
def topic
|
||||||
|
@topic ||= object.post.topic
|
||||||
|
end
|
||||||
|
|
||||||
|
def revisions
|
||||||
|
@revisions ||= all_revisions.select { |r| scope.can_view_hidden_post_revisions? || !r["hidden"] }
|
||||||
|
end
|
||||||
|
|
||||||
|
def all_revisions
|
||||||
|
return @all_revisions if @all_revisions
|
||||||
|
|
||||||
|
post_revisions = PostRevision.where(post_id: object.post_id).order(:number).to_a
|
||||||
|
post_revisions << PostRevision.new(
|
||||||
|
number: post_revisions.last.number + 1,
|
||||||
|
hidden: post.hidden,
|
||||||
|
modifications: {
|
||||||
|
"raw" => [post.raw],
|
||||||
|
"cooked" => [post.cooked],
|
||||||
|
"edit_reason" => [post.edit_reason],
|
||||||
|
"wiki" => [post.wiki],
|
||||||
|
"post_type" => [post.post_type],
|
||||||
|
"user_id" => [post.user_id],
|
||||||
|
"title" => [topic.title],
|
||||||
|
"category_id" => [topic.category_id],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
@all_revisions = []
|
||||||
|
|
||||||
|
# backtrack
|
||||||
|
post_revisions.each do |pr|
|
||||||
|
revision = HashWithIndifferentAccess.new
|
||||||
|
revision[:revision] = pr.number
|
||||||
|
revision[:hidden] = pr.hidden
|
||||||
|
|
||||||
|
pr.modifications.keys.each do |field|
|
||||||
|
revision[field] = pr.modifications[field][0]
|
||||||
|
end
|
||||||
|
|
||||||
|
@all_revisions << revision
|
||||||
|
end
|
||||||
|
|
||||||
|
# waterfall
|
||||||
|
(@all_revisions.count - 1).downto(1).each do |r|
|
||||||
|
cur = @all_revisions[r]
|
||||||
|
prev = @all_revisions[r - 1]
|
||||||
|
|
||||||
|
cur.keys.each do |field|
|
||||||
|
prev[field] = prev.has_key?(field) ? prev[field] : cur[field]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@all_revisions
|
||||||
|
end
|
||||||
|
|
||||||
|
def previous
|
||||||
|
@previous ||= revisions.select { |r| r["revision"] <= current_revision }.last
|
||||||
|
end
|
||||||
|
|
||||||
|
def current
|
||||||
|
@current ||= revisions.select { |r| r["revision"] > current_revision }.first
|
||||||
|
end
|
||||||
|
|
||||||
def user
|
def user
|
||||||
# if stuff goes pear shape attribute to system
|
# if stuff goes pear shape attribute to system
|
||||||
object.user || Discourse.system_user
|
object.user || Discourse.system_user
|
||||||
|
|
|
@ -233,7 +233,7 @@ class PostSerializer < BasicPostSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def can_view_edit_history
|
def can_view_edit_history
|
||||||
scope.can_view_post_revisions?(object)
|
scope.can_view_edit_history?(object)
|
||||||
end
|
end
|
||||||
|
|
||||||
def user_custom_fields
|
def user_custom_fields
|
||||||
|
@ -258,6 +258,10 @@ class PostSerializer < BasicPostSerializer
|
||||||
object.via_email?
|
object.via_email?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def version
|
||||||
|
scope.is_staff? ? object.version : object.public_version
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def post_actions
|
def post_actions
|
||||||
|
|
|
@ -94,16 +94,14 @@ class PostAlerter
|
||||||
.order("notifications.id desc")
|
.order("notifications.id desc")
|
||||||
.find_by(notification_type: type, topic_id: post.topic_id, post_number: post.post_number)
|
.find_by(notification_type: type, topic_id: post.topic_id, post_number: post.post_number)
|
||||||
|
|
||||||
if(existing_notification)
|
if existing_notification
|
||||||
return unless existing_notification.notification_type == Notification.types[:edited] &&
|
return unless existing_notification.notification_type == Notification.types[:edited] &&
|
||||||
existing_notification.data_hash["display_username"] = opts[:display_username]
|
existing_notification.data_hash["display_username"] = opts[:display_username]
|
||||||
end
|
end
|
||||||
|
|
||||||
collapsed = false
|
collapsed = false
|
||||||
|
|
||||||
if type == Notification.types[:replied] ||
|
if type == Notification.types[:replied] || type == Notification.types[:posted]
|
||||||
type == Notification.types[:posted]
|
|
||||||
|
|
||||||
destroy_notifications(user, Notification.types[:replied] , post.topic)
|
destroy_notifications(user, Notification.types[:replied] , post.topic)
|
||||||
destroy_notifications(user, Notification.types[:posted] , post.topic)
|
destroy_notifications(user, Notification.types[:posted] , post.topic)
|
||||||
collapsed = true
|
collapsed = true
|
||||||
|
@ -123,7 +121,7 @@ class PostAlerter
|
||||||
opts[:display_username] = I18n.t('embed.replies', count: count) if count > 1
|
opts[:display_username] = I18n.t('embed.replies', count: count) if count > 1
|
||||||
end
|
end
|
||||||
|
|
||||||
UserActionObserver.log_notification(original_post, user, type, opts[:post_revision].try(:user_id))
|
UserActionObserver.log_notification(original_post, user, type, opts[:acting_user_id])
|
||||||
|
|
||||||
# Create the notification
|
# Create the notification
|
||||||
user.notifications.create(notification_type: type,
|
user.notifications.create(notification_type: type,
|
||||||
|
|
|
@ -282,9 +282,10 @@ Discourse::Application.routes.draw do
|
||||||
put "rebake"
|
put "rebake"
|
||||||
put "unhide"
|
put "unhide"
|
||||||
get "replies"
|
get "replies"
|
||||||
get "revisions/:revision" => "posts#revisions"
|
get "revisions/latest" => "posts#latest_revision"
|
||||||
put "revisions/:revision/hide" => "posts#hide_revision"
|
get "revisions/:revision" => "posts#revisions", constraints: { revision: /\d+/ }
|
||||||
put "revisions/:revision/show" => "posts#show_revision"
|
put "revisions/:revision/hide" => "posts#hide_revision", constraints: { revision: /\d+/ }
|
||||||
|
put "revisions/:revision/show" => "posts#show_revision", constraints: { revision: /\d+/ }
|
||||||
put "recover"
|
put "recover"
|
||||||
collection do
|
collection do
|
||||||
delete "destroy_many"
|
delete "destroy_many"
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
class AddPublicVersionToPosts < ActiveRecord::Migration
|
||||||
|
def up
|
||||||
|
add_column :posts, :public_version, :integer, null: false, default: 1
|
||||||
|
|
||||||
|
execute <<-SQL
|
||||||
|
UPDATE posts
|
||||||
|
SET public_version = 1 + (SELECT COUNT(*) FROM post_revisions pr WHERE post_id = posts.id AND pr.hidden = 'f')
|
||||||
|
WHERE public_version <> 1 + (SELECT COUNT(*) FROM post_revisions pr WHERE post_id = posts.id AND pr.hidden = 'f')
|
||||||
|
SQL
|
||||||
|
end
|
||||||
|
|
||||||
|
def down
|
||||||
|
remove_column :posts, :public_version
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,47 +0,0 @@
|
||||||
#
|
|
||||||
# Support delegating after_create to an appropriate helper for that class name.
|
|
||||||
# For example, an observer on post will call after_create_post if that method
|
|
||||||
# is defined.
|
|
||||||
#
|
|
||||||
# It does this after_commit by default, and contains a hack to make this work
|
|
||||||
# even in test mode.
|
|
||||||
#
|
|
||||||
class DiscourseObserver < ActiveRecord::Observer
|
|
||||||
|
|
||||||
def after_create_delegator(model)
|
|
||||||
observer_method = :"after_create_#{model.class.name.underscore}"
|
|
||||||
send(observer_method, model) if respond_to?(observer_method)
|
|
||||||
end
|
|
||||||
|
|
||||||
def after_destroy_delegator(model)
|
|
||||||
observer_method = :"after_destroy_#{model.class.name.underscore}"
|
|
||||||
send(observer_method, model) if respond_to?(observer_method)
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
if Rails.env.test?
|
|
||||||
|
|
||||||
# In test mode, call the delegator right away
|
|
||||||
class DiscourseObserver < ActiveRecord::Observer
|
|
||||||
alias_method :after_create, :after_create_delegator
|
|
||||||
alias_method :after_destroy, :after_destroy_delegator
|
|
||||||
end
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
# Outside of test mode, use after_commit
|
|
||||||
class DiscourseObserver < ActiveRecord::Observer
|
|
||||||
def after_commit(model)
|
|
||||||
if model.send(:transaction_include_any_action?, [:create])
|
|
||||||
after_create_delegator(model)
|
|
||||||
end
|
|
||||||
|
|
||||||
if model.send(:transaction_include_any_action?, [:destroy])
|
|
||||||
after_destroy_delegator(model)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ require_dependency 'guardian/ensure_magic'
|
||||||
require_dependency 'guardian/post_guardian'
|
require_dependency 'guardian/post_guardian'
|
||||||
require_dependency 'guardian/topic_guardian'
|
require_dependency 'guardian/topic_guardian'
|
||||||
require_dependency 'guardian/user_guardian'
|
require_dependency 'guardian/user_guardian'
|
||||||
|
require_dependency 'guardian/post_revision_guardian'
|
||||||
|
|
||||||
# The guardian is responsible for confirming access to various site resources and operations
|
# The guardian is responsible for confirming access to various site resources and operations
|
||||||
class Guardian
|
class Guardian
|
||||||
|
@ -11,6 +12,7 @@ class Guardian
|
||||||
include PostGuardian
|
include PostGuardian
|
||||||
include TopicGuardian
|
include TopicGuardian
|
||||||
include UserGuardian
|
include UserGuardian
|
||||||
|
include PostRevisionGuardian
|
||||||
|
|
||||||
class AnonymousUser
|
class AnonymousUser
|
||||||
def blank?; true; end
|
def blank?; true; end
|
||||||
|
|
|
@ -140,12 +140,7 @@ module PostGuardian
|
||||||
can_see_topic?(post.topic)))
|
can_see_topic?(post.topic)))
|
||||||
end
|
end
|
||||||
|
|
||||||
def can_see_post_revision?(post_revision)
|
def can_view_edit_history?(post)
|
||||||
return false unless post_revision
|
|
||||||
can_view_post_revisions?(post_revision.post)
|
|
||||||
end
|
|
||||||
|
|
||||||
def can_view_post_revisions?(post)
|
|
||||||
return false unless post
|
return false unless post
|
||||||
|
|
||||||
if !post.hidden
|
if !post.hidden
|
||||||
|
@ -157,14 +152,6 @@ module PostGuardian
|
||||||
can_see_post?(post)
|
can_see_post?(post)
|
||||||
end
|
end
|
||||||
|
|
||||||
def can_hide_post_revision?(post_revision)
|
|
||||||
is_staff?
|
|
||||||
end
|
|
||||||
|
|
||||||
def can_show_post_revision?(post_revision)
|
|
||||||
is_staff?
|
|
||||||
end
|
|
||||||
|
|
||||||
def can_vote?(post, opts={})
|
def can_vote?(post, opts={})
|
||||||
post_can_act?(post,:vote, opts)
|
post_can_act?(post,:vote, opts)
|
||||||
end
|
end
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
# mixin for all Guardian methods dealing with post_revisions permissions
|
||||||
|
module PostRevisionGuardian
|
||||||
|
|
||||||
|
def can_see_post_revision?(post_revision)
|
||||||
|
return false unless post_revision
|
||||||
|
return false if post_revision.hidden && !can_view_hidden_post_revisions?
|
||||||
|
|
||||||
|
can_view_edit_history?(post_revision.post)
|
||||||
|
end
|
||||||
|
|
||||||
|
def can_hide_post_revision?(post_revision)
|
||||||
|
is_staff?
|
||||||
|
end
|
||||||
|
|
||||||
|
def can_show_post_revision?(post_revision)
|
||||||
|
is_staff?
|
||||||
|
end
|
||||||
|
|
||||||
|
def can_view_hidden_post_revisions?
|
||||||
|
is_staff?
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -96,7 +96,7 @@ class PostDestroyer
|
||||||
# When a user 'deletes' their own post. We just change the text.
|
# When a user 'deletes' their own post. We just change the text.
|
||||||
def mark_for_deletion
|
def mark_for_deletion
|
||||||
Post.transaction do
|
Post.transaction do
|
||||||
@post.revise(@user, I18n.t('js.post.deleted_by_author', count: SiteSetting.delete_removed_posts_after), force_new_version: true)
|
@post.revise(@user, { raw: I18n.t('js.post.deleted_by_author', count: SiteSetting.delete_removed_posts_after) }, force_new_version: true)
|
||||||
@post.update_column(:user_deleted, true)
|
@post.update_column(:user_deleted, true)
|
||||||
@post.update_flagged_posts_count
|
@post.update_flagged_posts_count
|
||||||
@post.topic_links.each(&:destroy)
|
@post.topic_links.each(&:destroy)
|
||||||
|
@ -110,7 +110,7 @@ class PostDestroyer
|
||||||
Post.transaction do
|
Post.transaction do
|
||||||
@post.update_column(:user_deleted, false)
|
@post.update_column(:user_deleted, false)
|
||||||
@post.skip_unique_check = true
|
@post.skip_unique_check = true
|
||||||
@post.revise(@user, @post.revisions.last.modifications["raw"][0], force_new_version: true)
|
@post.revise(@user, { raw: @post.revisions.last.modifications["raw"][0] }, force_new_version: true)
|
||||||
@post.update_flagged_posts_count
|
@post.update_flagged_posts_count
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,138 +1,269 @@
|
||||||
require 'edit_rate_limiter'
|
require "edit_rate_limiter"
|
||||||
|
|
||||||
class PostRevisor
|
class PostRevisor
|
||||||
|
|
||||||
|
POST_TRACKED_FIELDS = %w{raw cooked edit_reason user_id wiki post_type}
|
||||||
|
TOPIC_TRACKED_FIELDS = %w{title category_id}
|
||||||
|
|
||||||
attr_reader :category_changed
|
attr_reader :category_changed
|
||||||
|
|
||||||
def initialize(post)
|
def initialize(post, topic=nil)
|
||||||
@post = post
|
@post = post
|
||||||
|
@topic = topic || post.topic
|
||||||
end
|
end
|
||||||
|
|
||||||
# Recognized options:
|
# AVAILABLE OPTIONS:
|
||||||
# :edit_reason User-supplied edit reason
|
# - revised_at: changes the date of the revision
|
||||||
# :new_user New owner of the post
|
# - force_new_version: bypass ninja-edit window
|
||||||
# :revised_at changes the date of the revision
|
# - bypass_rate_limiter:
|
||||||
# :force_new_version bypass ninja-edit window
|
# - bypass_bump: do not bump the topic, even if last post
|
||||||
# :bypass_bump do not bump the topic, even if last post
|
# - skip_validations: ask ActiveRecord to skip validations
|
||||||
# :skip_validation ask ActiveRecord to skip validations
|
def revise!(editor, fields, opts={})
|
||||||
#
|
|
||||||
def revise!(editor, new_raw, opts = {})
|
|
||||||
@editor = editor
|
@editor = editor
|
||||||
|
@fields = fields.with_indifferent_access
|
||||||
@opts = opts
|
@opts = opts
|
||||||
@new_raw = TextCleaner.normalize_whitespaces(new_raw).gsub(/\s+\z/, "")
|
|
||||||
|
# some normalization
|
||||||
|
@fields[:raw] = cleanup_whitespaces(@fields[:raw]) if @fields.has_key?(:raw)
|
||||||
|
@fields[:user_id] = @fields[:user_id].to_i if @fields.has_key?(:user_id)
|
||||||
|
@fields[:category_id] = @fields[:category_id].to_i if @fields.has_key?(:category_id)
|
||||||
|
|
||||||
|
# always reset edit_reason unless provided
|
||||||
|
@fields[:edit_reason] = nil unless @fields.has_key?(:edit_reason)
|
||||||
|
|
||||||
return false unless should_revise?
|
return false unless should_revise?
|
||||||
|
|
||||||
@post.acting_user = @editor
|
@post.acting_user = @editor
|
||||||
|
@topic.acting_user = @editor
|
||||||
|
@revised_at = @opts[:revised_at] || Time.now
|
||||||
|
@last_version_at = @post.last_version_at || Time.now
|
||||||
|
|
||||||
|
@version_changed = false
|
||||||
|
@post_successfully_saved = true
|
||||||
|
@topic_successfully_saved = true
|
||||||
|
|
||||||
Post.transaction do
|
Post.transaction do
|
||||||
revise_post
|
revise_post
|
||||||
|
|
||||||
# TODO these callbacks are being called in a transaction
|
# TODO: these callbacks are being called in a transaction
|
||||||
# it is kind of odd, cause the callback is called before_edit
|
# it is kind of odd, because the callback is called "before_edit"
|
||||||
# but the post is already edited at this point
|
# but the post is already edited at this point
|
||||||
# trouble is that much of the logic of should I edit? is deeper
|
# Trouble is that much of the logic of should I edit? is deeper
|
||||||
# down so yanking this in front of the transaction will lead to
|
# down so yanking this in front of the transaction will lead to
|
||||||
# false positives. This system needs a review
|
# false positive.
|
||||||
plugin_callbacks
|
plugin_callbacks
|
||||||
|
|
||||||
update_category_description
|
revise_topic
|
||||||
update_topic_excerpt
|
advance_draft_sequence
|
||||||
@post.advance_draft_sequence
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# WARNING: do not pull this into the transaction, it can fire events in
|
# WARNING: do not pull this into the transaction
|
||||||
# sidekiq before the post is done saving leading to corrupt state
|
# it can fire events in sidekiq before the post is done saving
|
||||||
|
# leading to corrupt state
|
||||||
post_process_post
|
post_process_post
|
||||||
|
|
||||||
update_topic_word_counts
|
update_topic_word_counts
|
||||||
|
alert_users
|
||||||
|
publish_changes
|
||||||
|
grant_badge
|
||||||
|
|
||||||
PostAlerter.new.after_save_post(@post)
|
@post_successfully_saved && @topic_successfully_saved
|
||||||
|
|
||||||
@post.publish_change_to_clients! :revised
|
|
||||||
BadgeGranter.queue_badge_grant(Badge::Trigger::PostRevision, post: @post)
|
|
||||||
|
|
||||||
true
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
def cleanup_whitespaces(raw)
|
||||||
|
TextCleaner.normalize_whitespaces(raw).gsub(/\s+\z/, "")
|
||||||
|
end
|
||||||
|
|
||||||
def should_revise?
|
def should_revise?
|
||||||
@post.raw != @new_raw || @opts[:changed_owner]
|
post_changed? || topic_changed?
|
||||||
|
end
|
||||||
|
|
||||||
|
def post_changed?
|
||||||
|
POST_TRACKED_FIELDS.each do |field|
|
||||||
|
return true if @fields.has_key?(field) && @fields[field] != @post.send(field)
|
||||||
|
end
|
||||||
|
false
|
||||||
|
end
|
||||||
|
|
||||||
|
def topic_changed?
|
||||||
|
TOPIC_TRACKED_FIELDS.each do |field|
|
||||||
|
return true if @fields.has_key?(field) && @fields[field] != @topic.send(field)
|
||||||
|
end
|
||||||
|
false
|
||||||
end
|
end
|
||||||
|
|
||||||
def revise_post
|
def revise_post
|
||||||
if should_create_new_version?
|
if should_create_new_version?
|
||||||
revise_and_create_new_version
|
revise_and_create_new_version
|
||||||
else
|
else
|
||||||
update_post
|
revise
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def plugin_callbacks
|
|
||||||
DiscourseEvent.trigger :before_edit_post, @post
|
|
||||||
DiscourseEvent.trigger :validate_post, @post
|
|
||||||
end
|
|
||||||
|
|
||||||
def get_revised_at
|
|
||||||
@opts[:revised_at] || Time.now
|
|
||||||
end
|
|
||||||
|
|
||||||
def should_create_new_version?
|
def should_create_new_version?
|
||||||
@post.last_editor_id != @editor.id ||
|
edited_by_another_user? || !ninja_edit? || owner_changed? || force_new_version?
|
||||||
get_revised_at - @post.last_version_at > SiteSetting.ninja_edit_window.to_i ||
|
end
|
||||||
@opts[:changed_owner] == true ||
|
|
||||||
|
def edited_by_another_user?
|
||||||
|
@post.last_editor_id != @editor.id
|
||||||
|
end
|
||||||
|
|
||||||
|
def ninja_edit?
|
||||||
|
@revised_at - @last_version_at <= SiteSetting.ninja_edit_window.to_i
|
||||||
|
end
|
||||||
|
|
||||||
|
def owner_changed?
|
||||||
|
@fields.has_key?(:user_id) && @fields[:user_id] != @post.user_id
|
||||||
|
end
|
||||||
|
|
||||||
|
def force_new_version?
|
||||||
@opts[:force_new_version] == true
|
@opts[:force_new_version] == true
|
||||||
end
|
end
|
||||||
|
|
||||||
def revise_and_create_new_version
|
def revise_and_create_new_version
|
||||||
|
@version_changed = true
|
||||||
@post.version += 1
|
@post.version += 1
|
||||||
@post.last_version_at = get_revised_at
|
@post.public_version += 1
|
||||||
|
@post.last_version_at = @revised_at
|
||||||
|
|
||||||
|
revise
|
||||||
|
perform_edit
|
||||||
|
bump_topic
|
||||||
|
end
|
||||||
|
|
||||||
|
def revise
|
||||||
update_post
|
update_post
|
||||||
EditRateLimiter.new(@editor).performed! unless @opts[:bypass_rate_limiter] == true
|
update_topic if topic_changed?
|
||||||
bump_topic unless @opts[:bypass_bump]
|
create_or_update_revision
|
||||||
end
|
|
||||||
|
|
||||||
def bump_topic
|
|
||||||
unless Post.where('post_number > ? and topic_id = ?', @post.post_number, @post.topic_id).exists?
|
|
||||||
@post.topic.update_column(:bumped_at, Time.now)
|
|
||||||
TopicTrackingState.publish_latest(@post.topic)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def update_topic_word_counts
|
|
||||||
Topic.exec_sql("UPDATE topics SET word_count = (SELECT SUM(COALESCE(posts.word_count, 0))
|
|
||||||
FROM posts WHERE posts.topic_id = :topic_id)
|
|
||||||
WHERE topics.id = :topic_id", topic_id: @post.topic_id)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_post
|
def update_post
|
||||||
@post.raw = @new_raw
|
POST_TRACKED_FIELDS.each do |field|
|
||||||
@post.word_count = @new_raw.scan(/\w+/).size
|
@post.send("#{field}=", @fields[field]) if @fields.has_key?(field)
|
||||||
@post.last_editor_id = @editor.id
|
end
|
||||||
@post.edit_reason = @opts[:edit_reason] if @opts[:edit_reason]
|
|
||||||
@post.user_id = @opts[:new_user].id if @opts[:new_user]
|
|
||||||
@post.self_edits += 1 if @editor == @post.user
|
|
||||||
|
|
||||||
if @editor == @post.user && @post.hidden && @post.hidden_reason_id == Post.hidden_reasons[:flag_threshold_reached]
|
@post.last_editor_id = @editor.id
|
||||||
|
@post.word_count = @fields[:raw].scan(/\w+/).size if @fields.has_key?(:raw)
|
||||||
|
@post.self_edits += 1 if self_edit?
|
||||||
|
|
||||||
|
clear_flags_and_unhide_post
|
||||||
|
|
||||||
|
@post.extract_quoted_post_numbers
|
||||||
|
@post_successfully_saved = @post.save(validate: !@opts[:skip_validations])
|
||||||
|
@post.save_reply_relationships
|
||||||
|
end
|
||||||
|
|
||||||
|
def self_edit?
|
||||||
|
@editor == @post.user
|
||||||
|
end
|
||||||
|
|
||||||
|
def clear_flags_and_unhide_post
|
||||||
|
return unless editing_a_flagged_and_hidden_post?
|
||||||
PostAction.clear_flags!(@post, Discourse.system_user)
|
PostAction.clear_flags!(@post, Discourse.system_user)
|
||||||
@post.unhide!
|
@post.unhide!
|
||||||
end
|
end
|
||||||
|
|
||||||
@post.extract_quoted_post_numbers
|
def editing_a_flagged_and_hidden_post?
|
||||||
@post.save(validate: !@opts[:skip_validations])
|
self_edit? &&
|
||||||
|
@post.hidden &&
|
||||||
|
@post.hidden_reason_id == Post.hidden_reasons[:flag_threshold_reached]
|
||||||
|
end
|
||||||
|
|
||||||
@post.save_reply_relationships
|
def update_topic
|
||||||
|
@topic.title = @fields[:title] if @fields.has_key?(:title)
|
||||||
|
Topic.transaction do
|
||||||
|
@topic_successfully_saved = @topic.change_category_to_id(@fields[:category_id]) if @fields.has_key?(:category_id)
|
||||||
|
@topic_successfully_saved &&= @topic.save(validate: !@opts[:skip_validations])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def create_or_update_revision
|
||||||
|
if @version_changed
|
||||||
|
create_revision
|
||||||
|
else
|
||||||
|
update_revision
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def create_revision
|
||||||
|
modifications = post_changes.merge(topic_changes)
|
||||||
|
PostRevision.create!(
|
||||||
|
user_id: @post.last_editor_id,
|
||||||
|
post_id: @post.id,
|
||||||
|
number: @post.version,
|
||||||
|
modifications: modifications
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
def update_revision
|
||||||
|
return unless revision = PostRevision.find_by(post_id: @post.id, number: @post.version)
|
||||||
|
revision.user_id = @post.last_editor_id
|
||||||
|
modifications = post_changes.merge(topic_changes)
|
||||||
|
modifications.keys.each do |field|
|
||||||
|
if revision.modifications.has_key?(field)
|
||||||
|
old_value = revision.modifications[field][0]
|
||||||
|
new_value = modifications[field][1]
|
||||||
|
revision.modifications[field] = [old_value, new_value]
|
||||||
|
else
|
||||||
|
revision.modifications[field] = modifications[field]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
revision.save
|
||||||
|
end
|
||||||
|
|
||||||
|
def post_changes
|
||||||
|
@post.previous_changes.slice(*POST_TRACKED_FIELDS)
|
||||||
|
end
|
||||||
|
|
||||||
|
def topic_changes
|
||||||
|
@topic.previous_changes.slice(*TOPIC_TRACKED_FIELDS)
|
||||||
|
end
|
||||||
|
|
||||||
|
def perform_edit
|
||||||
|
return if bypass_rate_limiter?
|
||||||
|
EditRateLimiter.new(@editor).performed!
|
||||||
|
end
|
||||||
|
|
||||||
|
def bypass_rate_limiter?
|
||||||
|
@opts[:bypass_rate_limiter] == true
|
||||||
|
end
|
||||||
|
|
||||||
|
def bump_topic
|
||||||
|
return if bypass_bump? || !is_last_post?
|
||||||
|
@topic.update_column(:bumped_at, Time.now)
|
||||||
|
TopicTrackingState.publish_latest(@topic)
|
||||||
|
end
|
||||||
|
|
||||||
|
def bypass_bump?
|
||||||
|
@opts[:bypass_bump] == true
|
||||||
|
end
|
||||||
|
|
||||||
|
def is_last_post?
|
||||||
|
!Post.where(topic_id: @topic.id)
|
||||||
|
.where("post_number > ?", @post.post_number)
|
||||||
|
.exists?
|
||||||
|
end
|
||||||
|
|
||||||
|
def plugin_callbacks
|
||||||
|
DiscourseEvent.trigger(:before_edit_post, @post)
|
||||||
|
DiscourseEvent.trigger(:validate_post, @post)
|
||||||
|
end
|
||||||
|
|
||||||
|
def revise_topic
|
||||||
|
return unless @post.post_number == 1
|
||||||
|
|
||||||
|
update_topic_excerpt
|
||||||
|
update_category_description
|
||||||
|
end
|
||||||
|
|
||||||
|
def update_topic_excerpt
|
||||||
|
excerpt = @post.excerpt(220, strip_links: true)
|
||||||
|
@topic.update_column(:excerpt, excerpt)
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_category_description
|
def update_category_description
|
||||||
# If we're revising the first post, we might have to update the category description
|
return unless category = Category.find_by(topic_id: @topic.id)
|
||||||
return unless @post.post_number == 1
|
|
||||||
|
|
||||||
# Is there a category with our topic id?
|
|
||||||
category = Category.find_by(topic_id: @post.topic_id)
|
|
||||||
return unless category.present?
|
|
||||||
|
|
||||||
# If found, update its description
|
|
||||||
body = @post.cooked
|
body = @post.cooked
|
||||||
matches = body.scan(/\<p\>(.*)\<\/p\>/)
|
matches = body.scan(/\<p\>(.*)\<\/p\>/)
|
||||||
if matches && matches[0] && matches[0][0]
|
if matches && matches[0] && matches[0][0]
|
||||||
|
@ -143,12 +274,35 @@ class PostRevisor
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_topic_excerpt
|
def advance_draft_sequence
|
||||||
@post.topic.update_column(:excerpt, @post.excerpt(220, strip_links: true)) if @post.post_number == 1
|
@post.advance_draft_sequence
|
||||||
end
|
end
|
||||||
|
|
||||||
def post_process_post
|
def post_process_post
|
||||||
@post.invalidate_oneboxes = true
|
@post.invalidate_oneboxes = true
|
||||||
@post.trigger_post_process
|
@post.trigger_post_process
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def update_topic_word_counts
|
||||||
|
Topic.exec_sql("UPDATE topics
|
||||||
|
SET word_count = (
|
||||||
|
SELECT SUM(COALESCE(posts.word_count, 0))
|
||||||
|
FROM posts
|
||||||
|
WHERE posts.topic_id = :topic_id
|
||||||
|
)
|
||||||
|
WHERE topics.id = :topic_id", topic_id: @topic.id)
|
||||||
|
end
|
||||||
|
|
||||||
|
def alert_users
|
||||||
|
PostAlerter.new.after_save_post(@post)
|
||||||
|
end
|
||||||
|
|
||||||
|
def publish_changes
|
||||||
|
@post.publish_change_to_clients!(:revised)
|
||||||
|
end
|
||||||
|
|
||||||
|
def grant_badge
|
||||||
|
BadgeGranter.queue_badge_grant(Badge::Trigger::PostRevision, post: @post)
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -48,7 +48,7 @@ task 'posts:normalize_code' => :environment do
|
||||||
Post.where("raw like '%<pre>%<code>%'").each do |p|
|
Post.where("raw like '%<pre>%<code>%'").each do |p|
|
||||||
normalized = Import::Normalize.normalize_code_blocks(p.raw, lang)
|
normalized = Import::Normalize.normalize_code_blocks(p.raw, lang)
|
||||||
if normalized != p.raw
|
if normalized != p.raw
|
||||||
p.revise(Discourse.system_user, normalized)
|
p.revise(Discourse.system_user, { raw: normalized })
|
||||||
putc "."
|
putc "."
|
||||||
i += 1
|
i += 1
|
||||||
end
|
end
|
||||||
|
|
|
@ -413,7 +413,7 @@ class ImportScripts::PhpBB3 < ImportScripts::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
if new_raw != post.raw
|
if new_raw != post.raw
|
||||||
PostRevisor.new(post).revise!(post.user, new_raw, {bypass_bump: true, edit_reason: 'Migrate from PHPBB3'})
|
PostRevisor.new(post).revise!(post.user, { raw: new_raw }, { bypass_bump: true, edit_reason: 'Migrate from PHPBB3' })
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -86,7 +86,7 @@ describe CategoryList do
|
||||||
end
|
end
|
||||||
|
|
||||||
context "with a topic in a category" do
|
context "with a topic in a category" do
|
||||||
let!(:topic) { Fabricate(:topic, category: topic_category)}
|
let!(:topic) { Fabricate(:topic, category: topic_category) }
|
||||||
let(:category) { category_list.categories.first }
|
let(:category) { category_list.categories.first }
|
||||||
|
|
||||||
it "should return the category" do
|
it "should return the category" do
|
||||||
|
|
|
@ -16,7 +16,7 @@ describe PostRevisor do
|
||||||
describe 'with the same body' do
|
describe 'with the same body' do
|
||||||
it "doesn't change version" do
|
it "doesn't change version" do
|
||||||
lambda {
|
lambda {
|
||||||
subject.revise!(post.user, post.raw).should == false
|
subject.revise!(post.user, { raw: post.raw }).should == false
|
||||||
post.reload
|
post.reload
|
||||||
}.should_not change(post, :version)
|
}.should_not change(post, :version)
|
||||||
end
|
end
|
||||||
|
@ -25,10 +25,11 @@ describe PostRevisor do
|
||||||
describe 'ninja editing' do
|
describe 'ninja editing' do
|
||||||
it 'correctly applies edits' do
|
it 'correctly applies edits' do
|
||||||
SiteSetting.ninja_edit_window = 1.minute.to_i
|
SiteSetting.ninja_edit_window = 1.minute.to_i
|
||||||
subject.revise!(post.user, 'updated body', revised_at: post.updated_at + 10.seconds)
|
subject.revise!(post.user, { raw: 'updated body' }, revised_at: post.updated_at + 10.seconds)
|
||||||
post.reload
|
post.reload
|
||||||
|
|
||||||
post.version.should == 1
|
post.version.should == 1
|
||||||
|
post.public_version.should == 1
|
||||||
post.revisions.size.should == 0
|
post.revisions.size.should == 0
|
||||||
post.last_version_at.should == first_version_at
|
post.last_version_at.should == first_version_at
|
||||||
subject.category_changed.should be_blank
|
subject.category_changed.should be_blank
|
||||||
|
@ -41,7 +42,7 @@ describe PostRevisor do
|
||||||
|
|
||||||
before do
|
before do
|
||||||
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
||||||
subject.revise!(post.user, 'updated body', revised_at: revised_at)
|
subject.revise!(post.user, { raw: 'updated body' }, revised_at: revised_at)
|
||||||
post.reload
|
post.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -49,11 +50,12 @@ describe PostRevisor do
|
||||||
subject.category_changed.should be_blank
|
subject.category_changed.should be_blank
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'updates the version' do
|
it 'updates the versions' do
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'creates a new version' do
|
it 'creates a new revision' do
|
||||||
post.revisions.size.should == 1
|
post.revisions.size.should == 1
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -64,12 +66,13 @@ describe PostRevisor do
|
||||||
describe "new edit window" do
|
describe "new edit window" do
|
||||||
|
|
||||||
before do
|
before do
|
||||||
subject.revise!(post.user, 'yet another updated body', revised_at: revised_at)
|
subject.revise!(post.user, { raw: 'yet another updated body' }, revised_at: revised_at)
|
||||||
post.reload
|
post.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
it "doesn't create a new version if you do another" do
|
it "doesn't create a new version if you do another" do
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
end
|
end
|
||||||
|
|
||||||
it "doesn't change last_version_at" do
|
it "doesn't change last_version_at" do
|
||||||
|
@ -85,12 +88,13 @@ describe PostRevisor do
|
||||||
let!(:new_revised_at) {revised_at + 2.minutes}
|
let!(:new_revised_at) {revised_at + 2.minutes}
|
||||||
|
|
||||||
before do
|
before do
|
||||||
subject.revise!(post.user, 'yet another, another updated body', revised_at: new_revised_at)
|
subject.revise!(post.user, { raw: 'yet another, another updated body' }, revised_at: new_revised_at)
|
||||||
post.reload
|
post.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
it "does create a new version after the edit window" do
|
it "does create a new version after the edit window" do
|
||||||
post.version.should == 3
|
post.version.should == 3
|
||||||
|
post.public_version.should == 3
|
||||||
end
|
end
|
||||||
|
|
||||||
it "does create a new version after the edit window" do
|
it "does create a new version after the edit window" do
|
||||||
|
@ -116,7 +120,7 @@ describe PostRevisor do
|
||||||
|
|
||||||
context "one paragraph description" do
|
context "one paragraph description" do
|
||||||
before do
|
before do
|
||||||
subject.revise!(post.user, new_description)
|
subject.revise!(post.user, { raw: new_description })
|
||||||
category.reload
|
category.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -131,7 +135,7 @@ describe PostRevisor do
|
||||||
|
|
||||||
context "multiple paragraph description" do
|
context "multiple paragraph description" do
|
||||||
before do
|
before do
|
||||||
subject.revise!(post.user, "#{new_description}\n\nOther content goes here.")
|
subject.revise!(post.user, { raw: "#{new_description}\n\nOther content goes here." })
|
||||||
category.reload
|
category.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -147,7 +151,7 @@ describe PostRevisor do
|
||||||
context 'when updating back to the original paragraph' do
|
context 'when updating back to the original paragraph' do
|
||||||
before do
|
before do
|
||||||
category.update_column(:description, 'this is my description')
|
category.update_column(:description, 'this is my description')
|
||||||
subject.revise!(post.user, Category.post_template)
|
subject.revise!(post.user, { raw: Category.post_template })
|
||||||
category.reload
|
category.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -167,7 +171,7 @@ describe PostRevisor do
|
||||||
|
|
||||||
it "triggers a rate limiter" do
|
it "triggers a rate limiter" do
|
||||||
EditRateLimiter.any_instance.expects(:performed!)
|
EditRateLimiter.any_instance.expects(:performed!)
|
||||||
subject.revise!(changed_by, 'updated body')
|
subject.revise!(changed_by, { raw: 'updated body' })
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -178,7 +182,7 @@ describe PostRevisor do
|
||||||
SiteSetting.stubs(:newuser_max_images).returns(0)
|
SiteSetting.stubs(:newuser_max_images).returns(0)
|
||||||
url = "http://i.imgur.com/wfn7rgU.jpg"
|
url = "http://i.imgur.com/wfn7rgU.jpg"
|
||||||
Oneboxer.stubs(:onebox).with(url, anything).returns("<img src='#{url}'>")
|
Oneboxer.stubs(:onebox).with(url, anything).returns("<img src='#{url}'>")
|
||||||
subject.revise!(changed_by, "So, post them here!\n#{url}")
|
subject.revise!(changed_by, { raw: "So, post them here!\n#{url}" })
|
||||||
end
|
end
|
||||||
|
|
||||||
it "allows an admin to insert images into a new user's post" do
|
it "allows an admin to insert images into a new user's post" do
|
||||||
|
@ -196,7 +200,7 @@ describe PostRevisor do
|
||||||
SiteSetting.stubs(:newuser_max_images).returns(0)
|
SiteSetting.stubs(:newuser_max_images).returns(0)
|
||||||
url = "http://i.imgur.com/FGg7Vzu.gif"
|
url = "http://i.imgur.com/FGg7Vzu.gif"
|
||||||
Oneboxer.stubs(:cached_onebox).with(url, anything).returns("<img src='#{url}'>")
|
Oneboxer.stubs(:cached_onebox).with(url, anything).returns("<img src='#{url}'>")
|
||||||
subject.revise!(post.user, "So, post them here!\n#{url}")
|
subject.revise!(post.user, { raw: "So, post them here!\n#{url}" })
|
||||||
end
|
end
|
||||||
|
|
||||||
it "doesn't allow images to be inserted" do
|
it "doesn't allow images to be inserted" do
|
||||||
|
@ -208,7 +212,7 @@ describe PostRevisor do
|
||||||
|
|
||||||
describe 'with a new body' do
|
describe 'with a new body' do
|
||||||
let(:changed_by) { Fabricate(:coding_horror) }
|
let(:changed_by) { Fabricate(:coding_horror) }
|
||||||
let!(:result) { subject.revise!(changed_by, "lets update the body") }
|
let!(:result) { subject.revise!(changed_by, { raw: "lets update the body" }) }
|
||||||
|
|
||||||
it 'returns true' do
|
it 'returns true' do
|
||||||
result.should == true
|
result.should == true
|
||||||
|
@ -222,8 +226,9 @@ describe PostRevisor do
|
||||||
post.invalidate_oneboxes.should == true
|
post.invalidate_oneboxes.should == true
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'increased the version' do
|
it 'increased the versions' do
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'has the new revision' do
|
it 'has the new revision' do
|
||||||
|
@ -242,16 +247,14 @@ describe PostRevisor do
|
||||||
|
|
||||||
context 'second poster posts again quickly' do
|
context 'second poster posts again quickly' do
|
||||||
before do
|
before do
|
||||||
SiteSetting.expects(:ninja_edit_window).returns(1.minute.to_i)
|
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
||||||
subject.revise!(changed_by, 'yet another updated body', revised_at: post.updated_at + 10.seconds)
|
subject.revise!(changed_by, { raw: 'yet another updated body' }, revised_at: post.updated_at + 10.seconds)
|
||||||
post.reload
|
post.reload
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'is a ninja edit, because the second poster posted again quickly' do
|
it 'is a ninja edit, because the second poster posted again quickly' do
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
end
|
post.public_version.should == 2
|
||||||
|
|
||||||
it 'is a ninja edit, because the second poster posted again quickly' do
|
|
||||||
post.revisions.size.should == 1
|
post.revisions.size.should == 1
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -262,19 +265,19 @@ describe PostRevisor do
|
||||||
revisor = described_class.new(post)
|
revisor = described_class.new(post)
|
||||||
first_post = topic.posts.by_post_number.first
|
first_post = topic.posts.by_post_number.first
|
||||||
expect {
|
expect {
|
||||||
revisor.revise!(first_post.user, 'Edit the first post', revised_at: first_post.updated_at + 10.seconds)
|
revisor.revise!(first_post.user, { raw: 'Edit the first post' }, revised_at: first_post.updated_at + 10.seconds)
|
||||||
topic.reload
|
topic.reload
|
||||||
}.to change { topic.excerpt }
|
}.to change { topic.excerpt }
|
||||||
second_post = Fabricate(:post, post_args.merge(post_number: 2, topic_id: topic.id))
|
second_post = Fabricate(:post, post_args.merge(post_number: 2, topic_id: topic.id))
|
||||||
expect {
|
expect {
|
||||||
described_class.new(second_post).revise!(second_post.user, 'Edit the 2nd post')
|
described_class.new(second_post).revise!(second_post.user, { raw: 'Edit the 2nd post' })
|
||||||
topic.reload
|
topic.reload
|
||||||
}.to_not change { topic.excerpt }
|
}.to_not change { topic.excerpt }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
it "doesn't strip starting whitespaces" do
|
it "doesn't strip starting whitespaces" do
|
||||||
subject.revise!(post.user, " <-- whitespaces --> ")
|
subject.revise!(post.user, { raw: " <-- whitespaces --> " })
|
||||||
post.reload
|
post.reload
|
||||||
post.raw.should == " <-- whitespaces -->"
|
post.raw.should == " <-- whitespaces -->"
|
||||||
end
|
end
|
||||||
|
|
|
@ -307,7 +307,7 @@ describe PostsController do
|
||||||
end
|
end
|
||||||
|
|
||||||
it "calls revise with valid parameters" do
|
it "calls revise with valid parameters" do
|
||||||
PostRevisor.any_instance.expects(:revise!).with(post.user, 'edited body', edit_reason: 'typo')
|
PostRevisor.any_instance.expects(:revise!).with(post.user, { raw: 'edited body' , edit_reason: 'typo' })
|
||||||
xhr :put, :update, update_params
|
xhr :put, :update, update_params
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -605,7 +605,8 @@ describe PostsController do
|
||||||
|
|
||||||
describe "revisions" do
|
describe "revisions" do
|
||||||
|
|
||||||
let(:post_revision) { Fabricate(:post_revision) }
|
let(:post) { Fabricate(:post, version: 2) }
|
||||||
|
let(:post_revision) { Fabricate(:post_revision, post: post) }
|
||||||
|
|
||||||
it "throws an exception when revision is < 2" do
|
it "throws an exception when revision is < 2" do
|
||||||
expect {
|
expect {
|
||||||
|
@ -636,7 +637,7 @@ describe PostsController do
|
||||||
|
|
||||||
it "ensures poster can see the revisions" do
|
it "ensures poster can see the revisions" do
|
||||||
user = log_in(:active_user)
|
user = log_in(:active_user)
|
||||||
post = Fabricate(:post, user: user)
|
post = Fabricate(:post, user: user, version: 3)
|
||||||
pr = Fabricate(:post_revision, user: user, post: post)
|
pr = Fabricate(:post_revision, user: user, post: post)
|
||||||
xhr :get, :revisions, post_id: pr.post_id, revision: pr.number
|
xhr :get, :revisions, post_id: pr.post_id, revision: pr.number
|
||||||
response.should be_success
|
response.should be_success
|
||||||
|
@ -663,7 +664,7 @@ describe PostsController do
|
||||||
|
|
||||||
context "deleted post" do
|
context "deleted post" do
|
||||||
let(:admin) { log_in(:admin) }
|
let(:admin) { log_in(:admin) }
|
||||||
let(:deleted_post) { Fabricate(:post, user: admin) }
|
let(:deleted_post) { Fabricate(:post, user: admin, version: 3) }
|
||||||
let(:deleted_post_revision) { Fabricate(:post_revision, user: admin, post: deleted_post) }
|
let(:deleted_post_revision) { Fabricate(:post_revision, user: admin, post: deleted_post) }
|
||||||
|
|
||||||
before { deleted_post.trash!(admin) }
|
before { deleted_post.trash!(admin) }
|
||||||
|
@ -677,7 +678,7 @@ describe PostsController do
|
||||||
context "deleted topic" do
|
context "deleted topic" do
|
||||||
let(:admin) { log_in(:admin) }
|
let(:admin) { log_in(:admin) }
|
||||||
let(:deleted_topic) { Fabricate(:topic, user: admin) }
|
let(:deleted_topic) { Fabricate(:topic, user: admin) }
|
||||||
let(:post) { Fabricate(:post, user: admin, topic: deleted_topic) }
|
let(:post) { Fabricate(:post, user: admin, topic: deleted_topic, version: 3) }
|
||||||
let(:post_revision) { Fabricate(:post_revision, user: admin, post: post) }
|
let(:post_revision) { Fabricate(:post_revision, user: admin, post: post) }
|
||||||
|
|
||||||
before { deleted_topic.trash!(admin) }
|
before { deleted_topic.trash!(admin) }
|
||||||
|
|
|
@ -733,6 +733,7 @@ describe TopicsController do
|
||||||
describe 'when logged in' do
|
describe 'when logged in' do
|
||||||
before do
|
before do
|
||||||
@topic = Fabricate(:topic, user: log_in)
|
@topic = Fabricate(:topic, user: log_in)
|
||||||
|
Fabricate(:post, topic: @topic)
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'without permission' do
|
describe 'without permission' do
|
||||||
|
@ -778,7 +779,7 @@ describe TopicsController do
|
||||||
|
|
||||||
it "returns errors with invalid categories" do
|
it "returns errors with invalid categories" do
|
||||||
Topic.any_instance.expects(:change_category_to_id).returns(false)
|
Topic.any_instance.expects(:change_category_to_id).returns(false)
|
||||||
xhr :put, :update, topic_id: @topic.id, slug: @topic.title
|
xhr :put, :update, topic_id: @topic.id, slug: @topic.title, category_id: -1
|
||||||
expect(response).not_to be_success
|
expect(response).not_to be_success
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
Fabricator(:post_revision) do
|
Fabricator(:post_revision) do
|
||||||
post
|
post
|
||||||
user
|
user
|
||||||
number 3
|
number 2
|
||||||
modifications do
|
modifications do
|
||||||
{ "cooked" => ["<p>BEFORE</p>", "<p>AFTER</p>"], "raw" => ["BEFORE", "AFTER"] }
|
{ "cooked" => ["<p>BEFORE</p>", "<p>AFTER</p>"], "raw" => ["BEFORE", "AFTER"] }
|
||||||
end
|
end
|
||||||
|
|
|
@ -368,7 +368,7 @@ describe Category do
|
||||||
post = create_post(user: @category.user, category: @category.name)
|
post = create_post(user: @category.user, category: @category.name)
|
||||||
|
|
||||||
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
||||||
post.revise(post.user, 'updated body', revised_at: post.updated_at + 2.minutes)
|
post.revise(post.user, { raw: 'updated body' }, revised_at: post.updated_at + 2.minutes)
|
||||||
|
|
||||||
Category.update_stats
|
Category.update_stats
|
||||||
@category.reload
|
@category.reload
|
||||||
|
|
|
@ -73,7 +73,7 @@ describe Draft do
|
||||||
it 'nukes the post draft when a post is revised' do
|
it 'nukes the post draft when a post is revised' do
|
||||||
p = Fabricate(:post)
|
p = Fabricate(:post)
|
||||||
Draft.set(p.user, p.topic.draft_key, 0,'hello')
|
Draft.set(p.user, p.topic.draft_key, 0,'hello')
|
||||||
p.revise(p.user, 'another test')
|
p.revise(p.user, { raw: 'another test' })
|
||||||
s = DraftSequence.current(p.user, p.topic.draft_key)
|
s = DraftSequence.current(p.user, p.topic.draft_key)
|
||||||
Draft.get(p.user, p.topic.draft_key, s).should == nil
|
Draft.get(p.user, p.topic.draft_key, s).should == nil
|
||||||
end
|
end
|
||||||
|
|
|
@ -297,7 +297,7 @@ describe PostAction do
|
||||||
post.hidden_reason_id.should == Post.hidden_reasons[:flag_threshold_reached]
|
post.hidden_reason_id.should == Post.hidden_reasons[:flag_threshold_reached]
|
||||||
post.topic.visible.should == false
|
post.topic.visible.should == false
|
||||||
|
|
||||||
post.revise(post.user, post.raw + " ha I edited it ")
|
post.revise(post.user, { raw: post.raw + " ha I edited it " })
|
||||||
|
|
||||||
post.reload
|
post.reload
|
||||||
|
|
||||||
|
@ -316,7 +316,7 @@ describe PostAction do
|
||||||
post.hidden_reason_id.should == Post.hidden_reasons[:flag_threshold_reached_again]
|
post.hidden_reason_id.should == Post.hidden_reasons[:flag_threshold_reached_again]
|
||||||
post.topic.visible.should == false
|
post.topic.visible.should == false
|
||||||
|
|
||||||
post.revise(post.user, post.raw + " ha I edited it again ")
|
post.revise(post.user, { raw: post.raw + " ha I edited it again " })
|
||||||
|
|
||||||
post.reload
|
post.reload
|
||||||
|
|
||||||
|
|
|
@ -36,7 +36,7 @@ describe PostAlertObserver do
|
||||||
context 'when editing a post' do
|
context 'when editing a post' do
|
||||||
it 'notifies a user of the revision' do
|
it 'notifies a user of the revision' do
|
||||||
lambda {
|
lambda {
|
||||||
post.revise(evil_trout, "world is the new body of the message")
|
post.revise(evil_trout, { raw: "world is the new body of the message" })
|
||||||
}.should change(post.user.notifications, :count).by(1)
|
}.should change(post.user.notifications, :count).by(1)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -47,13 +47,13 @@ describe PostAlertObserver do
|
||||||
|
|
||||||
it 'notifies a user of the revision made by another user' do
|
it 'notifies a user of the revision made by another user' do
|
||||||
lambda {
|
lambda {
|
||||||
post.revise(evil_trout, "world is the new body of the message")
|
post.revise(evil_trout, { raw: "world is the new body of the message" })
|
||||||
}.should change(post.user.notifications, :count).by(1)
|
}.should change(post.user.notifications, :count).by(1)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'does not notifiy a user of the revision made by the system user' do
|
it 'does not notifiy a user of the revision made by the system user' do
|
||||||
lambda {
|
lambda {
|
||||||
post.revise(Discourse.system_user, "world is the new body of the message")
|
post.revise(Discourse.system_user, { raw: "world is the new body of the message" })
|
||||||
}.should_not change(post.user.notifications, :count)
|
}.should_not change(post.user.notifications, :count)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,119 +0,0 @@
|
||||||
require 'spec_helper'
|
|
||||||
require_dependency 'post_revision'
|
|
||||||
|
|
||||||
describe PostRevision do
|
|
||||||
|
|
||||||
before do
|
|
||||||
@number = 1
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_rev(modifications, post_id=1)
|
|
||||||
@number += 1
|
|
||||||
PostRevision.create!(post_id: post_id, user_id: 1, number: @number, modifications: modifications)
|
|
||||||
end
|
|
||||||
|
|
||||||
it "ignores deprecated current values in history" do
|
|
||||||
p = PostRevision.new(modifications: {"foo" => ["bar", "bar1"]})
|
|
||||||
p.previous("foo").should == "bar"
|
|
||||||
p.current("foo").should == "bar"
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can fallback to previous revisions if needed" do
|
|
||||||
r1 = create_rev("foo" => ["A", "B"])
|
|
||||||
r2 = create_rev("foo" => ["C", "D"])
|
|
||||||
|
|
||||||
r1.current("foo").should == "C"
|
|
||||||
r2.current("foo").should == "C"
|
|
||||||
r2.previous("foo").should == "C"
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can fallback to post if needed" do
|
|
||||||
post = Fabricate(:post)
|
|
||||||
r = create_rev({"foo" => ["A", "B"]}, post.id)
|
|
||||||
|
|
||||||
r.current("raw").should == post.raw
|
|
||||||
r.previous("raw").should == post.raw
|
|
||||||
r.current("cooked").should == post.cooked
|
|
||||||
r.previous("cooked").should == post.cooked
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can fallback to post for current rev only if needed" do
|
|
||||||
post = Fabricate(:post)
|
|
||||||
r = create_rev({"raw" => ["A"], "cooked" => ["AA"]}, post.id)
|
|
||||||
|
|
||||||
r.current("raw").should == post.raw
|
|
||||||
r.previous("raw").should == "A"
|
|
||||||
r.current("cooked").should == post.cooked
|
|
||||||
r.previous("cooked").should == "AA"
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can fallback to topic if needed" do
|
|
||||||
post = Fabricate(:post)
|
|
||||||
r = create_rev({"foo" => ["A", "B"]}, post.id)
|
|
||||||
|
|
||||||
r.current("title").should == post.topic.title
|
|
||||||
r.previous("title").should == post.topic.title
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can find title changes" do
|
|
||||||
r1 = create_rev({"title" => ["hello"]})
|
|
||||||
r2 = create_rev({"title" => ["frog"]})
|
|
||||||
r1.title_changes[:inline].should =~ /frog.*hello/
|
|
||||||
r1.title_changes[:side_by_side].should =~ /hello.*frog/
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can find category changes" do
|
|
||||||
cat1 = Fabricate(:category, name: "cat1")
|
|
||||||
cat2 = Fabricate(:category, name: "cat2")
|
|
||||||
|
|
||||||
r1 = create_rev({"category_id" => [cat1.id, cat2.id]})
|
|
||||||
r2 = create_rev({"category_id" => [cat2.id, cat1.id]})
|
|
||||||
|
|
||||||
changes = r1.category_changes
|
|
||||||
changes[:previous_category_id].should == cat1.id
|
|
||||||
changes[:current_category_id].should == cat2.id
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can find wiki changes" do
|
|
||||||
r1 = create_rev("wiki" => [false])
|
|
||||||
r2 = create_rev("wiki" => [true])
|
|
||||||
|
|
||||||
changes = r1.wiki_changes
|
|
||||||
changes[:previous_wiki].should == false
|
|
||||||
changes[:current_wiki].should == true
|
|
||||||
end
|
|
||||||
|
|
||||||
it "can find post_type changes" do
|
|
||||||
r1 = create_rev("post_type" => [1])
|
|
||||||
r2 = create_rev("post_type" => [2])
|
|
||||||
|
|
||||||
changes = r1.post_type_changes
|
|
||||||
changes[:previous_post_type].should == 1
|
|
||||||
changes[:current_post_type].should == 2
|
|
||||||
end
|
|
||||||
|
|
||||||
it "hides revisions that were hidden" do
|
|
||||||
r1 = create_rev({"raw" => ["one"]})
|
|
||||||
r2 = create_rev({"raw" => ["two"]})
|
|
||||||
r3 = create_rev({"raw" => ["three"]})
|
|
||||||
|
|
||||||
r2.hide!
|
|
||||||
|
|
||||||
r1.current("raw").should == "three"
|
|
||||||
r2.previous("raw").should == "one"
|
|
||||||
end
|
|
||||||
|
|
||||||
it "shows revisions that were shown" do
|
|
||||||
r1 = create_rev({"raw" => ["one"]})
|
|
||||||
r2 = create_rev({"raw" => ["two"]})
|
|
||||||
r3 = create_rev({"raw" => ["three"]})
|
|
||||||
|
|
||||||
r2.hide!
|
|
||||||
r2.show!
|
|
||||||
|
|
||||||
r2.previous("raw").should == "two"
|
|
||||||
r1.current("raw").should == "two"
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
|
@ -161,7 +161,7 @@ describe Post do
|
||||||
post_no_images.user.trust_level = TrustLevel[0]
|
post_no_images.user.trust_level = TrustLevel[0]
|
||||||
post_no_images.save
|
post_no_images.save
|
||||||
-> {
|
-> {
|
||||||
post_no_images.revise(post_no_images.user, post_two_images.raw)
|
post_no_images.revise(post_no_images.user, { raw: post_two_images.raw })
|
||||||
post_no_images.reload
|
post_no_images.reload
|
||||||
}.should_not change(post_no_images, :raw)
|
}.should_not change(post_no_images, :raw)
|
||||||
end
|
end
|
||||||
|
@ -209,7 +209,7 @@ describe Post do
|
||||||
post_no_attachments.user.trust_level = TrustLevel[0]
|
post_no_attachments.user.trust_level = TrustLevel[0]
|
||||||
post_no_attachments.save
|
post_no_attachments.save
|
||||||
-> {
|
-> {
|
||||||
post_no_attachments.revise(post_no_attachments.user, post_two_attachments.raw)
|
post_no_attachments.revise(post_no_attachments.user, { raw: post_two_attachments.raw })
|
||||||
post_no_attachments.reload
|
post_no_attachments.reload
|
||||||
}.should_not change(post_no_attachments, :raw)
|
}.should_not change(post_no_attachments, :raw)
|
||||||
end
|
end
|
||||||
|
@ -468,124 +468,99 @@ describe Post do
|
||||||
it 'has no revision' do
|
it 'has no revision' do
|
||||||
post.revisions.size.should == 0
|
post.revisions.size.should == 0
|
||||||
first_version_at.should be_present
|
first_version_at.should be_present
|
||||||
post.revise(post.user, post.raw).should == false
|
post.revise(post.user, { raw: post.raw }).should == false
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'with the same body' do
|
describe 'with the same body' do
|
||||||
|
|
||||||
it "doesn't change version" do
|
it "doesn't change version" do
|
||||||
lambda { post.revise(post.user, post.raw); post.reload }.should_not change(post, :version)
|
lambda { post.revise(post.user, { raw: post.raw }); post.reload }.should_not change(post, :version)
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'ninja editing' do
|
describe 'ninja editing & edit windows' do
|
||||||
before do
|
|
||||||
SiteSetting.expects(:ninja_edit_window).returns(1.minute.to_i)
|
before { SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i) }
|
||||||
post.revise(post.user, 'updated body', revised_at: post.updated_at + 10.seconds)
|
|
||||||
|
it 'works' do
|
||||||
|
revised_at = post.updated_at + 2.minutes
|
||||||
|
new_revised_at = revised_at + 2.minutes
|
||||||
|
|
||||||
|
# ninja edit
|
||||||
|
post.revise(post.user, { raw: 'updated body' }, revised_at: post.updated_at + 10.seconds)
|
||||||
post.reload
|
post.reload
|
||||||
end
|
|
||||||
|
|
||||||
it 'causes no update' do
|
|
||||||
post.version.should == 1
|
post.version.should == 1
|
||||||
|
post.public_version.should == 1
|
||||||
post.revisions.size.should == 0
|
post.revisions.size.should == 0
|
||||||
post.last_version_at.should == first_version_at
|
post.last_version_at.to_i.should == first_version_at.to_i
|
||||||
end
|
|
||||||
|
|
||||||
end
|
# revision much later
|
||||||
|
post.revise(post.user, { raw: 'another updated body' }, revised_at: revised_at)
|
||||||
describe 'revision much later' do
|
|
||||||
|
|
||||||
let!(:revised_at) { post.updated_at + 2.minutes }
|
|
||||||
|
|
||||||
before do
|
|
||||||
SiteSetting.stubs(:ninja_edit_window).returns(1.minute.to_i)
|
|
||||||
post.revise(post.user, 'updated body', revised_at: revised_at)
|
|
||||||
post.reload
|
post.reload
|
||||||
end
|
|
||||||
|
|
||||||
it 'updates the version' do
|
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
post.revisions.size.should == 1
|
post.revisions.size.should == 1
|
||||||
post.last_version_at.to_i.should == revised_at.to_i
|
post.last_version_at.to_i.should == revised_at.to_i
|
||||||
end
|
|
||||||
|
|
||||||
describe "new edit window" do
|
# new edit window
|
||||||
|
post.revise(post.user, { raw: 'yet another updated body' }, revised_at: revised_at + 10.seconds)
|
||||||
before do
|
|
||||||
post.revise(post.user, 'yet another updated body', revised_at: revised_at)
|
|
||||||
post.reload
|
post.reload
|
||||||
end
|
|
||||||
|
|
||||||
it "doesn't create a new version if you do another" do
|
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
end
|
post.public_version.should == 2
|
||||||
|
post.revisions.size.should == 1
|
||||||
it "doesn't change last_version_at" do
|
|
||||||
post.last_version_at.to_i.should == revised_at.to_i
|
post.last_version_at.to_i.should == revised_at.to_i
|
||||||
end
|
|
||||||
|
|
||||||
context "after second window" do
|
# after second window
|
||||||
|
post.revise(post.user, { raw: 'yet another, another updated body' }, revised_at: new_revised_at)
|
||||||
let!(:new_revised_at) {revised_at + 2.minutes}
|
|
||||||
|
|
||||||
before do
|
|
||||||
post.revise(post.user, 'yet another, another updated body', revised_at: new_revised_at)
|
|
||||||
post.reload
|
post.reload
|
||||||
end
|
|
||||||
|
|
||||||
it "does create a new version after the edit window" do
|
|
||||||
post.version.should == 3
|
post.version.should == 3
|
||||||
end
|
post.public_version.should == 3
|
||||||
|
post.revisions.size.should == 2
|
||||||
it "does create a new version after the edit window" do
|
|
||||||
post.last_version_at.to_i.should == new_revised_at.to_i
|
post.last_version_at.to_i.should == new_revised_at.to_i
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'rate limiter' do
|
describe 'rate limiter' do
|
||||||
let(:changed_by) { Fabricate(:coding_horror) }
|
let(:changed_by) { Fabricate(:coding_horror) }
|
||||||
|
|
||||||
it "triggers a rate limiter" do
|
it "triggers a rate limiter" do
|
||||||
EditRateLimiter.any_instance.expects(:performed!)
|
EditRateLimiter.any_instance.expects(:performed!)
|
||||||
post.revise(changed_by, 'updated body')
|
post.revise(changed_by, { raw: 'updated body' })
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'with a new body' do
|
describe 'with a new body' do
|
||||||
let(:changed_by) { Fabricate(:coding_horror) }
|
let(:changed_by) { Fabricate(:coding_horror) }
|
||||||
let!(:result) { post.revise(changed_by, 'updated body') }
|
let!(:result) { post.revise(changed_by, { raw: 'updated body' }) }
|
||||||
|
|
||||||
it 'acts correctly' do
|
it 'acts correctly' do
|
||||||
result.should == true
|
result.should == true
|
||||||
post.raw.should == 'updated body'
|
post.raw.should == 'updated body'
|
||||||
post.invalidate_oneboxes.should == true
|
post.invalidate_oneboxes.should == true
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
post.revisions.size.should == 1
|
post.revisions.size.should == 1
|
||||||
post.revisions.first.user.should be_present
|
post.revisions.first.user.should be_present
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'second poster posts again quickly' do
|
context 'second poster posts again quickly' do
|
||||||
before do
|
|
||||||
SiteSetting.expects(:ninja_edit_window).returns(1.minute.to_i)
|
|
||||||
post.revise(changed_by, 'yet another updated body', revised_at: post.updated_at + 10.seconds)
|
|
||||||
post.reload
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'is a ninja edit, because the second poster posted again quickly' do
|
it 'is a ninja edit, because the second poster posted again quickly' do
|
||||||
|
SiteSetting.expects(:ninja_edit_window).returns(1.minute.to_i)
|
||||||
|
post.revise(changed_by, { raw: 'yet another updated body' }, revised_at: post.updated_at + 10.seconds)
|
||||||
|
post.reload
|
||||||
|
|
||||||
post.version.should == 2
|
post.version.should == 2
|
||||||
|
post.public_version.should == 2
|
||||||
post.revisions.size.should == 1
|
post.revisions.size.should == 1
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -118,7 +118,7 @@ http://b.com/#{'a'*500}
|
||||||
context 'removing a link' do
|
context 'removing a link' do
|
||||||
|
|
||||||
before do
|
before do
|
||||||
post.revise(post.user, "no more linkies")
|
post.revise(post.user, { raw: "no more linkies" })
|
||||||
TopicLink.extract_from(post)
|
TopicLink.extract_from(post)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -352,21 +352,21 @@ describe Topic do
|
||||||
it "doesn't bump the topic on an edit to the last post that doesn't result in a new version" do
|
it "doesn't bump the topic on an edit to the last post that doesn't result in a new version" do
|
||||||
lambda {
|
lambda {
|
||||||
SiteSetting.expects(:ninja_edit_window).returns(5.minutes)
|
SiteSetting.expects(:ninja_edit_window).returns(5.minutes)
|
||||||
@last_post.revise(@last_post.user, 'updated contents', revised_at: @last_post.created_at + 10.seconds)
|
@last_post.revise(@last_post.user, { raw: 'updated contents' }, revised_at: @last_post.created_at + 10.seconds)
|
||||||
@topic.reload
|
@topic.reload
|
||||||
}.should_not change(@topic, :bumped_at)
|
}.should_not change(@topic, :bumped_at)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "bumps the topic when a new version is made of the last post" do
|
it "bumps the topic when a new version is made of the last post" do
|
||||||
lambda {
|
lambda {
|
||||||
@last_post.revise(Fabricate(:moderator), 'updated contents')
|
@last_post.revise(Fabricate(:moderator), { raw: 'updated contents' })
|
||||||
@topic.reload
|
@topic.reload
|
||||||
}.should change(@topic, :bumped_at)
|
}.should change(@topic, :bumped_at)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "doesn't bump the topic when a post that isn't the last post receives a new version" do
|
it "doesn't bump the topic when a post that isn't the last post receives a new version" do
|
||||||
lambda {
|
lambda {
|
||||||
@earlier_post.revise(Fabricate(:moderator), 'updated contents')
|
@earlier_post.revise(Fabricate(:moderator), { raw: 'updated contents' })
|
||||||
@topic.reload
|
@topic.reload
|
||||||
}.should_not change(@topic, :bumped_at)
|
}.should_not change(@topic, :bumped_at)
|
||||||
end
|
end
|
||||||
|
@ -689,16 +689,17 @@ describe Topic do
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'with category' do
|
describe 'with category' do
|
||||||
|
|
||||||
before do
|
before do
|
||||||
@category = Fabricate(:category)
|
@category = Fabricate(:category)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should not increase the topic_count with no category" do
|
it "should not increase the topic_count with no category" do
|
||||||
lambda { Fabricate(:topic, user: @category.user); @category.reload }.should_not change(@category, :topic_count)
|
-> { Fabricate(:topic, user: @category.user); @category.reload }.should_not change(@category, :topic_count)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should increase the category's topic_count" do
|
it "should increase the category's topic_count" do
|
||||||
lambda { Fabricate(:topic, user: @category.user, category_id: @category.id); @category.reload }.should change(@category, :topic_count).by(1)
|
-> { Fabricate(:topic, user: @category.user, category_id: @category.id); @category.reload }.should change(@category, :topic_count).by(1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -777,74 +778,6 @@ describe Topic do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'revisions' do
|
|
||||||
let(:post) { Fabricate(:post) }
|
|
||||||
let(:topic) { post.topic }
|
|
||||||
|
|
||||||
it "has no revisions by default" do
|
|
||||||
post.revisions.size.should == 0
|
|
||||||
end
|
|
||||||
|
|
||||||
context 'changing title' do
|
|
||||||
|
|
||||||
before do
|
|
||||||
topic.title = "new title for the topic"
|
|
||||||
topic.save
|
|
||||||
end
|
|
||||||
|
|
||||||
it "creates a new revision" do
|
|
||||||
post.revisions.size.should == 1
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
context 'changing category' do
|
|
||||||
let(:category) { Fabricate(:category) }
|
|
||||||
|
|
||||||
it "creates a new revision" do
|
|
||||||
topic.change_category_to_id(category.id)
|
|
||||||
post.revisions.size.should == 1
|
|
||||||
end
|
|
||||||
|
|
||||||
it "does nothing for private messages" do
|
|
||||||
topic.archetype = "private_message"
|
|
||||||
topic.category_id = nil
|
|
||||||
|
|
||||||
topic.change_category_to_id(category.id)
|
|
||||||
topic.category_id.should == nil
|
|
||||||
end
|
|
||||||
|
|
||||||
context "removing a category" do
|
|
||||||
before do
|
|
||||||
topic.change_category_to_id(category.id)
|
|
||||||
topic.change_category_to_id(nil)
|
|
||||||
end
|
|
||||||
|
|
||||||
it "creates a new revision" do
|
|
||||||
post.revisions.size.should == 2
|
|
||||||
last_rev = post.revisions.order(:number).last
|
|
||||||
last_rev.previous("category_id").should == category.id
|
|
||||||
last_rev.current("category_id").should == SiteSetting.uncategorized_category_id
|
|
||||||
post.reload
|
|
||||||
post.version.should == 3
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
context 'bumping the topic' do
|
|
||||||
before do
|
|
||||||
topic.bumped_at = 10.minutes.from_now
|
|
||||||
topic.save
|
|
||||||
end
|
|
||||||
|
|
||||||
it "doesn't create a new version" do
|
|
||||||
post.revisions.size.should == 0
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'change_category' do
|
describe 'change_category' do
|
||||||
|
|
||||||
before do
|
before do
|
||||||
|
|
|
@ -49,7 +49,6 @@ describe UserAction do
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'includes the events correctly' do
|
it 'includes the events correctly' do
|
||||||
|
|
||||||
mystats = stats_for_user(user)
|
mystats = stats_for_user(user)
|
||||||
expecting = [UserAction::NEW_TOPIC, UserAction::NEW_PRIVATE_MESSAGE, UserAction::GOT_PRIVATE_MESSAGE, UserAction::BOOKMARK].sort
|
expecting = [UserAction::NEW_TOPIC, UserAction::NEW_PRIVATE_MESSAGE, UserAction::GOT_PRIVATE_MESSAGE, UserAction::BOOKMARK].sort
|
||||||
mystats.should == expecting
|
mystats.should == expecting
|
||||||
|
@ -66,7 +65,6 @@ describe UserAction do
|
||||||
stream_count.should == 0
|
stream_count.should == 0
|
||||||
|
|
||||||
# groups
|
# groups
|
||||||
|
|
||||||
category = Fabricate(:category, read_restricted: true)
|
category = Fabricate(:category, read_restricted: true)
|
||||||
|
|
||||||
public_topic.recover!
|
public_topic.recover!
|
||||||
|
@ -90,18 +88,16 @@ describe UserAction do
|
||||||
# duplicate should not exception out
|
# duplicate should not exception out
|
||||||
log_test_action
|
log_test_action
|
||||||
|
|
||||||
|
|
||||||
# recategorize belongs to the right user
|
# recategorize belongs to the right user
|
||||||
category2 = Fabricate(:category)
|
category2 = Fabricate(:category)
|
||||||
admin = Fabricate(:admin)
|
admin = Fabricate(:admin)
|
||||||
public_topic.acting_user = admin
|
public_post.revise(admin, { category_id: category2.id})
|
||||||
public_topic.change_category_to_id(category2.id)
|
|
||||||
|
|
||||||
action = UserAction.stream(user_id: public_topic.user_id, guardian: Guardian.new)[0]
|
action = UserAction.stream(user_id: public_topic.user_id, guardian: Guardian.new)[0]
|
||||||
action.acting_user_id.should == admin.id
|
action.acting_user_id.should == admin.id
|
||||||
action.action_type.should == UserAction::EDIT
|
action.action_type.should == UserAction::EDIT
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'when user likes' do
|
describe 'when user likes' do
|
||||||
|
@ -121,7 +117,6 @@ describe UserAction do
|
||||||
it "creates a new stream entry" do
|
it "creates a new stream entry" do
|
||||||
PostAction.act(liker, post, PostActionType.types[:like])
|
PostAction.act(liker, post, PostActionType.types[:like])
|
||||||
likee_stream.count.should == @old_count + 1
|
likee_stream.count.should == @old_count + 1
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
context "successful like" do
|
context "successful like" do
|
||||||
|
|
|
@ -192,7 +192,7 @@ describe BadgeGranter do
|
||||||
|
|
||||||
UserBadge.where(user_id: user.id, badge_id: Badge::Editor).count.should eq(0)
|
UserBadge.where(user_id: user.id, badge_id: Badge::Editor).count.should eq(0)
|
||||||
|
|
||||||
PostRevisor.new(post).revise!(user, "This is my new test 1235 123")
|
PostRevisor.new(post).revise!(user, { raw: "This is my new test 1235 123" })
|
||||||
BadgeGranter.process_queue!
|
BadgeGranter.process_queue!
|
||||||
|
|
||||||
UserBadge.where(user_id: user.id, badge_id: Badge::Editor).count.should eq(1)
|
UserBadge.where(user_id: user.id, badge_id: Badge::Editor).count.should eq(1)
|
||||||
|
|
|
@ -20,7 +20,7 @@ describe PostAlerter do
|
||||||
it "won't notify the user a second time on revision" do
|
it "won't notify the user a second time on revision" do
|
||||||
p1 = create_post_with_alerts(raw: '[quote="Evil Trout, post:1"]whatup[/quote]')
|
p1 = create_post_with_alerts(raw: '[quote="Evil Trout, post:1"]whatup[/quote]')
|
||||||
lambda {
|
lambda {
|
||||||
p1.revise(p1.user, '[quote="Evil Trout, post:1"]whatup now?[/quote]')
|
p1.revise(p1.user, { raw: '[quote="Evil Trout, post:1"]whatup now?[/quote]' })
|
||||||
}.should_not change(evil_trout.notifications, :count)
|
}.should_not change(evil_trout.notifications, :count)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -67,7 +67,7 @@ describe PostAlerter do
|
||||||
it "won't notify the user a second time on revision" do
|
it "won't notify the user a second time on revision" do
|
||||||
mention_post
|
mention_post
|
||||||
lambda {
|
lambda {
|
||||||
mention_post.revise(mention_post.user, "New raw content that still mentions @eviltrout")
|
mention_post.revise(mention_post.user, { raw: "New raw content that still mentions @eviltrout" })
|
||||||
}.should_not change(evil_trout.notifications, :count)
|
}.should_not change(evil_trout.notifications, :count)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue