Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Léo-Paul Géneau
gitlab-ce
Commits
8a9597fc
Commit
8a9597fc
authored
Jan 25, 2017
by
Dmitriy Zaporozhets
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'dz-nested-groups-access-improvements' into 'master'
Nested groups feature improvemetns See merge request !8448
parents
bc7febff
52c5f9c9
Changes
11
Show whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
118 additions
and
15 deletions
+118
-15
app/helpers/groups_helper.rb
app/helpers/groups_helper.rb
+1
-1
app/models/concerns/routable.rb
app/models/concerns/routable.rb
+15
-0
app/models/group.rb
app/models/group.rb
+1
-1
app/models/namespace.rb
app/models/namespace.rb
+20
-2
app/models/route.rb
app/models/route.rb
+4
-3
app/models/user.rb
app/models/user.rb
+9
-0
app/services/users/refresh_authorized_projects_service.rb
app/services/users/refresh_authorized_projects_service.rb
+2
-1
spec/models/concerns/routable_spec.rb
spec/models/concerns/routable_spec.rb
+10
-0
spec/models/namespace_spec.rb
spec/models/namespace_spec.rb
+22
-6
spec/models/route_spec.rb
spec/models/route_spec.rb
+1
-1
spec/models/user_spec.rb
spec/models/user_spec.rb
+33
-0
No files found.
app/helpers/groups_helper.rb
View file @
8a9597fc
...
@@ -14,7 +14,7 @@ module GroupsHelper
...
@@ -14,7 +14,7 @@ module GroupsHelper
def
group_title
(
group
,
name
=
nil
,
url
=
nil
)
def
group_title
(
group
,
name
=
nil
,
url
=
nil
)
full_title
=
''
full_title
=
''
group
.
parent
s
.
each
do
|
parent
|
group
.
ancestor
s
.
each
do
|
parent
|
full_title
+=
link_to
(
simple_sanitize
(
parent
.
name
),
group_path
(
parent
))
full_title
+=
link_to
(
simple_sanitize
(
parent
.
name
),
group_path
(
parent
))
full_title
+=
' / '
.
html_safe
full_title
+=
' / '
.
html_safe
end
end
...
...
app/models/concerns/routable.rb
View file @
8a9597fc
...
@@ -60,6 +60,21 @@ module Routable
...
@@ -60,6 +60,21 @@ module Routable
joins
(
:route
).
where
(
wheres
.
join
(
' OR '
))
joins
(
:route
).
where
(
wheres
.
join
(
' OR '
))
end
end
end
end
# Builds a relation to find multiple objects that are nested under user membership
#
# Usage:
#
# Klass.member_descendants(1)
#
# Returns an ActiveRecord::Relation.
def
member_descendants
(
user_id
)
joins
(
:route
).
joins
(
"INNER JOIN routes r2 ON routes.path LIKE CONCAT(r2.path, '/%')
INNER JOIN members ON members.source_id = r2.source_id
AND members.source_type = r2.source_type"
).
where
(
'members.user_id = ?'
,
user_id
)
end
end
end
private
private
...
...
app/models/group.rb
View file @
8a9597fc
...
@@ -201,7 +201,7 @@ class Group < Namespace
...
@@ -201,7 +201,7 @@ class Group < Namespace
end
end
def
members_with_parents
def
members_with_parents
GroupMember
.
where
(
requested_at:
nil
,
source_id:
parent
s
.
map
(
&
:id
).
push
(
id
))
GroupMember
.
where
(
requested_at:
nil
,
source_id:
ancestor
s
.
map
(
&
:id
).
push
(
id
))
end
end
def
users_with_parents
def
users_with_parents
...
...
app/models/namespace.rb
View file @
8a9597fc
...
@@ -185,8 +185,26 @@ class Namespace < ActiveRecord::Base
...
@@ -185,8 +185,26 @@ class Namespace < ActiveRecord::Base
end
end
end
end
def
parents
# Scopes the model on ancestors of the record
@parents
||=
parent
?
parent
.
parents
+
[
parent
]
:
[]
def
ancestors
if
parent_id
path
=
route
.
path
paths
=
[]
until
path
.
blank?
path
=
path
.
rpartition
(
'/'
).
first
paths
<<
path
end
self
.
class
.
joins
(
:route
).
where
(
'routes.path IN (?)'
,
paths
).
reorder
(
'routes.path ASC'
)
else
self
.
class
.
none
end
end
# Scopes the model on direct and indirect children of the record
def
descendants
self
.
class
.
joins
(
:route
).
where
(
'routes.path LIKE ?'
,
"
#{
route
.
path
}
/%"
).
reorder
(
'routes.path ASC'
)
end
end
private
private
...
...
app/models/route.rb
View file @
8a9597fc
...
@@ -8,15 +8,16 @@ class Route < ActiveRecord::Base
...
@@ -8,15 +8,16 @@ class Route < ActiveRecord::Base
presence:
true
,
presence:
true
,
uniqueness:
{
case_sensitive:
false
}
uniqueness:
{
case_sensitive:
false
}
after_update
:rename_
children
,
if: :path_changed?
after_update
:rename_
descendants
,
if: :path_changed?
def
rename_
children
def
rename_
descendants
# We update each row separately because MySQL does not have regexp_replace.
# We update each row separately because MySQL does not have regexp_replace.
# rubocop:disable Rails/FindEach
# rubocop:disable Rails/FindEach
Route
.
where
(
'path LIKE ?'
,
"
#{
path_was
}
/%"
).
each
do
|
route
|
Route
.
where
(
'path LIKE ?'
,
"
#{
path_was
}
/%"
).
each
do
|
route
|
# Note that update column skips validation and callbacks.
# Note that update column skips validation and callbacks.
# We need this to avoid recursive call of rename_
children
method
# We need this to avoid recursive call of rename_
descendants
method
route
.
update_column
(
:path
,
route
.
path
.
sub
(
path_was
,
path
))
route
.
update_column
(
:path
,
route
.
path
.
sub
(
path_was
,
path
))
end
end
# rubocop:enable Rails/FindEach
end
end
end
end
app/models/user.rb
View file @
8a9597fc
...
@@ -439,6 +439,15 @@ class User < ActiveRecord::Base
...
@@ -439,6 +439,15 @@ class User < ActiveRecord::Base
Group
.
where
(
"namespaces.id IN (
#{
union
.
to_sql
}
)"
)
Group
.
where
(
"namespaces.id IN (
#{
union
.
to_sql
}
)"
)
end
end
def
nested_groups
Group
.
member_descendants
(
id
)
end
def
nested_projects
Project
.
joins
(
:namespace
).
where
(
'namespaces.parent_id IS NOT NULL'
).
member_descendants
(
id
)
end
def
refresh_authorized_projects
def
refresh_authorized_projects
Users
::
RefreshAuthorizedProjectsService
.
new
(
self
).
execute
Users
::
RefreshAuthorizedProjectsService
.
new
(
self
).
execute
end
end
...
...
app/services/users/refresh_authorized_projects_service.rb
View file @
8a9597fc
...
@@ -118,7 +118,8 @@ module Users
...
@@ -118,7 +118,8 @@ module Users
user
.
personal_projects
.
select
(
"
#{
user
.
id
}
AS user_id, projects.id AS project_id,
#{
Gitlab
::
Access
::
MASTER
}
AS access_level"
),
user
.
personal_projects
.
select
(
"
#{
user
.
id
}
AS user_id, projects.id AS project_id,
#{
Gitlab
::
Access
::
MASTER
}
AS access_level"
),
user
.
groups_projects
.
select_for_project_authorization
,
user
.
groups_projects
.
select_for_project_authorization
,
user
.
projects
.
select_for_project_authorization
,
user
.
projects
.
select_for_project_authorization
,
user
.
groups
.
joins
(
:shared_projects
).
select_for_project_authorization
user
.
groups
.
joins
(
:shared_projects
).
select_for_project_authorization
,
user
.
nested_projects
.
select_for_project_authorization
]
]
Gitlab
::
SQL
::
Union
.
new
(
relations
)
Gitlab
::
SQL
::
Union
.
new
(
relations
)
...
...
spec/models/concerns/routable_spec.rb
View file @
8a9597fc
...
@@ -68,4 +68,14 @@ describe Group, 'Routable' do
...
@@ -68,4 +68,14 @@ describe Group, 'Routable' do
end
end
end
end
end
end
describe
'.member_descendants'
do
let!
(
:user
)
{
create
(
:user
)
}
let!
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
before
{
group
.
add_owner
(
user
)
}
subject
{
described_class
.
member_descendants
(
user
.
id
)
}
it
{
is_expected
.
to
eq
([
nested_group
])
}
end
end
end
spec/models/namespace_spec.rb
View file @
8a9597fc
...
@@ -5,6 +5,8 @@ describe Namespace, models: true do
...
@@ -5,6 +5,8 @@ describe Namespace, models: true do
it
{
is_expected
.
to
have_many
:projects
}
it
{
is_expected
.
to
have_many
:projects
}
it
{
is_expected
.
to
have_many
:project_statistics
}
it
{
is_expected
.
to
have_many
:project_statistics
}
it
{
is_expected
.
to
belong_to
:parent
}
it
{
is_expected
.
to
have_many
:children
}
it
{
is_expected
.
to
validate_presence_of
(
:name
)
}
it
{
is_expected
.
to
validate_presence_of
(
:name
)
}
it
{
is_expected
.
to
validate_uniqueness_of
(
:name
).
scoped_to
(
:parent_id
)
}
it
{
is_expected
.
to
validate_uniqueness_of
(
:name
).
scoped_to
(
:parent_id
)
}
...
@@ -189,17 +191,31 @@ describe Namespace, models: true do
...
@@ -189,17 +191,31 @@ describe Namespace, models: true do
it
{
expect
(
nested_group
.
full_name
).
to
eq
(
"
#{
group
.
name
}
/
#{
nested_group
.
name
}
"
)
}
it
{
expect
(
nested_group
.
full_name
).
to
eq
(
"
#{
group
.
name
}
/
#{
nested_group
.
name
}
"
)
}
end
end
describe
'#
parent
s'
do
describe
'#
ancestor
s'
do
let
(
:group
)
{
create
(
:group
)
}
let
(
:group
)
{
create
(
:group
)
}
let
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
let
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
let
(
:deep_nested_group
)
{
create
(
:group
,
parent:
nested_group
)
}
let
(
:deep_nested_group
)
{
create
(
:group
,
parent:
nested_group
)
}
let
(
:very_deep_nested_group
)
{
create
(
:group
,
parent:
deep_nested_group
)
}
let
(
:very_deep_nested_group
)
{
create
(
:group
,
parent:
deep_nested_group
)
}
it
'returns the correct parents'
do
it
'returns the correct ancestors'
do
expect
(
very_deep_nested_group
.
parents
).
to
eq
([
group
,
nested_group
,
deep_nested_group
])
expect
(
very_deep_nested_group
.
ancestors
).
to
eq
([
group
,
nested_group
,
deep_nested_group
])
expect
(
deep_nested_group
.
parents
).
to
eq
([
group
,
nested_group
])
expect
(
deep_nested_group
.
ancestors
).
to
eq
([
group
,
nested_group
])
expect
(
nested_group
.
parents
).
to
eq
([
group
])
expect
(
nested_group
.
ancestors
).
to
eq
([
group
])
expect
(
group
.
parents
).
to
eq
([])
expect
(
group
.
ancestors
).
to
eq
([])
end
end
describe
'#descendants'
do
let!
(
:group
)
{
create
(
:group
)
}
let!
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
let!
(
:deep_nested_group
)
{
create
(
:group
,
parent:
nested_group
)
}
let!
(
:very_deep_nested_group
)
{
create
(
:group
,
parent:
deep_nested_group
)
}
it
'returns the correct descendants'
do
expect
(
very_deep_nested_group
.
descendants
.
to_a
).
to
eq
([])
expect
(
deep_nested_group
.
descendants
.
to_a
).
to
eq
([
very_deep_nested_group
])
expect
(
nested_group
.
descendants
.
to_a
).
to
eq
([
deep_nested_group
,
very_deep_nested_group
])
expect
(
group
.
descendants
.
to_a
).
to
eq
([
nested_group
,
deep_nested_group
,
very_deep_nested_group
])
end
end
end
end
end
end
spec/models/route_spec.rb
View file @
8a9597fc
...
@@ -14,7 +14,7 @@ describe Route, models: true do
...
@@ -14,7 +14,7 @@ describe Route, models: true do
it
{
is_expected
.
to
validate_uniqueness_of
(
:path
)
}
it
{
is_expected
.
to
validate_uniqueness_of
(
:path
)
}
end
end
describe
'#rename_
children
'
do
describe
'#rename_
descendants
'
do
let!
(
:nested_group
)
{
create
(
:group
,
path:
"test"
,
parent:
group
)
}
let!
(
:nested_group
)
{
create
(
:group
,
path:
"test"
,
parent:
group
)
}
let!
(
:deep_nested_group
)
{
create
(
:group
,
path:
"foo"
,
parent:
nested_group
)
}
let!
(
:deep_nested_group
)
{
create
(
:group
,
path:
"foo"
,
parent:
nested_group
)
}
let!
(
:similar_group
)
{
create
(
:group
,
path:
'gitlab-org'
)
}
let!
(
:similar_group
)
{
create
(
:group
,
path:
'gitlab-org'
)
}
...
...
spec/models/user_spec.rb
View file @
8a9597fc
...
@@ -1363,6 +1363,39 @@ describe User, models: true do
...
@@ -1363,6 +1363,39 @@ describe User, models: true do
end
end
end
end
describe
'#nested_groups'
do
let!
(
:user
)
{
create
(
:user
)
}
let!
(
:group
)
{
create
(
:group
)
}
let!
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
before
do
group
.
add_owner
(
user
)
# Add more data to ensure method does not include wrong groups
create
(
:group
).
add_owner
(
create
(
:user
))
end
it
{
expect
(
user
.
nested_groups
).
to
eq
([
nested_group
])
}
end
describe
'#nested_projects'
do
let!
(
:user
)
{
create
(
:user
)
}
let!
(
:group
)
{
create
(
:group
)
}
let!
(
:nested_group
)
{
create
(
:group
,
parent:
group
)
}
let!
(
:project
)
{
create
(
:project
,
namespace:
group
)
}
let!
(
:nested_project
)
{
create
(
:project
,
namespace:
nested_group
)
}
before
do
group
.
add_owner
(
user
)
# Add more data to ensure method does not include wrong projects
other_project
=
create
(
:project
,
namespace:
create
(
:group
,
:nested
))
other_project
.
add_developer
(
create
(
:user
))
end
it
{
expect
(
user
.
nested_projects
).
to
eq
([
nested_project
])
}
end
describe
'#refresh_authorized_projects'
,
redis:
true
do
describe
'#refresh_authorized_projects'
,
redis:
true
do
let
(
:project1
)
{
create
(
:empty_project
)
}
let
(
:project1
)
{
create
(
:empty_project
)
}
let
(
:project2
)
{
create
(
:empty_project
)
}
let
(
:project2
)
{
create
(
:empty_project
)
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment