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
1
Merge Requests
1
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
nexedi
gitlab-ce
Commits
dbbecb88
Commit
dbbecb88
authored
May 11, 2021
by
Vijay Hawoldar
Committed by
Douglas Barbosa Alexandre
May 11, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Establish billable member membership type with a Group
parent
b93a4223
Changes
11
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
254 additions
and
72 deletions
+254
-72
doc/api/members.md
doc/api/members.md
+6
-3
ee/app/finders/billed_users_finder.rb
ee/app/finders/billed_users_finder.rb
+13
-3
ee/app/models/ee/group.rb
ee/app/models/ee/group.rb
+41
-17
ee/app/models/ee/namespace.rb
ee/app/models/ee/namespace.rb
+7
-1
ee/app/models/ee/user.rb
ee/app/models/ee/user.rb
+1
-1
ee/changelogs/unreleased/vij-billable-member-source.yml
ee/changelogs/unreleased/vij-billable-member-source.yml
+5
-0
ee/lib/ee/api/entities/billable_member.rb
ee/lib/ee/api/entities/billable_member.rb
+14
-0
ee/lib/ee/api/members.rb
ee/lib/ee/api/members.rb
+12
-7
ee/spec/finders/billed_users_finder_spec.rb
ee/spec/finders/billed_users_finder_spec.rb
+39
-8
ee/spec/lib/ee/api/entities/billable_member_spec.rb
ee/spec/lib/ee/api/entities/billable_member_spec.rb
+40
-2
ee/spec/models/ee/namespace_spec.rb
ee/spec/models/ee/namespace_spec.rb
+76
-30
No files found.
doc/api/members.md
View file @
dbbecb88
...
@@ -288,7 +288,8 @@ Example response:
...
@@ -288,7 +288,8 @@ Example response:
"state"
:
"active"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"last_activity_on"
:
"2021-01-27"
"last_activity_on"
:
"2021-01-27"
,
"membership_type"
:
"group_member"
},
},
{
{
"id"
:
2
,
"id"
:
2
,
...
@@ -298,7 +299,8 @@ Example response:
...
@@ -298,7 +299,8 @@ Example response:
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"email"
:
"john@example.com"
,
"email"
:
"john@example.com"
,
"last_activity_on"
:
"2021-01-25"
"last_activity_on"
:
"2021-01-25"
,
"membership_type"
:
"group_member"
},
},
{
{
"id"
:
3
,
"id"
:
3
,
...
@@ -307,7 +309,8 @@ Example response:
...
@@ -307,7 +309,8 @@ Example response:
"state"
:
"active"
,
"state"
:
"active"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"avatar_url"
:
"https://www.gravatar.com/avatar/c2525a7f58ae3776070e44c106c48e15?s=80&d=identicon"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"web_url"
:
"http://192.168.1.8:3000/root"
,
"last_activity_on"
:
"2021-01-20"
"last_activity_on"
:
"2021-01-20"
,
"membership_type"
:
"group_invite"
}
}
]
]
```
```
...
...
ee/app/finders/billed_users_finder.rb
View file @
dbbecb88
...
@@ -8,12 +8,18 @@ class BilledUsersFinder
...
@@ -8,12 +8,18 @@ class BilledUsersFinder
end
end
def
execute
def
execute
return
User
.
none
unless
group_billed_
user_ids
.
any?
return
{}
unless
user_ids
.
any?
users
=
::
User
.
id_in
(
group_billed_
user_ids
)
users
=
::
User
.
id_in
(
user_ids
)
users
=
users
.
search
(
search_term
)
if
search_term
users
=
users
.
search
(
search_term
)
if
search_term
users
.
sort_by_attribute
(
order_by
)
{
users:
users
.
sort_by_attribute
(
order_by
),
group_member_user_ids:
group_billed_user_ids
[
:group_member_user_ids
],
project_member_user_ids:
group_billed_user_ids
[
:project_member_user_ids
],
shared_group_user_ids:
group_billed_user_ids
[
:shared_group_user_ids
],
shared_project_user_ids:
group_billed_user_ids
[
:shared_project_user_ids
]
}
end
end
private
private
...
@@ -23,4 +29,8 @@ class BilledUsersFinder
...
@@ -23,4 +29,8 @@ class BilledUsersFinder
def
group_billed_user_ids
def
group_billed_user_ids
@group_billed_user_ids
||=
group
.
billed_user_ids
@group_billed_user_ids
||=
group
.
billed_user_ids
end
end
def
user_ids
group_billed_user_ids
[
:user_ids
]
end
end
end
ee/app/models/ee/group.rb
View file @
dbbecb88
...
@@ -295,31 +295,21 @@ module EE
...
@@ -295,31 +295,21 @@ module EE
override
:billable_members_count
override
:billable_members_count
def
billable_members_count
(
requested_hosted_plan
=
nil
)
def
billable_members_count
(
requested_hosted_plan
=
nil
)
billed_user_ids
(
requested_hosted_plan
).
count
billable_ids
=
billed_user_ids
(
requested_hosted_plan
)
billable_ids
[
:user_ids
].
count
end
end
# For now, we are not billing for members with a Guest role for subscriptions
# For now, we are not billing for members with a Guest role for subscriptions
# with a Gold/Ultimate plan. The other plans will treat Guest members as a regular member
# with a Gold/Ultimate plan. The other plans will treat Guest members as a regular member
# for billing purposes.
# for billing purposes.
#
#
#
W
e are plucking the user_ids from the "Members" table in an array and
#
For the user_ids key, w
e are plucking the user_ids from the "Members" table in an array and
# converting the array of user_ids to a Set which will have unique user_ids.
# converting the array of user_ids to a Set which will have unique user_ids.
def
billed_user_ids
(
requested_hosted_plan
=
nil
)
def
billed_user_ids
(
requested_hosted_plan
=
nil
)
if
([
actual_plan_name
,
requested_hosted_plan
]
&
[
::
Plan
::
GOLD
,
::
Plan
::
ULTIMATE
]).
any?
exclude_guests
=
([
actual_plan_name
,
requested_hosted_plan
]
&
[
::
Plan
::
GOLD
,
::
Plan
::
ULTIMATE
]).
any?
strong_memoize
(
:billed_user_ids
)
do
(
billed_group_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
+
exclude_guests
?
billed_user_ids_excluding_guests
:
billed_user_ids_including_guests
billed_project_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
+
billed_shared_non_guests_group_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
+
billed_invited_non_guests_group_to_project_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)).
to_set
end
else
strong_memoize
(
:non_billed_user_ids
)
do
(
billed_group_members
.
distinct
.
pluck
(
:user_id
)
+
billed_project_members
.
distinct
.
pluck
(
:user_id
)
+
billed_shared_group_members
.
distinct
.
pluck
(
:user_id
)
+
billed_invited_group_to_project_members
.
distinct
.
pluck
(
:user_id
)).
to_set
end
end
end
end
override
:supports_events?
override
:supports_events?
...
@@ -516,6 +506,40 @@ module EE
...
@@ -516,6 +506,40 @@ module EE
errors
.
add
(
:custom_project_templates_group_id
,
'has to be a subgroup of the group'
)
errors
.
add
(
:custom_project_templates_group_id
,
'has to be a subgroup of the group'
)
end
end
def
billed_user_ids_excluding_guests
strong_memoize
(
:billed_user_ids_excluding_guests
)
do
group_member_ids
=
billed_group_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
project_member_ids
=
billed_project_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
shared_group_ids
=
billed_shared_non_guests_group_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
shared_project_ids
=
billed_invited_non_guests_group_to_project_members
.
non_guests
.
distinct
.
pluck
(
:user_id
)
{
user_ids:
(
group_member_ids
+
project_member_ids
+
shared_group_ids
+
shared_project_ids
).
to_set
,
group_member_user_ids:
group_member_ids
.
to_set
,
project_member_user_ids:
project_member_ids
.
to_set
,
shared_group_user_ids:
shared_group_ids
.
to_set
,
shared_project_user_ids:
shared_project_ids
.
to_set
}
end
end
def
billed_user_ids_including_guests
strong_memoize
(
:billed_user_ids_including_guests
)
do
group_member_ids
=
billed_group_members
.
distinct
.
pluck
(
:user_id
)
project_member_ids
=
billed_project_members
.
distinct
.
pluck
(
:user_id
)
shared_group_ids
=
billed_shared_group_members
.
distinct
.
pluck
(
:user_id
)
shared_project_ids
=
billed_invited_group_to_project_members
.
distinct
.
pluck
(
:user_id
)
{
user_ids:
(
group_member_ids
+
project_member_ids
+
shared_group_ids
+
shared_project_ids
).
to_set
,
group_member_user_ids:
group_member_ids
.
to_set
,
project_member_user_ids:
project_member_ids
.
to_set
,
shared_group_user_ids:
shared_group_ids
.
to_set
,
shared_project_user_ids:
shared_project_ids
.
to_set
}
end
end
# Members belonging directly to Group or its subgroups
# Members belonging directly to Group or its subgroups
def
billed_group_members
def
billed_group_members
::
GroupMember
.
active_without_invites_and_requests
.
where
(
::
GroupMember
.
active_without_invites_and_requests
.
where
(
...
...
ee/app/models/ee/namespace.rb
View file @
dbbecb88
...
@@ -277,7 +277,13 @@ module EE
...
@@ -277,7 +277,13 @@ module EE
# This method is overwritten in Group where we made the calculation
# This method is overwritten in Group where we made the calculation
# for Group namespaces.
# for Group namespaces.
def
billed_user_ids
(
_requested_hosted_plan
=
nil
)
def
billed_user_ids
(
_requested_hosted_plan
=
nil
)
[
owner_id
]
{
user_ids:
[
owner_id
],
group_member_user_ids:
[],
project_member_user_ids:
[],
shared_group_user_ids:
[],
shared_project_user_ids:
[]
}
end
end
def
eligible_for_trial?
def
eligible_for_trial?
...
...
ee/app/models/ee/user.rb
View file @
dbbecb88
...
@@ -281,7 +281,7 @@ module EE
...
@@ -281,7 +281,7 @@ module EE
namespace
.
present?
&&
namespace
.
present?
&&
active?
&&
active?
&&
!
namespace
.
root_ancestor
.
free_plan?
&&
!
namespace
.
root_ancestor
.
free_plan?
&&
namespace
.
root_ancestor
.
billed_user_ids
.
include?
(
self
.
id
)
namespace
.
root_ancestor
.
billed_user_ids
[
:user_ids
]
.
include?
(
self
.
id
)
end
end
def
group_sso?
(
group
)
def
group_sso?
(
group
)
...
...
ee/changelogs/unreleased/vij-billable-member-source.yml
0 → 100644
View file @
dbbecb88
---
title
:
Return billable member relationship to a group in REST API
merge_request
:
60600
author
:
type
:
changed
ee/lib/ee/api/entities/billable_member.rb
View file @
dbbecb88
...
@@ -6,6 +6,20 @@ module EE
...
@@ -6,6 +6,20 @@ module EE
class
BillableMember
<
::
API
::
Entities
::
UserBasic
class
BillableMember
<
::
API
::
Entities
::
UserBasic
expose
:public_email
,
as: :email
expose
:public_email
,
as: :email
expose
:last_activity_on
expose
:last_activity_on
expose
:membership_type
private
def
membership_type
return
'group_member'
if
user_in_array?
(
:group_member_user_ids
)
return
'project_member'
if
user_in_array?
(
:project_member_user_ids
)
return
'group_invite'
if
user_in_array?
(
:shared_group_user_ids
)
return
'project_invite'
if
user_in_array?
(
:shared_project_user_ids
)
end
def
user_in_array?
(
name
)
options
.
fetch
(
name
,
[]).
include?
(
object
.
id
)
end
end
end
end
end
end
end
...
...
ee/lib/ee/api/members.rb
View file @
dbbecb88
...
@@ -69,13 +69,18 @@ module EE
...
@@ -69,13 +69,18 @@ module EE
bad_request!
(
nil
)
unless
::
Ability
.
allowed?
(
current_user
,
:admin_group_member
,
group
)
bad_request!
(
nil
)
unless
::
Ability
.
allowed?
(
current_user
,
:admin_group_member
,
group
)
sorting
=
params
[
:sort
]
||
'id_asc'
sorting
=
params
[
:sort
]
||
'id_asc'
users
=
paginate
(
BilledUsersFinder
.
new
(
group
,
result
=
BilledUsersFinder
.
new
(
group
,
search_term:
params
[
:search
],
search_term:
params
[
:search
],
order_by:
sorting
).
execute
order_by:
sorting
).
execute
)
present
users
,
with:
::
EE
::
API
::
Entities
::
BillableMember
,
current_user:
current_user
present
paginate
(
result
[
:users
]),
with:
::
EE
::
API
::
Entities
::
BillableMember
,
current_user:
current_user
,
group_member_user_ids:
result
[
:group_member_user_ids
],
project_member_user_ids:
result
[
:project_member_user_ids
],
shared_group_user_ids:
result
[
:shared_group_user_ids
],
shared_project_user_ids:
result
[
:shared_project_user_ids
]
end
end
desc
'Get the memberships of a billable user of a root group.'
do
desc
'Get the memberships of a billable user of a root group.'
do
...
@@ -93,7 +98,7 @@ module EE
...
@@ -93,7 +98,7 @@ module EE
user
=
::
User
.
find
(
params
[
:user_id
])
user
=
::
User
.
find
(
params
[
:user_id
])
not_found!
(
'User'
)
unless
group
.
billed_user_ids
.
include?
(
user
.
id
)
not_found!
(
'User'
)
unless
group
.
billed_user_ids
[
:user_ids
]
.
include?
(
user
.
id
)
memberships
=
user
.
members
.
in_hierarchy
(
group
).
including_source
memberships
=
user
.
members
.
in_hierarchy
(
group
).
including_source
...
...
ee/spec/finders/billed_users_finder_spec.rb
View file @
dbbecb88
...
@@ -3,7 +3,7 @@
...
@@ -3,7 +3,7 @@
require
'spec_helper'
require
'spec_helper'
RSpec
.
describe
BilledUsersFinder
do
RSpec
.
describe
BilledUsersFinder
do
let_it_be
(
:group
)
{
create
(
:group
)
}
let_it_be
_with_refind
(
:group
)
{
create
(
:group
)
}
let
(
:search_term
)
{
nil
}
let
(
:search_term
)
{
nil
}
let
(
:order_by
)
{
nil
}
let
(
:order_by
)
{
nil
}
...
@@ -17,10 +17,10 @@ RSpec.describe BilledUsersFinder do
...
@@ -17,10 +17,10 @@ RSpec.describe BilledUsersFinder do
subject
{
described_class
.
new
(
group
,
search_term:
search_term
,
order_by:
order_by
)
}
subject
{
described_class
.
new
(
group
,
search_term:
search_term
,
order_by:
order_by
)
}
context
'when a group does not have any billed users'
do
context
'when a group does not have any billed users'
do
it
'returns an empty
array
'
do
it
'returns an empty
object
'
do
allow
(
group
).
to
receive
(
:billed_user_ids
).
and_return
(
[]
)
allow
(
group
).
to
receive
(
:billed_user_ids
).
and_return
(
{
user_ids:
[]
}
)
expect
(
subject
.
execute
).
to
be_empty
expect
(
subject
.
execute
).
to
eq
({})
end
end
end
end
...
@@ -31,7 +31,7 @@ RSpec.describe BilledUsersFinder do
...
@@ -31,7 +31,7 @@ RSpec.describe BilledUsersFinder do
let
(
:order_by
)
{
'name_desc'
}
let
(
:order_by
)
{
'name_desc'
}
it
'sorts results accordingly'
do
it
'sorts results accordingly'
do
expect
(
subject
.
execute
).
to
eq
([
john_smith
,
john_doe
].
map
(
&
:user
))
expect
(
subject
.
execute
[
:users
]
).
to
eq
([
john_smith
,
john_doe
].
map
(
&
:user
))
end
end
end
end
...
@@ -39,7 +39,7 @@ RSpec.describe BilledUsersFinder do
...
@@ -39,7 +39,7 @@ RSpec.describe BilledUsersFinder do
subject
{
described_class
.
new
(
group
,
search_term:
search_term
)
}
subject
{
described_class
.
new
(
group
,
search_term:
search_term
)
}
it
'sorts expected results in name_asc order'
do
it
'sorts expected results in name_asc order'
do
expect
(
subject
.
execute
).
to
eq
([
john_doe
,
john_smith
].
map
(
&
:user
))
expect
(
subject
.
execute
[
:users
]
).
to
eq
([
john_doe
,
john_smith
].
map
(
&
:user
))
end
end
end
end
end
end
...
@@ -50,7 +50,7 @@ RSpec.describe BilledUsersFinder do
...
@@ -50,7 +50,7 @@ RSpec.describe BilledUsersFinder do
it
'returns expected users in name asc order when a sorting is not provided either'
do
it
'returns expected users in name asc order when a sorting is not provided either'
do
allow
(
group
).
to
receive
(
:billed_user_members
).
and_return
([
john_doe
,
john_smith
,
sophie
,
maria
])
allow
(
group
).
to
receive
(
:billed_user_members
).
and_return
([
john_doe
,
john_smith
,
sophie
,
maria
])
expect
(
subject
.
execute
).
to
eq
([
john_doe
,
john_smith
,
maria
,
sophie
].
map
(
&
:user
))
expect
(
subject
.
execute
[
:users
]
).
to
eq
([
john_doe
,
john_smith
,
maria
,
sophie
].
map
(
&
:user
))
end
end
context
'and when a sorting parameter is provided (eg name descending)'
do
context
'and when a sorting parameter is provided (eg name descending)'
do
...
@@ -59,7 +59,38 @@ RSpec.describe BilledUsersFinder do
...
@@ -59,7 +59,38 @@ RSpec.describe BilledUsersFinder do
subject
{
described_class
.
new
(
group
,
search_term:
search_term
,
order_by:
order_by
)
}
subject
{
described_class
.
new
(
group
,
search_term:
search_term
,
order_by:
order_by
)
}
it
'sorts results accordingly'
do
it
'sorts results accordingly'
do
expect
(
subject
.
execute
).
to
eq
([
sophie
,
maria
,
john_smith
,
john_doe
].
map
(
&
:user
))
expect
(
subject
.
execute
[
:users
]).
to
eq
([
sophie
,
maria
,
john_smith
,
john_doe
].
map
(
&
:user
))
end
end
end
context
'with billable group members including shared members'
do
let_it_be
(
:shared_with_group_member
)
{
create
(
:group_member
,
user:
create
(
:user
,
name:
'Shared Group User'
))
}
let_it_be
(
:shared_with_project_member
)
{
create
(
:group_member
,
user:
create
(
:user
,
name:
'Shared Project User'
))
}
let_it_be
(
:project
)
{
create
(
:project
,
group:
group
)
}
before
do
create
(
:group_group_link
,
shared_group:
group
,
shared_with_group:
shared_with_group_member
.
group
)
create
(
:project_group_link
,
group:
shared_with_project_member
.
group
,
project:
project
)
end
it
'returns a hash of users and user ids'
do
expect
(
subject
.
execute
.
keys
).
to
eq
([
:users
,
:group_member_user_ids
,
:project_member_user_ids
,
:shared_group_user_ids
,
:shared_project_user_ids
])
end
it
'returns the correct user ids'
do
result
=
subject
.
execute
aggregate_failures
do
expect
(
result
[
:group_member_user_ids
]).
to
contain_exactly
(
*
[
maria
,
john_smith
,
john_doe
,
sophie
].
map
(
&
:user_id
))
expect
(
result
[
:shared_group_user_ids
]).
to
contain_exactly
(
shared_with_group_member
.
user_id
)
expect
(
result
[
:shared_project_user_ids
]).
to
contain_exactly
(
shared_with_project_member
.
user_id
)
end
end
end
end
end
end
...
...
ee/spec/lib/ee/api/entities/billable_member_spec.rb
View file @
dbbecb88
...
@@ -5,9 +5,17 @@ require 'spec_helper'
...
@@ -5,9 +5,17 @@ require 'spec_helper'
RSpec
.
describe
::
EE
::
API
::
Entities
::
BillableMember
do
RSpec
.
describe
::
EE
::
API
::
Entities
::
BillableMember
do
let
(
:last_activity_on
)
{
Date
.
today
}
let
(
:last_activity_on
)
{
Date
.
today
}
let
(
:public_email
)
{
nil
}
let
(
:public_email
)
{
nil
}
let
(
:member
)
{
build
(
:user
,
public_email:
public_email
,
email:
'private@email.com'
,
last_activity_on:
last_activity_on
)
}
let
(
:member
)
{
build
(
:user
,
id:
non_existing_record_id
,
public_email:
public_email
,
email:
'private@email.com'
,
last_activity_on:
last_activity_on
)
}
let
(
:options
)
do
{
group_member_user_ids:
[],
project_member_user_ids:
[],
shared_group_user_ids:
[],
shared_project_user_ids:
[]
}
end
subject
(
:entity_representation
)
{
described_class
.
new
(
member
).
as_json
}
subject
(
:entity_representation
)
{
described_class
.
new
(
member
,
options
).
as_json
}
it
'returns the last_activity_on attribute'
do
it
'returns the last_activity_on attribute'
do
expect
(
entity_representation
[
:last_activity_on
]).
to
eq
last_activity_on
expect
(
entity_representation
[
:last_activity_on
]).
to
eq
last_activity_on
...
@@ -35,4 +43,34 @@ RSpec.describe ::EE::API::Entities::BillableMember do
...
@@ -35,4 +43,34 @@ RSpec.describe ::EE::API::Entities::BillableMember do
end
end
end
end
end
end
context
'with different group membership types'
do
using
RSpec
::
Parameterized
::
TableSyntax
where
(
:user_ids
,
:expected_value
)
do
:group_member_user_ids
|
'group_member'
:project_member_user_ids
|
'project_member'
:shared_group_user_ids
|
'group_invite'
:shared_project_user_ids
|
'project_invite'
end
with_them
do
let
(
:options
)
{
super
().
merge
(
user_ids
=>
[
member
.
id
])
}
it
'returns the expected value'
do
expect
(
entity_representation
[
:membership_type
]).
to
eq
expected_value
end
end
context
'with a missing membership type'
do
before
do
options
.
delete
(
:group_member_user_ids
)
end
it
'does not raise an error'
do
expect
(
options
[
:group_member_user_ids
]).
to
be_nil
expect
{
entity_representation
}.
not_to
raise_error
end
end
end
end
end
ee/spec/models/ee/namespace_spec.rb
View file @
dbbecb88
This diff is collapsed.
Click to expand it.
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