OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/lib/api/decorators/schema_representer.rb

376 lines
14 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2021 the OpenProject GmbH
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See docs/COPYRIGHT.rdoc for more details.
#++
module API
module Decorators
class SchemaRepresenter < Single
module InstanceMethods
module_function
def call_or_use(object)
if object.respond_to? :call
instance_exec(&object)
else
object
end
end
def call_or_translate(object, rep_class = self.class.represented_class)
if object.respond_to? :call
instance_exec(&object)
else
rep_class.human_attribute_name(object)
end
end
end
class << self
def schema(property,
type:,
name_source: property,
as: camelize(property),
required: true,
has_default: false,
writable: default_writable_property(property),
attribute_group: nil,
min_length: nil,
max_length: nil,
regular_expression: nil,
options: {},
show_if: true)
getter = ->(*) do
schema_property_getter(type,
name_source,
required,
has_default,
writable,
attribute_group,
min_length,
max_length,
regular_expression,
options)
end
schema_property(property,
getter,
show_if,
required,
has_default,
name_source,
as)
end
def schema_with_allowed_link(property,
type: make_type(property),
name_source: property,
as: camelize(property),
href_callback:,
required: true,
has_default: false,
writable: default_writable_property(property),
attribute_group: nil,
show_if: true)
getter = ->(*) do
schema_with_allowed_link_property_getter(type,
name_source,
required,
has_default,
writable,
attribute_group,
href_callback)
end
schema_property(property,
getter,
show_if,
required,
has_default,
name_source,
as)
end
def schema_with_allowed_collection(property,
type: make_type(property),
name_source: property,
as: camelize(property),
values_callback: -> do
represented.assignable_values(property, current_user)
end,
value_representer:,
link_factory:,
required: true,
has_default: false,
writable: default_writable_property(property),
attribute_group: nil,
show_if: true)
getter = ->(*) do
schema_with_allowed_collection_getter(type,
name_source,
current_user,
value_representer,
link_factory,
required,
has_default,
writable,
attribute_group,
values_callback,
nil)
end
schema_property(property,
getter,
show_if,
required,
has_default,
name_source,
as)
end
def schema_with_allowed_string_collection(property,
type: make_type(property),
name_source: property,
as: camelize(property),
values_callback: -> do
represented.assignable_values(property, current_user)
end,
required: true,
has_default: false,
writable: default_writable_property(property),
attribute_group: nil,
show_if: true)
getter = ->(*) do
schema_with_allowed_collection_getter(type,
name_source,
current_user,
nil,
nil,
required,
has_default,
writable,
attribute_group,
values_callback,
->(*) {
next unless allowed_values
allowed_values
})
end
schema_property(property,
getter,
show_if,
required,
has_default,
name_source,
as)
end
def schema_property(property,
getter,
show_if,
required,
has_default,
name_source,
property_alias)
raise ArgumentError unless property
property property,
as: property_alias,
exec_context: :decorator,
getter: getter,
if: show_if,
required: required,
has_default: has_default,
name_source: lambda {
API::Decorators::SchemaRepresenter::InstanceMethods
.call_or_translate name_source, represented_class
}
end
def represented_class; end
private
def make_type(property_name)
property_name.to_s.camelize
end
def default_writable_property(property)
-> do
if represented.respond_to?(:writable?)
represented.writable?(property)
else
false
end
end
end
end
include InstanceMethods
def self.create(represented, current_user:, self_link: nil, form_embedded: false)
new(represented,
self_link: self_link,
current_user: current_user,
form_embedded: form_embedded)
end
def self.representable_definitions
representable_config = self.representable_attrs
# For reasons beyond me, Representable::Config contains the definitions
# * nested in [:definitions] in some envs, e.g. development
# * directly in other envs, e.g. test
representable_config.try(:definitions) || representable_config
end
def initialize(represented,
current_user:,
self_link: nil,
form_embedded: false)
self.form_embedded = form_embedded
self.self_link = self_link
super(represented, current_user: current_user)
end
link :self do
{ href: self_link } if self_link
end
property :_dependencies,
exec_context: :decorator
attr_accessor :form_embedded,
:self_link
def _type
'Schema'
end
def _dependencies
[]
end
def schema_property_getter(type,
name_source,
required,
has_default,
writable,
attribute_group,
min_length,
max_length,
regular_expression,
options)
name = call_or_translate(name_source)
schema = ::API::Decorators::PropertySchemaRepresenter
.new(type: call_or_use(type),
name: name,
required: call_or_use(required),
has_default: call_or_use(has_default),
writable: call_or_use(writable),
attribute_group: call_or_use(attribute_group))
schema.min_length = min_length
schema.max_length = max_length
schema.regular_expression = regular_expression
schema.options = options
schema
end
def schema_with_allowed_link_property_getter(type,
name_source,
required,
has_default,
writable,
attribute_group,
href_callback)
representer = ::API::Decorators::AllowedValuesByLinkRepresenter
.new(type: call_or_use(type),
name: call_or_translate(name_source),
required: call_or_use(required),
has_default: call_or_use(has_default),
writable: call_or_use(writable),
attribute_group: call_or_use(attribute_group))
if form_embedded
representer.allowed_values_href = instance_eval(&href_callback)
end
representer
end
def schema_with_allowed_collection_getter(type,
name_source,
current_user,
value_representer,
link_factory,
required,
has_default,
writable,
attribute_group,
values_callback,
allowed_values_getter)
wrapped_link_factory = if link_factory
->(value) { instance_exec(value, &link_factory) }
else
link_factory
end
attributes = { type: call_or_use(type),
name: call_or_translate(name_source),
current_user: current_user,
value_representer: value_representer,
link_factory: wrapped_link_factory,
required: call_or_use(required),
has_default: call_or_use(has_default),
writable: call_or_use(writable),
attribute_group: call_or_use(attribute_group) }
attributes[:allowed_values_getter] = allowed_values_getter if allowed_values_getter
representer = ::API::Decorators::AllowedValuesByCollectionRepresenter
.new(**attributes)
if form_embedded
representer.allowed_values = instance_exec(&values_callback)
end
representer
end
def self.camelize(symbol)
symbol.to_s.camelize(:lower).to_sym
end
end
end
end