Defect #3722
openNested projects can get in disorder
0%
Description
How to reproduce (using redmine SVN revision 2835):
- create project Toplevel A
- create project Sublevel A as child of project Toplevel A
- create project Subsublevel A as child of project Sublevel A
- create project Toplevel B
- copy project Sublevel A, name it Toplevel C. Leave the field Subproject of empty (that's also the default coming up)
- delete project Sublevel A
- now, delete project Toplevel A. You are prompted:
Toplevel A Are you sure you want to delete this project and related data ? Its subproject(s): Toplevel B, Toplevel C will be also deleted
This is definitely wrong, since Toplevel B was never a subproject of Toplevel A.
If I click OK project Toplevel B is not deleted, however, project Toplevel C now has a minus sign on its left, indicating that is has subprojects, though it has none.
Note:
My first encounter with that bug was when I wanted to delete all but one project from a set of deeply nested projects (~20 projects). I inadvertently deleted the project I wanted to keep. First I thought it was my fault, so I restored my data from a backup and tried again. Same effect, again the project was deleted. Then it became obvious to me that something must be wrong with the new functionality of nesting projects as deep as you want. I played around a bit and finally found the steps above that reproducibely lead to an inconsistent state of the nesting of the projects (as far as I can see).
Related issues
Updated by Augusto Dias Pereira dos Santos over 15 years ago
For those who had the nested set disordered in MySQL DataBase, I develop a procedure that recalculate the nested set base on the parent_id. It's not a solution, just a way to rearange the mess. It can be better.
Watch out to the max_sp_recursion_depth variable inside the procedure.
Here is:
-- Procedure BEGIN
DROP PROCEDURE `recalculateNetstedSet`//
CREATE PROCEDURE `recalculateNetstedSet`(parent INT, INOUT left_value INT)
BEGIN
DECLARE done INT DEFAULT 0;
DECLARE son_ID INTEGER;
DECLARE sons CURSOR FOR SELECT id FROM projects WHERE parent_id IS NULL;
DECLARE sons2 CURSOR FOR SELECT id FROM projects WHERE parent_id = parent;
DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = 1;
set max_sp_recursion_depth=10;
IF parent IS NOT NULL THEN
UPDATE projects set lft = left_value where id = parent;
SET left_value = left_value+1;
END IF;
OPEN sons;
OPEN sons2;
IF parent IS NULL THEN
FETCH sons INTO son_ID;
REPEAT
IF son_ID IS NOT NULL THEN
CALL recalculateNetstedSet(son_ID,left_value);
SET left_value = left_value+1;
END IF;
FETCH sons INTO son_ID;
UNTIL done END REPEAT;
ELSE
FETCH sons2 INTO son_ID;
REPEAT
IF son_ID IS NOT NULL THEN
CALL recalculateNetstedSet(son_ID,left_value);
SET left_value = left_value+1;
END IF;
FETCH sons2 INTO son_ID;
UNTIL done END REPEAT;
END IF;
UPDATE projects set rgt = left_value where id = parent;
CLOSE sons;
CLOSE sons2;
END
-- Procedure END
To call the procedure:
set @parent = NULL;
set @left = 1;
CALL recalculateNetstedSet(@parent,@left);
Updated by Augusto Dias Pereira dos Santos over 15 years ago
I think the procedure got wrong for the wiki sintax. I will try again, just the procedure this time.
Sorry I should have seen the preview ;)
DROP PROCEDURE `recalculateNetstedSet`// CREATE PROCEDURE `recalculateNetstedSet`(parent INT, INOUT left_value INT) BEGIN DECLARE done INT DEFAULT 0; DECLARE son_ID INTEGER; DECLARE sons CURSOR FOR SELECT id FROM projects WHERE parent_id IS NULL; DECLARE sons2 CURSOR FOR SELECT id FROM projects WHERE parent_id = parent; DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = 1; set max_sp_recursion_depth=10; IF parent IS NOT NULL THEN UPDATE projects set lft = left_value where id = parent; SET left_value = left_value+1; END IF; OPEN sons; OPEN sons2; IF parent IS NULL THEN FETCH sons INTO son_ID; REPEAT IF son_ID IS NOT NULL THEN CALL recalculateNetstedSet(son_ID,left_value); SET left_value = left_value+1; END IF; FETCH sons INTO son_ID; UNTIL done END REPEAT; ELSE FETCH sons2 INTO son_ID; REPEAT IF son_ID IS NOT NULL THEN CALL recalculateNetstedSet(son_ID,left_value); SET left_value = left_value+1; END IF; FETCH sons2 INTO son_ID; UNTIL done END REPEAT; END IF; UPDATE projects set rgt = left_value where id = parent; CLOSE sons; CLOSE sons2; END
Updated by Paul Groves over 15 years ago
I too experienced this, I removed the "dependent_destroy" declaration from app/project.rb as I believe this is not applicable for the awesome_nested_set gem. This seems to have made things better.
app/project.rb Line 51
-- acts_as_nested_set :order => 'name', :dependent => :destroy
++ acts_as_nested_set :order => 'name'
Hope this helps,
Paul
Updated by Chris Platts about 15 years ago
Here's another version of the stored proc which tries to keep a sane alphabetical sort order on projects and subprojects. It also corrects the typo (recalculate Netsted Set) :)
CREATE DEFINER=`redmine`@`` PROCEDURE `recalculateNestedSet`(parent INT, INOUT left_value INT) BEGIN DECLARE done INT DEFAULT 0; DECLARE son_ID INTEGER; DECLARE sons CURSOR FOR SELECT id FROM projects WHERE parent_id IS NULL ORDER BY name; DECLARE sons2 CURSOR FOR SELECT id FROM projects WHERE parent_id = parent ORDER BY name; DECLARE CONTINUE HANDLER FOR NOT FOUND SET done = 1; set max_sp_recursion_depth=10; IF parent IS NOT NULL THEN UPDATE projects set lft = left_value where id = parent; SET left_value = left_value+1; END IF; OPEN sons; OPEN sons2; IF parent IS NULL THEN FETCH sons INTO son_ID; REPEAT IF son_ID IS NOT NULL THEN CALL recalculateNestedSet(son_ID,left_value); SET left_value = left_value+1; END IF; FETCH sons INTO son_ID; UNTIL done END REPEAT; ELSE FETCH sons2 INTO son_ID; REPEAT IF son_ID IS NOT NULL THEN CALL recalculateNestedSet(son_ID,left_value); SET left_value = left_value+1; END IF; FETCH sons2 INTO son_ID; UNTIL done END REPEAT; END IF; UPDATE projects set rgt = left_value where id = parent; CLOSE sons; CLOSE sons2; END
Updated by Jean-Baptiste Barth over 14 years ago
Why not: Project.update_all(:lft=>nil,:rgt=>nil); Project.rebuild!
or something like that ?
Introducing stored procedure is limited to your own database system, while Redmine tries to support sqlite3, Postgresql and Mysql the same time.
Updated by Sven Culley over 13 years ago
This is what I did to get the order back to work:
- Go to your database
- Make a backup of your "projects" table
- Set the value for "lft" and "rgt" to "0"
- Go to you console and use "ruby script/runner -e production 'Project.rebuild!'" to rebuild the project order
In case anything goes wrong you still have the backup.
Updated by provetza provetza over 13 years ago
Hi all,
I did what Sven suggests (zero lft and rgt on projects and then Project.rebuild). Nested projects appear alphabetically inside toplevel projects, but toplevel projects are out of order. How can I make the toplevel projects appear in order?
Thanks!
Updated by Toshiya TSURU over 12 years ago
Sven Culley wrote:
This is what I did to get the order back to work:
- Go to your database
- Make a backup of your "projects" table
- Set the value for "lft" and "rgt" to "0"
- Go to you console and use "ruby script/runner -e production 'Project.rebuild!'" to rebuild the project order
In case anything goes wrong you still have the backup.
in redmine 2,
4. Go to you console and use "ruby script/runner -e production 'Project.rebuild!'" to rebuild the project order
the command above doesn't work. maybe :
ruby script/rails runner -e production 'Project.rebuild!'
Updated by Toshi MARUYAMA about 10 years ago
- Related to Feature #18860: Replace awesome_nested_set gem with a custom implementation of nested sets added