| // Copyright (C) 2013 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package com.google.gerrit.server.group; |
| |
| import com.google.common.base.MoreObjects; |
| import com.google.common.base.Strings; |
| import com.google.gerrit.common.data.GlobalCapability; |
| import com.google.gerrit.common.data.GroupDescription; |
| import com.google.gerrit.common.data.GroupDescriptions; |
| import com.google.gerrit.extensions.annotations.RequiresCapability; |
| import com.google.gerrit.extensions.api.groups.GroupInput; |
| import com.google.gerrit.extensions.client.ListGroupsOption; |
| import com.google.gerrit.extensions.common.GroupInfo; |
| import com.google.gerrit.extensions.registration.DynamicSet; |
| import com.google.gerrit.extensions.restapi.AuthException; |
| import com.google.gerrit.extensions.restapi.BadRequestException; |
| import com.google.gerrit.extensions.restapi.ResourceConflictException; |
| import com.google.gerrit.extensions.restapi.RestModifyView; |
| import com.google.gerrit.extensions.restapi.TopLevelResource; |
| import com.google.gerrit.extensions.restapi.UnprocessableEntityException; |
| import com.google.gerrit.extensions.restapi.Url; |
| import com.google.gerrit.reviewdb.client.Account; |
| import com.google.gerrit.reviewdb.client.AccountGroup; |
| import com.google.gerrit.reviewdb.client.AccountGroupName; |
| import com.google.gerrit.reviewdb.server.ReviewDb; |
| import com.google.gerrit.server.GerritPersonIdent; |
| import com.google.gerrit.server.IdentifiedUser; |
| import com.google.gerrit.server.account.CreateGroupArgs; |
| import com.google.gerrit.server.account.GroupCache; |
| import com.google.gerrit.server.account.GroupUUID; |
| import com.google.gerrit.server.config.GerritServerConfig; |
| import com.google.gerrit.server.validators.GroupCreationValidationListener; |
| import com.google.gerrit.server.validators.ValidationException; |
| import com.google.gwtorm.server.OrmDuplicateKeyException; |
| import com.google.gwtorm.server.OrmException; |
| import com.google.inject.Inject; |
| import com.google.inject.Provider; |
| import com.google.inject.assistedinject.Assisted; |
| import java.io.IOException; |
| import java.util.ArrayList; |
| import java.util.Collection; |
| import java.util.Collections; |
| import java.util.List; |
| import java.util.Locale; |
| import org.eclipse.jgit.lib.Config; |
| import org.eclipse.jgit.lib.PersonIdent; |
| |
| @RequiresCapability(GlobalCapability.CREATE_GROUP) |
| public class CreateGroup implements RestModifyView<TopLevelResource, GroupInput> { |
| public interface Factory { |
| CreateGroup create(@Assisted String name); |
| } |
| |
| private final Provider<IdentifiedUser> self; |
| private final PersonIdent serverIdent; |
| private final ReviewDb db; |
| private final GroupCache groupCache; |
| private final GroupsCollection groups; |
| private final GroupJson json; |
| private final DynamicSet<GroupCreationValidationListener> groupCreationValidationListeners; |
| private final AddMembers addMembers; |
| private final SystemGroupBackend systemGroupBackend; |
| private final boolean defaultVisibleToAll; |
| private final String name; |
| |
| @Inject |
| CreateGroup( |
| Provider<IdentifiedUser> self, |
| @GerritPersonIdent PersonIdent serverIdent, |
| ReviewDb db, |
| GroupCache groupCache, |
| GroupsCollection groups, |
| GroupJson json, |
| DynamicSet<GroupCreationValidationListener> groupCreationValidationListeners, |
| AddMembers addMembers, |
| SystemGroupBackend systemGroupBackend, |
| @GerritServerConfig Config cfg, |
| @Assisted String name) { |
| this.self = self; |
| this.serverIdent = serverIdent; |
| this.db = db; |
| this.groupCache = groupCache; |
| this.groups = groups; |
| this.json = json; |
| this.groupCreationValidationListeners = groupCreationValidationListeners; |
| this.addMembers = addMembers; |
| this.systemGroupBackend = systemGroupBackend; |
| this.defaultVisibleToAll = cfg.getBoolean("groups", "newGroupsVisibleToAll", false); |
| this.name = name; |
| } |
| |
| public CreateGroup addOption(ListGroupsOption o) { |
| json.addOption(o); |
| return this; |
| } |
| |
| public CreateGroup addOptions(Collection<ListGroupsOption> o) { |
| json.addOptions(o); |
| return this; |
| } |
| |
| @Override |
| public GroupInfo apply(TopLevelResource resource, GroupInput input) |
| throws AuthException, BadRequestException, UnprocessableEntityException, |
| ResourceConflictException, OrmException, IOException { |
| if (input == null) { |
| input = new GroupInput(); |
| } |
| if (input.name != null && !name.equals(input.name)) { |
| throw new BadRequestException("name must match URL"); |
| } |
| |
| AccountGroup.Id ownerId = owner(input); |
| CreateGroupArgs args = new CreateGroupArgs(); |
| args.setGroupName(name); |
| args.groupDescription = Strings.emptyToNull(input.description); |
| args.visibleToAll = MoreObjects.firstNonNull(input.visibleToAll, defaultVisibleToAll); |
| args.ownerGroupId = ownerId; |
| if (input.members != null && !input.members.isEmpty()) { |
| List<Account.Id> members = new ArrayList<>(); |
| for (String nameOrEmailOrId : input.members) { |
| Account a = addMembers.findAccount(nameOrEmailOrId); |
| if (!a.isActive()) { |
| throw new UnprocessableEntityException( |
| String.format("Account Inactive: %s", nameOrEmailOrId)); |
| } |
| members.add(a.getId()); |
| } |
| args.initialMembers = members; |
| } else { |
| args.initialMembers = |
| ownerId == null |
| ? Collections.singleton(self.get().getAccountId()) |
| : Collections.<Account.Id>emptySet(); |
| } |
| |
| for (GroupCreationValidationListener l : groupCreationValidationListeners) { |
| try { |
| l.validateNewGroup(args); |
| } catch (ValidationException e) { |
| throw new ResourceConflictException(e.getMessage(), e); |
| } |
| } |
| |
| return json.format(GroupDescriptions.forAccountGroup(createGroup(args))); |
| } |
| |
| private AccountGroup.Id owner(GroupInput input) throws UnprocessableEntityException { |
| if (input.ownerId != null) { |
| GroupDescription.Basic d = groups.parseInternal(Url.decode(input.ownerId)); |
| return GroupDescriptions.toAccountGroup(d).getId(); |
| } |
| return null; |
| } |
| |
| private AccountGroup createGroup(CreateGroupArgs createGroupArgs) |
| throws OrmException, ResourceConflictException, IOException { |
| |
| // Do not allow creating groups with the same name as system groups |
| for (String name : systemGroupBackend.getNames()) { |
| if (name.toLowerCase(Locale.US) |
| .equals(createGroupArgs.getGroupName().toLowerCase(Locale.US))) { |
| throw new ResourceConflictException("group '" + name + "' already exists"); |
| } |
| } |
| |
| for (String name : systemGroupBackend.getReservedNames()) { |
| if (name.toLowerCase(Locale.US) |
| .equals(createGroupArgs.getGroupName().toLowerCase(Locale.US))) { |
| throw new ResourceConflictException("group name '" + name + "' is reserved"); |
| } |
| } |
| |
| AccountGroup.Id groupId = new AccountGroup.Id(db.nextAccountGroupId()); |
| AccountGroup.UUID uuid = |
| GroupUUID.make( |
| createGroupArgs.getGroupName(), |
| self.get().newCommitterIdent(serverIdent.getWhen(), serverIdent.getTimeZone())); |
| AccountGroup group = new AccountGroup(createGroupArgs.getGroup(), groupId, uuid); |
| group.setVisibleToAll(createGroupArgs.visibleToAll); |
| if (createGroupArgs.ownerGroupId != null) { |
| AccountGroup ownerGroup = groupCache.get(createGroupArgs.ownerGroupId); |
| if (ownerGroup != null) { |
| group.setOwnerGroupUUID(ownerGroup.getGroupUUID()); |
| } |
| } |
| if (createGroupArgs.groupDescription != null) { |
| group.setDescription(createGroupArgs.groupDescription); |
| } |
| AccountGroupName gn = new AccountGroupName(group); |
| // first insert the group name to validate that the group name hasn't |
| // already been used to create another group |
| try { |
| db.accountGroupNames().insert(Collections.singleton(gn)); |
| } catch (OrmDuplicateKeyException e) { |
| throw new ResourceConflictException( |
| "group '" + createGroupArgs.getGroupName() + "' already exists"); |
| } |
| db.accountGroups().insert(Collections.singleton(group)); |
| |
| addMembers.addMembers(groupId, createGroupArgs.initialMembers); |
| |
| groupCache.onCreateGroup(createGroupArgs.getGroup()); |
| |
| return group; |
| } |
| } |