421 lines
14 KiB
Ruby
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
|