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/app/models/versions/project_sharing.rb

158 lines
5.4 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2020 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-2017 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 Versions::ProjectSharing
# Returns all projects the version is available in
def projects
Project.joins(project_sharing_join)
end
private
def project_sharing_join
projects = Project.all
projects_table = projects.arel_table
versions_table = Version.arel_table
sharing_inner_select = project_sharing_select(versions_table)
join_condition = project_sharing_join_condition(sharing_inner_select, projects_table)
join_on = projects_table.create_on(join_condition)
projects_table.create_join(sharing_inner_select, join_on)
end
def project_sharing_select(versions_table)
sharing_select = if sharing == 'tree'
project_sharing_tree_select(versions_table)
else
project_sharing_default_select(versions_table)
end
sharing_id_condition = versions_table[:id].eq(id)
sharing_select
.where(sharing_id_condition)
.as('sharing')
end
def project_sharing_tree_select(versions_table)
hierarchy_table = Project.arel_table
roots_table = Project.arel_table.alias('roots')
roots_join_condition = project_sharing_tree_root_join_condition(roots_table, hierarchy_table)
sharing_select = join_project_and_version(hierarchy_table, versions_table)
sharing_select
.join(roots_table)
.on(roots_join_condition)
necessary_sharing_fields(sharing_select,
roots_table,
versions_table)
end
def project_sharing_default_select(versions_table)
hierarchy_table = Project.arel_table
sharing_select = join_project_and_version(hierarchy_table, versions_table)
necessary_sharing_fields(sharing_select,
hierarchy_table,
versions_table)
end
def necessary_sharing_fields(sharing_select, projects_table, versions_table)
sharing_select
.project(projects_table[:id],
versions_table[:id].as('version_id'),
projects_table[:lft],
projects_table[:rgt],
versions_table[:sharing])
end
def join_project_and_version(projects_table, versions_table)
join_condition = projects_table[:id].eq(versions_table[:project_id])
projects_table
.join(versions_table)
.on(join_condition)
end
def project_sharing_tree_root_join_condition(roots_table, hierarchy_table)
roots_table[:lft].lteq(hierarchy_table[:lft])
.and(roots_table[:rgt].gteq(hierarchy_table[:rgt]))
.and(roots_table[:parent_id].eq(nil))
end
def project_sharing_join_condition(sharing_table, projects_table)
case self[:sharing]
when 'tree'
project_sharing_tree_join_condition(sharing_table, projects_table)
when 'descendants'
project_sharing_descendants_join_condition(sharing_table, projects_table)
when 'hierarchy'
project_sharing_hierarchy_join_condition(sharing_table, projects_table)
when 'system'
Arel::Nodes::True.new
else
sharing_table[:id].eq(projects_table[:id])
end
end
def project_sharing_tree_join_condition(sharing_table, projects_table)
projects_table[:lft].gteq(sharing_table[:lft])
.and(projects_table[:rgt].lteq(sharing_table[:rgt]))
end
def project_sharing_descendants_join_condition(sharing_table, projects_table)
project_sharing_equal_condition(sharing_table, projects_table)
.or(project_sharing_below_condition(sharing_table, projects_table))
end
def project_sharing_hierarchy_join_condition(sharing_table, projects_table)
project_sharing_descendants_join_condition(sharing_table, projects_table)
.or(project_sharing_above_condition(sharing_table, projects_table))
end
def project_sharing_equal_condition(sharing_table, projects_table)
sharing_table[:id].eq(projects_table[:id])
end
def project_sharing_above_condition(sharing_table, projects_table)
projects_table[:lft].lt(sharing_table[:lft])
.and(projects_table[:rgt].gt(sharing_table[:rgt]))
end
def project_sharing_below_condition(sharing_table, projects_table)
projects_table[:lft].gt(sharing_table[:lft])
.and(projects_table[:rgt].lt(sharing_table[:rgt]))
end
end