canvas-lms/app/models/wiki_page.rb

421 lines
14 KiB
Ruby

#
# Copyright (C) 2011 Instructure, Inc.
#
# This file is part of Canvas.
#
# Canvas is free software: you can redistribute it and/or modify it under
# the terms of the GNU Affero General Public License as published by the Free
# Software Foundation, version 3 of the License.
#
# Canvas is distributed in the hope that it will be useful, but WITHOUT ANY
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
# A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along
# with this program. If not, see <http://www.gnu.org/licenses/>.
#
require 'atom'
class WikiPage < ActiveRecord::Base
attr_accessible :title, :body, :url, :user_id, :editing_roles, :notify_of_update
attr_readonly :wiki_id
validates_length_of :body, :maximum => maximum_long_text_length, :allow_nil => true, :allow_blank => true
validates_presence_of :wiki_id
include Workflow
include HasContentTags
include CopyAuthorizedLinks
include ContextModuleItem
include SearchTermHelper
after_update :post_to_pandapub_when_revised
belongs_to :wiki, :touch => true
belongs_to :user
acts_as_url :title, :scope => [:wiki_id, :not_deleted], :sync_url => true
validate :validate_front_page_visibility
before_save :set_revised_at
before_validation :ensure_unique_title
after_save :touch_wiki_context
TITLE_LENGTH = WikiPage.columns_hash['title'].limit rescue 255
SIMPLY_VERSIONED_EXCLUDE_FIELDS = [:workflow_state, :editing_roles, :notify_of_update]
def touch_wiki_context
self.wiki.touch_context if self.wiki && self.wiki.context
end
def validate_front_page_visibility
if !published? && self.is_front_page?
self.errors.add(:published, t(:cannot_unpublish_front_page, "cannot unpublish front page"))
end
end
def ensure_unique_title
return if deleted?
to_cased_title = ->(string) { string.gsub(/[^\w]+/, " ").gsub(/\b('?[a-z])/){$1.capitalize}.strip }
self.title ||= to_cased_title.call(self.url || "page")
return unless self.wiki
# TODO i18n (see wiki.rb)
if self.title == "Front Page" && self.new_record?
baddies = self.wiki.wiki_pages.not_deleted.where(title: "Front Page").select{|p| p.url != "front-page" }
baddies.each{|p| p.title = to_cased_title.call(p.url); p.save_without_broadcasting! }
end
if existing = self.wiki.wiki_pages.not_deleted.where(title: self.title).first
return if existing == self
real_title = self.title.gsub(/-(\d*)\z/, '') # remove any "-#" at the end
n = $1 ? $1.to_i + 1 : 2
begin
mod = "-#{n}"
new_title = real_title[0...(TITLE_LENGTH - mod.length)] + mod
n = n.succ
end while self.wiki.wiki_pages.not_deleted.where(title: new_title).exists?
self.title = new_title
end
end
def self.title_order_by_clause
best_unicode_collation_key('wiki_pages.title')
end
def ensure_unique_url
url_attribute = self.class.url_attribute
base_url = self.send(url_attribute)
base_url = self.send(self.class.attribute_to_urlify).to_s.to_url if base_url.blank? || !self.only_when_blank
conditions = [wildcard("#{url_attribute}", base_url, :type => :right)]
unless new_record?
conditions.first << " and id != ?"
conditions << id
end
# make stringex scoping a little more useful/flexible... in addition to
# the normal constructed attribute scope(s), it also supports paramater-
# less scopeds. note that there needs to be an instance_method of
# the same name for this to work
scopes = self.class.scope_for_url ? Array(self.class.scope_for_url) : []
base_scope = self.class
scopes.each do |scope|
next unless self.respond_to?(scope)
if base_scope.respond_to?(scope)
return unless send(scope)
base_scope = base_scope.send(scope)
else
conditions.first << " and #{self.class.connection.quote_column_name(scope)} = ?"
conditions << send(scope)
end
end
url_owners = base_scope.where(conditions).to_a
# This is the part in stringex that messed us up, since it will never allow
# a url of "front-page" once "front-page-1" or "front-page-2" is created
# We modify it to allow "front-page" and start the indexing at "front-page-2"
# instead of "front-page-1"
if url_owners.size > 0 && url_owners.detect{|u| u.send(url_attribute) == base_url}
n = 2
while url_owners.detect{|u| u.send(url_attribute) == "#{base_url}-#{n}"}
n = n.succ
end
write_attribute url_attribute, "#{base_url}-#{n}"
else
write_attribute url_attribute, base_url
end
end
sanitize_field :body, CanvasSanitize::SANITIZE
copy_authorized_links(:body) { [self.context, self.user] }
validates_each :title do |record, attr, value|
if value.blank?
record.errors.add(attr, t('errors.blank_title', "Title can't be blank"))
elsif value.size > maximum_string_length
record.errors.add(attr, t('errors.title_too_long', "Title can't exceed %{max_characters} characters", :max_characters => maximum_string_length))
elsif value.to_url.blank?
record.errors.add(attr, t('errors.title_characters', "Title must contain at least one letter or number")) # it's a bit more liberal than this, but let's not complicate things
end
end
has_a_broadcast_policy
simply_versioned :exclude => SIMPLY_VERSIONED_EXCLUDE_FIELDS, :when => Proc.new { |wp|
# :user_id and :updated_at do not merit creating a version, but should be saved
exclude_fields = [:user_id, :updated_at].concat(SIMPLY_VERSIONED_EXCLUDE_FIELDS).map(&:to_s)
(wp.changes.keys.map(&:to_s) - exclude_fields).present?
}
after_save :remove_changed_flag
workflow do
state :active do
event :unpublish, :transitions_to => :unpublished
end
state :unpublished do
event :publish, :transitions_to => :active
end
state :post_delayed do
event :delayed_post, :transitions_to => :active
end
state :deleted
end
alias_method :published?, :active?
def restore
self.workflow_state = 'unpublished'
self.save
end
def set_revised_at
self.revised_at ||= Time.now
self.revised_at = Time.now if self.body_changed?
@page_changed = self.body_changed? || self.title_changed?
true
end
attr_reader :wiki_page_changed
def notify_of_update=(val)
@wiki_page_changed = Canvas::Plugin.value_to_boolean(val)
end
def notify_of_update
false
end
def remove_changed_flag
@wiki_page_changed = false
end
def version_history
self.versions.map(&:model)
end
scope :active, -> { where(:workflow_state => 'active') }
scope :deleted_last, -> { order("workflow_state='deleted'") }
scope :not_deleted, -> { where("wiki_pages.workflow_state<>'deleted'") }
scope :published, -> { where("wiki_pages.workflow_state='active'", false) }
scope :unpublished, -> { where("wiki_pages.workflow_state='unpublished'", true) }
# needed for ensure_unique_url
def not_deleted
!deleted?
end
scope :order_by_id, -> { order(:id) }
def locked_for?(user, opts={})
return false unless self.could_be_locked
Rails.cache.fetch(locked_cache_key(user), :expires_in => 1.minute) do
locked = false
if item = locked_by_module_item?(user, opts[:deep_check_if_needed])
locked = {:asset_string => self.asset_string, :context_module => item.context_module.attributes}
locked[:unlock_at] = locked[:context_module]["unlock_at"] if locked[:context_module]["unlock_at"]
end
locked
end
end
def is_front_page?
return false if self.deleted?
self.url == self.wiki.get_front_page_url # wiki.get_front_page_url checks has_front_page?
end
def set_as_front_page!
if self.unpublished?
self.errors.add(:front_page, t(:cannot_set_unpublished_front_page, 'could not set as front page because it is unpublished'))
return false
end
self.wiki.set_front_page_url!(self.url)
end
def context_module_tag_for(context)
@tag ||= self.context_module_tags.where(context_id: context, context_type: context.class.base_class.name).first
end
def context_module_action(user, context, action)
self.context_module_tags.where(context_id: context, context_type: context.class.base_class.name).each do |tag|
tag.context_module_action(user, action)
end
end
set_policy do
given {|user, session| self.can_read_page?(user, session)}
can :read
given {|user| user && self.can_edit_page?(user)}
can :update_content and can :read_revisions
given {|user, session| user && self.can_edit_page?(user) && self.wiki.grants_right?(user, session, :create_page)}
can :create
given {|user, session| user && self.can_edit_page?(user) && self.wiki.grants_right?(user, session, :update_page)}
can :update and can :read_revisions
given {|user, session| user && self.can_edit_page?(user) && self.published? && self.wiki.grants_right?(user, session, :update_page_content)}
can :update_content and can :read_revisions
given {|user, session| user && self.can_edit_page?(user) && self.published? && self.wiki.grants_right?(user, session, :delete_page)}
can :delete
given {|user, session| user && self.can_edit_page?(user) && self.unpublished? && self.wiki.grants_right?(user, session, :delete_unpublished_page)}
can :delete
end
def can_read_page?(user, session=nil)
return true if self.unpublished? && self.wiki.grants_right?(user, session, :view_unpublished_items)
self.published? && self.wiki.grants_right?(user, session, :read)
end
def can_edit_page?(user, session=nil)
return false unless can_read_page?(user, session)
# wiki managers are always allowed to edit
return true if wiki.grants_right?(user, session, :manage)
roles = effective_roles
# teachers implies all course admins (teachers, TAs, etc)
return true if roles.include?('teachers') && context.respond_to?(:admins) && context.admins.include?(user)
# the page must be available for users of the following roles
return false unless available_for?(user, session)
return true if roles.include?('students') && context.respond_to?(:students) && context.includes_student?(user)
return true if roles.include?('members') && context.respond_to?(:users) && context.users.include?(user)
return true if roles.include?('public')
false
end
def effective_roles
context_roles = context.default_wiki_editing_roles rescue nil
roles = (editing_roles || context_roles || default_roles).split(',')
roles == %w(teachers) ? [] : roles # "Only teachers" option doesn't grant rights excluded by RoleOverrides
end
def available_for?(user, session=nil)
return true if wiki.grants_right?(user, session, :manage)
return false unless published? || (unpublished? && wiki.grants_right?(user, session, :view_unpublished_items))
return false if locked_for?(user)
true
end
def default_roles
if context.is_a?(Group)
'members'
elsif context.is_a?(Course)
'teachers'
else
'members'
end
end
set_broadcast_policy do |p|
p.dispatch :updated_wiki_page
p.to { participants }
p.whenever do |wiki_page|
BroadcastPolicies::WikiPagePolicy.new(wiki_page).
should_dispatch_updated_wiki_page?
end
end
def context(user=nil)
shard.activate do
@context ||= Course.where(wiki_id: self.wiki_id).first || Group.where(wiki_id: self.wiki_id).first
end
end
def participants
res = []
if context && context.available?
if !self.active?
res += context.participating_admins
else
res += context.participants
end
end
res.flatten.uniq
end
def to_atom(opts={})
context = opts[:context]
Atom::Entry.new do |entry|
entry.title = t(:atom_entry_title, "Wiki Page, %{course_or_group_name}: %{page_title}", :course_or_group_name => context.name, :page_title => self.title)
entry.authors << Atom::Person.new(:name => t(:atom_author, "Wiki Page"))
entry.updated = self.updated_at
entry.published = self.created_at
entry.id = "tag:#{HostUrl.default_host},#{self.created_at.strftime("%Y-%m-%d")}:/wiki_pages/#{self.feed_code}_#{self.updated_at.strftime("%Y-%m-%d")}"
entry.links << Atom::Link.new(:rel => 'alternate',
:href => "http://#{HostUrl.context_host(context)}/#{self.context.class.to_s.downcase.pluralize}/#{self.context.id}/pages/#{self.url}")
entry.content = Atom::Content::Html.new(self.body || t('defaults.no_content', "no content"))
end
end
def user_name
(user && user.name) || t('unknown_user_name', "Unknown")
end
def to_param
url
end
def last_revision_at
res = self.revised_at || self.updated_at
res = Time.now if res.is_a?(String)
res
end
def increment_view_count(user, context = nil)
unless self.new_record?
self.with_versioning(false) do |p|
context ||= p.context
WikiPage.where(id: p).update_all("view_count=COALESCE(view_count, 0) + 1")
p.context_module_action(user, context, :read)
end
end
end
def can_unpublish?
return @can_unpublish unless @can_unpublish.nil?
@can_unpublish = !is_front_page?
end
attr_writer :can_unpublish
def self.preload_can_unpublish(context, wiki_pages)
return unless wiki_pages.any?
front_page_url = context.wiki.get_front_page_url
wiki_pages.each{|wp| wp.can_unpublish = !(wp.url == front_page_url)}
end
def initialize_wiki_page(user)
if wiki.grants_right?(user, :publish_page)
# Leave the page unpublished if the user is allowed to publish it later
self.workflow_state = 'unpublished'
else
# If they aren't, publish it automatically
self.workflow_state = 'active'
end
self.editing_roles = (context.default_wiki_editing_roles rescue nil) || default_roles
if is_front_page?
self.body = t "#application.wiki_front_page_default_content_course", "Welcome to your new course wiki!" if context.is_a?(Course)
self.body = t "#application.wiki_front_page_default_content_group", "Welcome to your new group wiki!" if context.is_a?(Group)
self.workflow_state = 'active'
end
end
def post_to_pandapub_when_revised
if revised_at_changed?
CanvasPandaPub.post_update(
"/private/wiki_page/#{self.global_id}/update", {
revised_at: self.revised_at,
revision: self.versions.current.number
})
end
end
end