require "json"
require 'openssl'
module RailsConnector
# The CMS file class
#
# [children] an Array of objects, Obj, of which this one is the parent
# [parent] the Obj of which this one is a child - nil for the root object
# @api public
class Obj < CmsBaseModel
CRYPT_KEY = "\xd7\x28\x9c\x63\xd6\x29\xdf\x20\xcd\x32\xcf\x30\xcf\x30\xcf\x30\xdf\x20\xb6\x49"\
"\x91\x6e\x99\x66\x90\x6f\x8f\x70\x9e\x61\x8d\x72\x94\x6b\xdf\x20\xbe\x41\xb8\x47\xc4\x3b"\
"\xdf\x20\xbd\x42\x9a\x65\x8d\x72\x93\x6c\x96\x69\x91\x6e".freeze
include DateAttribute
include SEO
self.store_full_sti_class = false
def self.configure_for_content(which)
case which
when :released then configure_column_information("objs", true)
when :edited
configure_column_information("preview_objs", false)
has_many(:permissions, :class_name => "::RailsConnector::Permission", :foreign_key => "object_id")
else
raise "configure_for_content called with unknown parameter #{which}"
end
end
def self.configure_column_information(table_name_postfix, use_cached_permissions)
reset_column_information
self.table_name = "#{table_name_prefix}#{table_name_postfix}"
self.primary_key = "obj_id"
self.inheritance_column = "obj_class"
@@use_cached_permissions = use_cached_permissions
end
# use released contents as a default
configure_for_content(:released)
# Patch to avoid a type_condition being added by ActiveRecord::Base.add_conditions! for Obj.find(params):
def self.descends_from_active_record?
superclass == CmsBaseModel
end
# A CMS administrator can specify the obj_class for a given CMS object.
# In Rails, this could be either:
#
# * A valid and existing model name
# * A valid and non-existing model name
# * An invalid model name
#
# Rails' STI mechanism only considers the first case.
# In any other case, RailsConnector::Obj is used, except when explicitely asked
# for a model in the RailsConnector namespace (RailsConnector::Permission etc.)
def self.compute_type(type_name)
try_type { type_name.constantize } ||
self
end
def self.reset_type_cache
# We don't cache types at all here.
end
# @api public
def permissions
@@use_cached_permissions ? attr_dict.permissions : super
end
# @api public
def permitted_for_user?(user)
if permitted_groups.blank?
true
else
if user
(permitted_groups & user.live_server_groups).any?
else
false
end
end
end
# Returns the root Obj. Its id is 2001 and cannot be changed.
# @api public
def self.root
Obj.find(2001)
end
# Returns the homepage object. This can be overwritten in your application's +ObjExtensions+.
# Use Obj#homepage? to check if an object is the homepage.
# @api public
def self.homepage
root
end
# for testing purposes only
def self.reset_homepage
@@homepage_id = nil
end
# returns the obj's permalink.
# @api public
def permalink
self[:permalink]
end
# This method determines the controller that should be invoked when the Obj is requested.
# By default a controller matching the Obj's obj_class will be used.
# If the controller does not exist, the CmsController will be used as a fallback.
# Overwrite this method to force a different controller to be used.
# @api public
def controller_name
obj_class
end
# This method determines the action that should be invoked when the Obj is requested.
# The default action is 'index'.
# Overwrite this method to force a different action to be used.
# @api public
def controller_action_name
"index"
end
@@homepage_id = nil
# Returns true if the current object has the same id as the homepage object. Always use this method instead of
# manually comparing an object to Obj.homepage, as Obj#homepage? caches the object id
# and thus requires no extra database access.
# @api public
def homepage?
self.id == (@@homepage_id ||= self.class.homepage.id)
end
# This method is used to calculate a part of a URL of an obj.
#
# The routing schema: /
#
# The default is +obj.name+.
#
# You can customize this part by overwriting +obj.slug+ in {ObjExtentions}.
# @return [String]
# @api public
def slug
name
end
# Returns the title of the content or the name.
# @api public
def display_title
self.title || name
end
OBJECT_TYPES = {
'2' => :document,
'5' => :publication,
'B' => :image,
'C' => :generic
}.freeze unless defined?(OBJECT_TYPES)
# Returns the type of the object: :document, :publication, :image or :generic
# @api public
def object_type
OBJECT_TYPES[obj_type_code]
end
# Returns true if image? or generic?
# @api public
def binary?
[:image, :generic].include? object_type
end
# Returns true if object_type == :image
# @api public
def image?
object_type == :image
end
# Returns true if object_type == :generic
# @api public
def generic?
object_type == :generic
end
# Returns true if object_type == :publication (for folders)
# @api public
def publication?
object_type == :publication
end
# Returns true if object_type == :document
# @api public
def document?
object_type == :document
end
# Returns true if this object is active (time_when is in objects time interval)
# @api public
def active?(time_when = nil)
return false if !valid_from
time_then = time_when || Obj.preview_time
valid_from <= time_then && (!valid_until || time_then <= valid_until)
end
# Returns true if this object has edited content.
# Note that edited content is not available when Configuration.mode == :live
# and edited? will always return false in this case.
# @api public
def edited?
(is_edited == 1)
end
# Returns true if this object has released content
# @api public
def released?
(is_released == 1)
end
# Returns true if the Obj is suppressed.
# A suppressed Obj does not represent an entire web page, but only a part of a page
# (for example a teaser) and will not be delivered by the rails application
# as a standalone web page.
# @api public
def suppressed?
suppress_export == 1
end
# Returns true if the export of the object is not suppressed and the content is active?
# @api public
def exportable?(current_time = nil)
!suppressed? && active?(current_time)
end
# Returns the file name to which the Content.file_extension has been appended.
# @api public
def filename
extension = ".#{file_extension}" unless file_extension.blank?
"#{name}#{extension}"
rescue NoMethodError
name
end
# Returns an array with the names of groups that are permitted to access this Obj.
# This corresponds to the cms permission "permissionLiveServerRead".
# @api public
def permitted_groups
attr_dict.permitted_groups
end
# Returns true if this object is the root object.
# @api public
def root?
parent_obj_id.nil?
end
has_many :children, :class_name => 'Obj', :foreign_key => 'parent_obj_id'
belongs_to :parent, :class_name => 'Obj', :foreign_key => 'parent_obj_id'
# Returns a list of exportable? children excluding the binary? ones unless :all is specfied.
# This is mainly used for navigations.
# @api public
def toclist(*args)
return [] unless publication?
time = args.detect {|value| value.kind_of? Time}
toclist = children.find(:all).select{ |toc| toc.exportable?(time) }
toclist = toclist.reject { |toc| toc.binary? } unless args.include?(:all)
toclist
end
# Returns the sorted +toclist+, respecting sort order and type of this Obj.
# @api public
def sorted_toclist(*args)
list = self.toclist(*args)
return [] if list.blank?
cached_sort_key1 = self.sort_key1
cached_sort_type1 = self.sort_type1
sorted_list =
if cached_sort_key1.blank?
list.sort { |left_obj, right_obj| left_obj.name <=> right_obj.name }
else
cached_sort_key2 = self.sort_key2
cached_sort_type2 = self.sort_type2
cached_sort_key3 = self.sort_key3
cached_sort_type3 = self.sort_type3
list.sort do |left_obj, right_obj|
compare = compare_on_sort_key(left_obj, right_obj, cached_sort_key1, cached_sort_type1)
if compare == 0 && cached_sort_key2
compare = compare_on_sort_key(left_obj, right_obj, cached_sort_key2, cached_sort_type2)
if compare == 0 && cached_sort_key3
compare = compare_on_sort_key(left_obj, right_obj, cached_sort_key3, cached_sort_type3)
end
end
compare
end
end
return self.sort_order == "descending" ? sorted_list.reverse : sorted_list
end
def sort_key1
self[:sort_key1]
end
def sort_key2
self[:sort_key2]
end
def sort_key3
self[:sort_key3]
end
# Returns an Array of all the ancestor objects, starting at the root and ending at this object's parent.
# @api public
def ancestors
if parent
parent.ancestors + [parent]
else
[]
end
end
# Returns the Object with the given name next in the hierarchy
# returns nil if no object with the given name was found.
# @api public
def find_nearest(name)
obj = self.children.find_by_name(name)
return obj if obj and obj.active?
parent.find_nearest(name) unless self.root?
end
# Returns the value of the attribute specified by its name.
#
# Passing an invalid key will not raise an error, but return nil.
# @api public
def [](key)
# convenience access to name
return name if key.to_sym == :name
# regular activerecord attributes
if @attributes.include?(key.to_s)
if key == :valid_from or key == :valid_until or key == :last_changed
return as_date(super(key))
else
return super(key)
end
end
# Unknown Obj attributes are delegated to the corresponding instance of AttrDict.
begin
return (attr_dict.send key)
rescue NoMethodError
end
# fallback
return nil
end
# Reloads the attributes of this object from the database, invalidates custom attributes
# @api public
def reload
super
@attr_dict = nil
@attr_values = nil
@attr_defs = nil
self
end
# object_name is a legacy alias to name. Please use name instead, because only name
# works with ActiveRecord's dynamic finder methods like find_by_name(...)
def object_name
name
end
# object_class is a legacy alias to name. Please use obj_class instead, because only obj_class
# works with ActiveRecord's dynamic finder methods like find_by_obj_class(...)
# @api public
def object_class
obj_class
end
# for binary Objs body_length equals the file size
# for non-binary Objs body_length equals the number of characters in the body (main content)
# @api public
def body_length
attr_dict.body_length
end
# Override this method to provide an external url
# where the content (the body) of this Obj can be downloaded.
# The Rails Connector will then use this url when creating links to this Obj.
# This is useful when delivering images via a content delivery network (CDN), for example.
# Returns nil by default.
#
# Note: When this method returns an url, the Rails Connector's DefaultCmsController
# will redirect to this url instead of delivering this Obj's body.
# And the Rails Connector's cms_path and cms_url helpers will link to this url
# instead of linking to the Obj.
#
# Note also that the url will be used regardless of the Obj's permissions, so be careful not
# to provide urls that contain unprotected secrets.
# @api public
def body_data_url
nil
end
def set_attr_values(dictionary)
@attr_values = dictionary
@attr_dict = nil
end
# Returns an instance of AttrDict which provides access to formatted attribute values (i.e. content).
# The method uses attr_defs to determine the right formatting depending on the particular field type.
def attr_dict
@attr_dict ||= AttrDict.new(self, attr_values, attr_defs)
end
# Returns a nested hash of attribute values.
def attr_values
@attr_values ||= begin
encoded_and_encrypted_attr_values = read_attribute(:attr_values)
return {} unless encoded_and_encrypted_attr_values
encrypted_attr_values = Base64.decode64(encoded_and_encrypted_attr_values)
if !encrypted_attr_values.starts_with?('Salted__') || encrypted_attr_values.length < 16
raise 'attr_values has wrong format'
end
cipher = OpenSSL::Cipher.new('rc4')
cipher.decrypt
salt = encrypted_attr_values[8..15]
cipher.pkcs5_keyivgen(CRYPT_KEY, salt, 1)
encrypted_attr_values = encrypted_attr_values[16..-1]
if encrypted_attr_values.present?
decrypted_attr_values = cipher.update(encrypted_attr_values)
decrypted_attr_values << cipher.final
JSON.parse(decrypted_attr_values)
else
{}
end
end
end
# Returns a nested hash of attribute definitions.
def attr_defs
@attr_defs ||= JSON.parse(read_attribute(:attr_defs) || "{}")
end
# Provides access to field metadata:
#
# <%= @obj.metadata_for_field(:body_de, :titles, :en) %>
#
# In addition to the field name, the method takes an arbitrary number of arguments
# constituting a path through the nested (hash) structures of the attribute definition (attr_defs).
#
# If the path doesn't fit the metadata structure, the method returns nil and doesn't raise an exception.
def metadata_for_field(field, *args)
rslt = fiona_fields[field.to_s] || attr_defs[field.to_s]
args.each do |key|
rslt = rslt[key.to_s] unless rslt.nil?
end
rslt
end
# @api public
def last_changed
self[:last_changed]
end
# @api public
def valid_from
self[:valid_from]
end
# @api public
def valid_until
self[:valid_until]
end
# deprecated, use file_extension instead
def content_type
logger.warn "DEPRECATION WARNING: Obj#content_type is deprecated, use file_extension instead"
file_extension
end
# Returns the MIME-type as determined from the file_extension - see MIME::Types
# @api public
def mime_type
@mime_type ||= compute_mime_type
end
def to_liquid
LiquidSupport::ObjDrop.new(self)
end
def respond_to?(method_id, include_private=false)
if super
true
elsif %w(_attr_dict _attr_defs _attr_values).include?(method_id.to_s)
# prevent infinite recursion when calling "attr_*" below,
# since rails checks the absence of an "_attr_*" method internally
return false
else
attr_dict.respond_to?(method_id)
end
end
def self.preview_time=(t)
Thread.current[:preview_time] = t
end
def self.preview_time
Thread.current[:preview_time] || Time.now
end
private
def as_date(value)
DateAttribute.parse(value) unless value.nil?
end
# Forwards any unknown method call to a corresponding instance
# of AttrDict and thus provides access to object fields, i.e. content.
#
# In case of an invalid method call, an error is raised.
#
# Hint: Use [] instead to suppress error messages.
def method_missing(method_id, *args)
super
rescue NoMethodError, NameError
# prevent infinite recursion when calling "attr_*" below,
# since rails checks the absence of an "_attr_*" method internally
raise if %w(_attr_dict _attr_defs _attr_values).include?(method_id.to_s)
if attr_dict.respond_to?(method_id)
attr_dict.send method_id, *args
else
raise
end
end
def fiona_fields
@fiona_fields ||=
['name', 'obj_class', 'workflow', 'suppressexport', 'permalink'].inject({}) do |all,field|
all.merge! field => {
'titles' => {'de' => field.humanize, 'en' => field.humanize},
'type' => 'string',
'help_texts' => {'de' => field, 'en' => field}
}
end
end
def compute_mime_type
MIME::Types.type_for(file_extension).first.content_type
rescue
binary? ? "application/octet-stream" : "text/plain"
end
def compare_on_sort_key(left_obj, right_obj, my_sort_key, my_sort_type)
left_value = left_obj[my_sort_key]
right_value = right_obj[my_sort_key]
if left_value.nil?
1
elsif right_value.nil?
-1
# hardcoded type check needed for speed
elsif left_value.is_a?(Time) && right_value.is_a?(Time)
left_value <=> right_value
else
if my_sort_type == "numeric"
(left_value.to_i rescue 0) <=> (right_value.to_i rescue 0)
else
left_value.to_s.downcase <=> right_value.to_s.downcase
end
end
end
def self.try_type
result = yield
result if class_of_active_record_descendant(result)
rescue NameError, ActiveRecord::ActiveRecordError
nil
end
end
end