| 
									
										
										
										
											2019-03-16 09:37:35 +00:00
										 |  |  | # frozen_string_literal: true | 
					
						
							| 
									
										
										
										
											2019-03-15 01:32:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  | # Redmine - project management software | 
					
						
							| 
									
										
										
										
											2024-02-26 22:55:54 +00:00
										 |  |  | # Copyright (C) 2006-  Jean-Philippe Lang | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  | # | 
					
						
							|  |  |  | # 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module Redmine | 
					
						
							|  |  |  |   module NestedSet | 
					
						
							|  |  |  |     module IssueNestedSet | 
					
						
							|  |  |  |       def self.included(base) | 
					
						
							|  |  |  |         base.class_eval do | 
					
						
							|  |  |  |           belongs_to :parent, :class_name => self.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           before_create :add_to_nested_set, :if => lambda {|issue| issue.parent.present?} | 
					
						
							|  |  |  |           after_create :add_as_root, :if => lambda {|issue| issue.parent.blank?} | 
					
						
							|  |  |  |           before_update :handle_parent_change, :if => lambda {|issue| issue.parent_id_changed?} | 
					
						
							|  |  |  |           before_destroy :destroy_children | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         base.extend ClassMethods | 
					
						
							|  |  |  |         base.send :include, Redmine::NestedSet::Traversing | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       private | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def target_lft | 
					
						
							|  |  |  |         scope_for_max_rgt = self.class.where(:root_id => root_id).where(:parent_id => parent_id) | 
					
						
							|  |  |  |         if id | 
					
						
							| 
									
										
										
										
											2024-05-18 06:17:51 +00:00
										 |  |  |           scope_for_max_rgt = scope_for_max_rgt.where(id: ...id) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |         max_rgt = scope_for_max_rgt.maximum(:rgt) | 
					
						
							|  |  |  |         if max_rgt | 
					
						
							|  |  |  |           max_rgt + 1
 | 
					
						
							|  |  |  |         elsif parent | 
					
						
							|  |  |  |           parent.lft + 1
 | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           1
 | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def add_to_nested_set(lock=true) | 
					
						
							| 
									
										
										
										
											2023-11-18 14:39:28 +00:00
										 |  |  |         if lock | 
					
						
							|  |  |  |           lock_nested_set { add_to_nested_set_without_lock } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           add_to_nested_set_without_lock | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def add_to_nested_set_without_lock | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         parent.send :reload_nested_set_values | 
					
						
							|  |  |  |         self.root_id = parent.root_id | 
					
						
							|  |  |  |         self.lft = target_lft | 
					
						
							|  |  |  |         self.rgt = lft + 1
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:45:21 +00:00
										 |  |  |         self.class.where(:root_id => root_id).where("lft >= ? OR rgt >= ?", lft, lft).update_all( | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             "lft = CASE WHEN lft >= :lft THEN lft + 2 ELSE lft END, " + | 
					
						
							|  |  |  |               "rgt = CASE WHEN rgt >= :lft THEN rgt + 2 ELSE rgt END", | 
					
						
							|  |  |  |             {:lft => lft} | 
					
						
							|  |  |  |           ] | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def add_as_root | 
					
						
							|  |  |  |         self.root_id = id | 
					
						
							|  |  |  |         self.lft = 1
 | 
					
						
							|  |  |  |         self.rgt = 2
 | 
					
						
							|  |  |  |         self.class.where(:id => id).update_all(:root_id => root_id, :lft => lft, :rgt => rgt) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def handle_parent_change | 
					
						
							| 
									
										
										
										
											2023-11-18 14:39:28 +00:00
										 |  |  |         lock_nested_set do | 
					
						
							|  |  |  |           reload_nested_set_values | 
					
						
							|  |  |  |           if parent_id_was | 
					
						
							|  |  |  |             remove_from_nested_set | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           if parent | 
					
						
							|  |  |  |             move_to_nested_set | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |           reload_nested_set_values | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def move_to_nested_set | 
					
						
							|  |  |  |         if parent | 
					
						
							|  |  |  |           previous_root_id = root_id | 
					
						
							|  |  |  |           self.root_id = parent.root_id | 
					
						
							| 
									
										
										
										
											2019-06-06 14:50:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |           lft_after_move = target_lft | 
					
						
							| 
									
										
										
										
											2020-07-03 11:45:21 +00:00
										 |  |  |           self.class.where(:root_id => parent.root_id).update_all( | 
					
						
							|  |  |  |             [ | 
					
						
							|  |  |  |               "lft = CASE WHEN lft >= :lft THEN lft + :shift ELSE lft END, " + | 
					
						
							|  |  |  |                 "rgt = CASE WHEN rgt >= :lft THEN rgt + :shift ELSE rgt END", | 
					
						
							|  |  |  |               {:lft => lft_after_move, :shift => (rgt - lft + 1)} | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |           ) | 
					
						
							|  |  |  |           self.class.where(:root_id => previous_root_id).update_all( | 
					
						
							|  |  |  |             [ | 
					
						
							|  |  |  |               "root_id = :root_id, lft = lft + :shift, rgt = rgt + :shift", | 
					
						
							|  |  |  |               {:root_id => parent.root_id, :shift => lft_after_move - lft} | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |           ) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |           self.lft, self.rgt = lft_after_move, (rgt - lft + lft_after_move) | 
					
						
							|  |  |  |           parent.send :reload_nested_set_values | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def remove_from_nested_set | 
					
						
							|  |  |  |         self.class.where(:root_id => root_id).where("lft >= ? AND rgt <= ?", lft, rgt). | 
					
						
							|  |  |  |           update_all(["root_id = :id, lft = lft - :shift, rgt = rgt - :shift", {:id => id, :shift => lft - 1}]) | 
					
						
							| 
									
										
										
										
											2019-06-06 14:50:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 11:45:21 +00:00
										 |  |  |         self.class.where(:root_id => root_id).update_all( | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             "lft = CASE WHEN lft >= :lft THEN lft - :shift ELSE lft END, " + | 
					
						
							|  |  |  |               "rgt = CASE WHEN rgt >= :lft THEN rgt - :shift ELSE rgt END", | 
					
						
							|  |  |  |             {:lft => lft, :shift => rgt - lft + 1} | 
					
						
							|  |  |  |           ] | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         self.root_id = id | 
					
						
							|  |  |  |         self.lft, self.rgt = 1, (rgt - lft + 1) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def destroy_children | 
					
						
							| 
									
										
										
										
											2023-11-18 14:39:28 +00:00
										 |  |  |         if @without_nested_set_update | 
					
						
							|  |  |  |           children.each {|c| c.send :destroy_without_nested_set_update} | 
					
						
							|  |  |  |           reload | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           lock_nested_set do | 
					
						
							|  |  |  |             reload_nested_set_values | 
					
						
							|  |  |  |             children.each {|c| c.send :destroy_without_nested_set_update} | 
					
						
							|  |  |  |             reload | 
					
						
							|  |  |  |             self.class.where(:root_id => root_id).where("lft > ? OR rgt > ?", lft, lft).update_all( | 
					
						
							|  |  |  |               [ | 
					
						
							|  |  |  |                 "lft = CASE WHEN lft > :lft THEN lft - :shift ELSE lft END, " + | 
					
						
							|  |  |  |                   "rgt = CASE WHEN rgt > :lft THEN rgt - :shift ELSE rgt END", | 
					
						
							|  |  |  |                 {:lft => lft, :shift => rgt - lft + 1} | 
					
						
							|  |  |  |               ] | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def destroy_without_nested_set_update | 
					
						
							|  |  |  |         @without_nested_set_update = true | 
					
						
							|  |  |  |         destroy | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def reload_nested_set_values | 
					
						
							| 
									
										
										
										
											2021-04-03 01:34:45 +00:00
										 |  |  |         self.root_id, self.lft, self.rgt = self.class.where(:id => id).pick(:root_id, :lft, :rgt) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def save_nested_set_values | 
					
						
							|  |  |  |         self.class.where(:id => id).update_all(:root_id => root_id, :lft => lft, :rgt => rgt) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def move_possible?(issue) | 
					
						
							| 
									
										
										
										
											2015-01-07 22:18:58 +00:00
										 |  |  |         new_record? || !is_or_is_ancestor_of?(issue) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def lock_nested_set | 
					
						
							| 
									
										
										
										
											2019-03-27 02:15:24 +00:00
										 |  |  |         if /sqlserver/i.match?(self.class.connection.adapter_name) | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |           lock = "WITH (ROWLOCK HOLDLOCK UPDLOCK)" | 
					
						
							| 
									
										
										
										
											2015-01-07 21:11:05 +00:00
										 |  |  |           # Custom lock for SQLServer | 
					
						
							|  |  |  |           # This can be problematic if root_id or parent root_id changes | 
					
						
							|  |  |  |           # before locking | 
					
						
							|  |  |  |           sets_to_lock = [root_id, parent.try(:root_id)].compact.uniq | 
					
						
							|  |  |  |           self.class.reorder(:id).where(:root_id => sets_to_lock).lock(lock).ids | 
					
						
							| 
									
										
										
										
											2023-11-18 14:39:28 +00:00
										 |  |  |           yield | 
					
						
							|  |  |  |         elsif Redmine::Database.mysql? | 
					
						
							|  |  |  |           # Use a global lock to prevent concurrent modifications - MySQL row locks are broken, this will run into | 
					
						
							|  |  |  |           # deadlock errors all the time otherwise. | 
					
						
							|  |  |  |           # Trying to lock just the sets in question (by basing the lock name on root_id and parent&.root_id) will run | 
					
						
							|  |  |  |           # into the same issues as the sqlserver branch above | 
					
						
							|  |  |  |           Issue.with_advisory_lock!("lock_issues", timeout_seconds: 30) do | 
					
						
							|  |  |  |             # still lock the issues in question, for good measure | 
					
						
							|  |  |  |             sets_to_lock = [id, parent_id].compact | 
					
						
							|  |  |  |             inner_join_statement = self.class.select(:root_id).where(id: sets_to_lock).distinct(:root_id).to_sql | 
					
						
							|  |  |  |             self.class.reorder(:id). | 
					
						
							|  |  |  |               joins("INNER JOIN (#{inner_join_statement}) as i2 ON #{self.class.table_name}.root_id = i2.root_id"). | 
					
						
							|  |  |  |               lock.ids | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             yield | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2015-01-07 21:11:05 +00:00
										 |  |  |         else | 
					
						
							|  |  |  |           sets_to_lock = [id, parent_id].compact | 
					
						
							| 
									
										
										
										
											2016-12-10 09:44:39 +00:00
										 |  |  |           self.class.reorder(:id).where("root_id IN (SELECT root_id FROM #{self.class.table_name} WHERE id IN (?))", sets_to_lock).lock.ids | 
					
						
							| 
									
										
										
										
											2023-11-18 14:39:28 +00:00
										 |  |  |           yield | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def nested_set_scope | 
					
						
							|  |  |  |         self.class.order(:lft).where(:root_id => root_id) | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       def same_nested_set_scope?(issue) | 
					
						
							|  |  |  |         root_id == issue.root_id | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       module ClassMethods | 
					
						
							|  |  |  |         def rebuild_tree! | 
					
						
							|  |  |  |           transaction do | 
					
						
							|  |  |  |             reorder(:id).lock.ids | 
					
						
							|  |  |  |             update_all(:root_id => nil, :lft => nil, :rgt => nil) | 
					
						
							|  |  |  |             where(:parent_id => nil).update_all(["root_id = id, lft = ?, rgt = ?", 1, 2]) | 
					
						
							| 
									
										
										
										
											2016-07-14 07:15:13 +00:00
										 |  |  |             roots_with_children = joins("JOIN #{table_name} parent ON parent.id = #{table_name}.parent_id AND parent.id = parent.root_id").distinct.pluck("parent.id") | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |             roots_with_children.each do |root_id| | 
					
						
							|  |  |  |               rebuild_nodes(root_id) | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-21 10:35:36 +00:00
										 |  |  |         def rebuild_single_tree!(root_id) | 
					
						
							|  |  |  |           root = Issue.where(:parent_id => nil).find(root_id) | 
					
						
							|  |  |  |           transaction do | 
					
						
							|  |  |  |             where(root_id: root_id).reorder(:id).lock.ids | 
					
						
							|  |  |  |             where(root_id: root_id).update_all(:lft => nil, :rgt => nil) | 
					
						
							|  |  |  |             where(root_id: root_id, parent_id: nil).update_all(["lft = ?, rgt = ?", 1, 2]) | 
					
						
							|  |  |  |             rebuild_nodes(root_id) | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 20:19:49 +00:00
										 |  |  |         private | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def rebuild_nodes(parent_id = nil) | 
					
						
							|  |  |  |           nodes = where(:parent_id => parent_id, :rgt => nil, :lft => nil).order(:id).to_a | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           nodes.each do |node| | 
					
						
							|  |  |  |             node.send :add_to_nested_set, false | 
					
						
							|  |  |  |             node.send :save_nested_set_values | 
					
						
							|  |  |  |             rebuild_nodes node.id | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |