require "traits/recordable_actions" require "workflow" class Edition include Mongoid::Document include Mongoid::Timestamps include Workflow include RecordableActions field :panopticon_id, type: String field :version_number, type: Integer, default: 1 field :sibling_in_progress, type: Integer, default: nil field :business_proposition, type: Boolean, default: false field :title, type: String field :in_beta, type: Boolean, default: false field :created_at, type: DateTime, default: lambda { Time.zone.now } field :publish_at, type: DateTime field :overview, type: String field :slug, type: String field :department, type: String field :rejected_count, type: Integer, default: 0 field :browse_pages, type: Array, default: [] field :primary_topic, type: String field :additional_topics, type: Array, default: [] field :assignee, type: String field :reviewer, type: String field :creator, type: String field :publisher, type: String field :archiver, type: String field :major_change, type: Boolean, default: false field :change_note, type: String field :review_requested_at, type: DateTime GOVSPEAK_FIELDS = [] belongs_to :assigned_to, class_name: "User" # state_machine comes from Workflow state_machine.states.map(&:name).each do |state| scope state, where(state: state) end scope :in_progress, where(:state.nin => ["archived", "published"]) scope :assigned_to, lambda { |user| if user where(assigned_to_id: user.id) else where(:assigned_to_id.exists => false) end } scope :major_updates, lambda { where(major_change: true) } validates :title, presence: true validates :version_number, presence: true, uniqueness: {scope: :panopticon_id} validates :panopticon_id, presence: true validates_with SafeHtml validates_with LinkValidator, on: :update validates_with TopicValidator, BrowsePageValidator, ReviewerValidator validates_presence_of :change_note, if: :major_change before_save :check_for_archived_artefact before_destroy :destroy_artefact index "assigned_to_id" index [["panopticon_id", Mongo::ASCENDING], ["version_number", Mongo::ASCENDING]], :unique => true index "state" class << self; attr_accessor :fields_to_clone end @fields_to_clone = [] alias_method :admin_list_title, :title def series Edition.where(panopticon_id: panopticon_id) end def history series.order([:version_number, :desc]) end def siblings series.excludes(id: id) end def previous_siblings siblings.where(:version_number.lt => version_number) end def subsequent_siblings siblings.where(:version_number.gt => version_number) end def latest_edition? subsequent_siblings.empty? end def published_edition series.where(state: "published").order(version_number: "desc").first end def previous_published_edition series.where(state: "published").order(version_number: "desc").second end def in_progress_sibling subsequent_siblings.in_progress.order(version_number: "desc").first end def can_create_new_edition? !scheduled_for_publishing? && subsequent_siblings.in_progress.empty? end def major_updates_in_series history.published.major_updates end def latest_major_update major_updates_in_series.first end def latest_change_note if latest_major_update.present? latest_major_update.change_note end end def public_updated_at if latest_major_update.present? latest_major_update.updated_at elsif (first_published_edition = series.published.order(version_number: "asc").first) first_published_edition.updated_at end end def meta_data PublicationMetadata.new self end def get_next_version_number latest_version = series.order(version_number: "desc").first.version_number latest_version + 1 end def indexable_content respond_to?(:parts) ? indexable_content_with_parts : indexable_content_without_parts end def indexable_content_without_parts if respond_to?(:body) "#{Govspeak::Document.new(body).to_text}".strip else "" end end def indexable_content_with_parts content = indexable_content_without_parts return content unless published_edition parts.inject([content]) { |acc, part| acc.concat([part.title, Govspeak::Document.new(part.body).to_text]) }.compact.join(" ").strip end # If the new clone is of the same type, we can copy all its fields over; if # we are changing the type of the edition, any fields other than the base # fields will likely be meaningless. def fields_to_copy(edition_class) edition_class == self.class ? self.class.fields_to_clone : [] end def build_clone(edition_class=nil) unless state == "published" raise "Cloning of non published edition not allowed" end unless can_create_new_edition? raise "Cloning of a published edition when an in-progress edition exists is not allowed" end edition_class = self.class unless edition_class new_edition = edition_class.new(title: self.title, version_number: get_next_version_number) real_fields_to_merge = fields_to_copy(edition_class) + [ :panopticon_id, :overview, :slug, :department, :browse_pages, :primary_topic, :additional_topics ] real_fields_to_merge.each do |attr| new_edition[attr] = read_attribute(attr) end if edition_class == AnswerEdition and %w(GuideEdition ProgrammeEdition TransactionEdition).include?(self.class.name) new_edition.body = whole_body end if edition_class == TransactionEdition and %w(AnswerEdition GuideEdition ProgrammeEdition).include?(self.class.name) new_edition.more_information = whole_body end if edition_class == GuideEdition and self.is_a?(AnswerEdition) new_edition.parts.build(title: "Part One", body: whole_body, slug: "part-one") end new_edition end def self.find_or_create_from_panopticon_data(panopticon_id, importing_user, api_credentials) existing_publication = Edition.where(panopticon_id: panopticon_id) .order_by([:version_number, :desc]).first return existing_publication if existing_publication raise "Artefact not found" unless metadata = Artefact.find(panopticon_id) importing_user.create_edition(metadata.kind.to_sym, panopticon_id: metadata.id, slug: metadata.slug, title: metadata.name, department: metadata.department, business_proposition: metadata.business_proposition) end def self.find_and_identify(slug, edition) scope = where(slug: slug) if edition.present? and edition == "latest" scope.order_by(:version_number).last elsif edition.present? scope.where(version_number: edition).first else scope.where(state: "published").order(version_number: "desc").first end end def panopticon_uri Plek.current.find("panopticon") + "/artefacts/" + (panopticon_id || slug).to_s end def format self.class.to_s.gsub("Edition", "") end def format_name format end def has_video? false end def safe_to_preview? true end def has_sibling_in_progress? ! sibling_in_progress.nil? end # Stop broadcasting a delete message unless there are no siblings. def broadcast_action(callback_action) unless callback_action == "destroyed" and self.siblings.any? super(callback_action) end end def was_published previous_siblings.each { |s| s.perform_event_without_validations(:archive) } notify_siblings_of_published_edition end def update_slug_from_artefact(artefact) self.slug = artefact.slug self.save! end def check_for_archived_artefact if panopticon_id a = Artefact.find(panopticon_id) if a.state == "archived" and changes.any? # If we're only changing the state to archived, that's ok # Any other changes are not allowed allowed_keys = ["state", "updated_at"] unless ((changes.keys - allowed_keys).empty?) and state == "archived" raise "Editing of an edition with an Archived artefact is not allowed" end end end end def artefact Artefact.find(panopticon_id) end # When we delete an edition is the only one in its series # we delete the associated artefact to remove all trace of the # item from the system. # # We don't do this by notifying panopticon as this will only ever # happen for artefacts representing editions that haven't been # published (and therefore aren't registered in the rest of the) # system. def destroy_artefact if can_destroy? && siblings.empty? Artefact.find(self.panopticon_id).destroy end end end