Support removing of groups from a group via REST

A group can be removed from a group by DELETE on
'/groups/*/groups/<group>'.

In addition multiple groups can be removed from a group by
POST on '/groups/*/groups.delete'.

This is consistent with how members can be removed from a group.

The WebUI was adapted to use the new REST API for removing groups
from a group.

Change-Id: I924611b6c413b65d666166758b7d446b782379c7
Signed-off-by: Edwin Kempin <edwin.kempin@sap.com>
diff --git a/gerrit-gwtui/src/main/java/com/google/gerrit/client/admin/AccountGroupMembersScreen.java b/gerrit-gwtui/src/main/java/com/google/gerrit/client/admin/AccountGroupMembersScreen.java
index e4086d2..bb8f5db 100644
--- a/gerrit-gwtui/src/main/java/com/google/gerrit/client/admin/AccountGroupMembersScreen.java
+++ b/gerrit-gwtui/src/main/java/com/google/gerrit/client/admin/AccountGroupMembersScreen.java
@@ -16,6 +16,7 @@
 
 import com.google.gerrit.client.Dispatcher;
 import com.google.gerrit.client.Gerrit;
+import com.google.gerrit.client.VoidResult;
 import com.google.gerrit.client.groups.GroupApi;
 import com.google.gerrit.client.groups.MemberInfo;
 import com.google.gerrit.client.rpc.GerritCallback;
@@ -41,7 +42,6 @@
 import com.google.gwt.user.client.ui.FlexTable.FlexCellFormatter;
 import com.google.gwt.user.client.ui.FlowPanel;
 import com.google.gwt.user.client.ui.Panel;
-import com.google.gwtjsonrpc.common.VoidResult;
 
 import java.util.HashSet;
 import java.util.List;
@@ -252,8 +252,8 @@
       }
       if (!ids.isEmpty()) {
         GroupApi.removeMembers(getGroupUUID(), ids,
-            new GerritCallback<com.google.gerrit.client.VoidResult>() {
-              public void onSuccess(final com.google.gerrit.client.VoidResult result) {
+            new GerritCallback<VoidResult>() {
+              public void onSuccess(final VoidResult result) {
                 for (int row = 1; row < table.getRowCount();) {
                   final AccountGroupMember k = getRowItem(row);
                   if (k != null && ids.contains(k.getAccountId())) {
@@ -329,7 +329,7 @@
         }
       }
       if (!keys.isEmpty()) {
-        Util.GROUP_SVC.deleteGroupIncludes(getGroupId(), keys,
+        GroupApi.removeIncludedGroups(getGroupUUID(), keys,
             new GerritCallback<VoidResult>() {
               public void onSuccess(final VoidResult result) {
                 for (int row = 1; row < table.getRowCount();) {
diff --git a/gerrit-gwtui/src/main/java/com/google/gerrit/client/groups/GroupApi.java b/gerrit-gwtui/src/main/java/com/google/gerrit/client/groups/GroupApi.java
index d6a5dfb..c4e276a 100644
--- a/gerrit-gwtui/src/main/java/com/google/gerrit/client/groups/GroupApi.java
+++ b/gerrit-gwtui/src/main/java/com/google/gerrit/client/groups/GroupApi.java
@@ -19,6 +19,7 @@
 import com.google.gerrit.client.rpc.RestApi;
 import com.google.gerrit.reviewdb.client.Account;
 import com.google.gerrit.reviewdb.client.AccountGroup;
+import com.google.gerrit.reviewdb.client.AccountGroupIncludeByUuid;
 import com.google.gwt.core.client.JavaScriptObject;
 import com.google.gwt.user.client.rpc.AsyncCallback;
 
@@ -129,6 +130,21 @@
     }
   }
 
+  /** Remove included groups from a group. */
+  public static void removeIncludedGroups(AccountGroup.UUID group,
+      Set<AccountGroupIncludeByUuid.Key> ids, final AsyncCallback<VoidResult> cb) {
+    if (ids.size() == 1) {
+      AccountGroupIncludeByUuid.Key g = ids.iterator().next();
+      groups(group).id(g.getIncludeUUID().get()).delete(cb);
+    } else {
+      IncludedGroupInput in = IncludedGroupInput.create();
+      for (AccountGroupIncludeByUuid.Key g : ids) {
+        in.add_group(g.getIncludeUUID().get());
+      }
+      group(group).view("groups.delete").post(in, cb);
+    }
+  }
+
   private static RestApi members(AccountGroup.UUID group) {
     return group(group).view("members");
   }
diff --git a/gerrit-server/src/main/java/com/google/gerrit/server/group/DeleteIncludedGroups.java b/gerrit-server/src/main/java/com/google/gerrit/server/group/DeleteIncludedGroups.java
new file mode 100644
index 0000000..c91db96
--- /dev/null
+++ b/gerrit-server/src/main/java/com/google/gerrit/server/group/DeleteIncludedGroups.java
@@ -0,0 +1,163 @@
+// 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.collect.ImmutableList;
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.gerrit.common.errors.NoSuchGroupException;
+import com.google.gerrit.extensions.restapi.AuthException;
+import com.google.gerrit.extensions.restapi.BadRequestException;
+import com.google.gerrit.extensions.restapi.MethodNotAllowedException;
+import com.google.gerrit.extensions.restapi.ResourceNotFoundException;
+import com.google.gerrit.extensions.restapi.Response;
+import com.google.gerrit.extensions.restapi.RestModifyView;
+import com.google.gerrit.reviewdb.client.Account;
+import com.google.gerrit.reviewdb.client.AccountGroup;
+import com.google.gerrit.reviewdb.client.AccountGroupIncludeByUuid;
+import com.google.gerrit.reviewdb.client.AccountGroupIncludeByUuidAudit;
+import com.google.gerrit.reviewdb.server.ReviewDb;
+import com.google.gerrit.server.BadRequestHandler;
+import com.google.gerrit.server.CurrentUser;
+import com.google.gerrit.server.IdentifiedUser;
+import com.google.gerrit.server.account.GroupControl;
+import com.google.gerrit.server.account.GroupIncludeCache;
+import com.google.gerrit.server.group.AddIncludedGroups.Input;
+import com.google.gwtorm.server.OrmException;
+import com.google.inject.Inject;
+import com.google.inject.Provider;
+
+import java.util.List;
+import java.util.Map;
+
+public class DeleteIncludedGroups implements RestModifyView<GroupResource, Input> {
+  private final Provider<GroupsCollection> groupsCollection;
+  private final GroupIncludeCache groupIncludeCache;
+  private final ReviewDb db;
+  private final Provider<CurrentUser> self;
+
+  @Inject
+  DeleteIncludedGroups(Provider<GroupsCollection> groupsCollection,
+      GroupIncludeCache groupIncludeCache, ReviewDb db,
+      Provider<CurrentUser> self) {
+    this.groupsCollection = groupsCollection;
+    this.groupIncludeCache = groupIncludeCache;
+    this.db = db;
+    this.self = self;
+  }
+
+  @Override
+  public Object apply(GroupResource resource, Input input)
+      throws MethodNotAllowedException, AuthException, BadRequestException,
+      OrmException {
+    AccountGroup internalGroup = resource.toAccountGroup();
+    if (internalGroup == null) {
+      throw new MethodNotAllowedException();
+    }
+    input = Input.init(input);
+
+    final GroupControl control = resource.getControl();
+    final Map<AccountGroup.UUID, AccountGroupIncludeByUuid> includedGroups = getIncludedGroups(internalGroup.getId());
+    final List<AccountGroupIncludeByUuid> toRemove = Lists.newLinkedList();
+    final BadRequestHandler badRequest = new BadRequestHandler("removing included groups");
+
+    for (final String includedGroup : input.groups) {
+      try {
+        final GroupResource includedGroupResource = groupsCollection.get().parse(includedGroup);
+
+        if (!control.canRemoveGroup(includedGroupResource.getGroupUUID())) {
+          throw new AuthException(String.format("Cannot delete group: %s",
+              includedGroupResource.getName()));
+        }
+
+        final AccountGroupIncludeByUuid g =
+            includedGroups.remove(includedGroupResource.getGroupUUID());
+        if (g != null) {
+          toRemove.add(g);
+        }
+      } catch (ResourceNotFoundException e) {
+        badRequest.addError(new NoSuchGroupException(includedGroup));
+      }
+    }
+
+    badRequest.failOnError();
+
+    if (!toRemove.isEmpty()) {
+      writeAudits(toRemove);
+      db.accountGroupIncludesByUuid().delete(toRemove);
+      for (final AccountGroupIncludeByUuid g : toRemove) {
+        groupIncludeCache.evictMemberIn(g.getIncludeUUID());
+      }
+      groupIncludeCache.evictMembersOf(internalGroup.getGroupUUID());
+    }
+
+    return Response.none();
+  }
+
+  private Map<AccountGroup.UUID, AccountGroupIncludeByUuid> getIncludedGroups(
+      final AccountGroup.Id groupId) throws OrmException {
+    final Map<AccountGroup.UUID, AccountGroupIncludeByUuid> groups =
+        Maps.newHashMap();
+    for (final AccountGroupIncludeByUuid g : db.accountGroupIncludesByUuid().byGroup(groupId)) {
+      groups.put(g.getIncludeUUID(), g);
+    }
+    return groups;
+  }
+
+  private void writeAudits(final List<AccountGroupIncludeByUuid> toBeRemoved)
+      throws OrmException {
+    final Account.Id me = ((IdentifiedUser) self.get()).getAccountId();
+    final List<AccountGroupIncludeByUuidAudit> auditUpdates = Lists.newLinkedList();
+    for (final AccountGroupIncludeByUuid g : toBeRemoved) {
+      AccountGroupIncludeByUuidAudit audit = null;
+      for (AccountGroupIncludeByUuidAudit a : db
+          .accountGroupIncludesByUuidAudit().byGroupInclude(g.getGroupId(),
+              g.getIncludeUUID())) {
+        if (a.isActive()) {
+          audit = a;
+          break;
+        }
+      }
+
+      if (audit != null) {
+        audit.removed(me);
+        auditUpdates.add(audit);
+      }
+    }
+    db.accountGroupIncludesByUuidAudit().update(auditUpdates);
+  }
+
+  static class DeleteIncludedGroup implements
+      RestModifyView<IncludedGroupResource, DeleteIncludedGroup.Input> {
+    static class Input {
+    }
+
+    private final Provider<DeleteIncludedGroups> delete;
+
+    @Inject
+    DeleteIncludedGroup(final Provider<DeleteIncludedGroups> delete) {
+      this.delete = delete;
+    }
+
+    @Override
+    public Object apply(IncludedGroupResource resource, Input input)
+        throws MethodNotAllowedException, AuthException, BadRequestException,
+        OrmException {
+      AddIncludedGroups.Input in = new AddIncludedGroups.Input();
+      in.groups = ImmutableList.of(resource.getMember().toString());
+      return delete.get().apply(resource, in);
+    }
+  }
+}
diff --git a/gerrit-server/src/main/java/com/google/gerrit/server/group/Module.java b/gerrit-server/src/main/java/com/google/gerrit/server/group/Module.java
index 4bc3177..1cc11c4 100644
--- a/gerrit-server/src/main/java/com/google/gerrit/server/group/Module.java
+++ b/gerrit-server/src/main/java/com/google/gerrit/server/group/Module.java
@@ -22,6 +22,7 @@
 import com.google.gerrit.extensions.restapi.RestApiModule;
 import com.google.gerrit.server.group.AddIncludedGroups.UpdateIncludedGroup;
 import com.google.gerrit.server.group.AddMembers.UpdateMember;
+import com.google.gerrit.server.group.DeleteIncludedGroups.DeleteIncludedGroup;
 import com.google.gerrit.server.group.DeleteMembers.DeleteMember;
 import com.google.inject.assistedinject.FactoryModuleBuilder;
 
@@ -40,6 +41,7 @@
     post(GROUP_KIND, "members.delete").to(DeleteMembers.class);
     post(GROUP_KIND, "groups").to(AddIncludedGroups.class);
     post(GROUP_KIND, "groups.add").to(AddIncludedGroups.class);
+    post(GROUP_KIND, "groups.delete").to(DeleteIncludedGroups.class);
     get(GROUP_KIND, "description").to(GetDescription.class);
     put(GROUP_KIND, "description").to(PutDescription.class);
     delete(GROUP_KIND, "description").to(PutDescription.class);
@@ -52,6 +54,7 @@
     child(GROUP_KIND, "groups").to(IncludedGroupsCollection.class);
     get(INCLUDED_GROUP_KIND).to(GetIncludedGroup.class);
     put(INCLUDED_GROUP_KIND).to(UpdateIncludedGroup.class);
+    delete(INCLUDED_GROUP_KIND).to(DeleteIncludedGroup.class);
 
     install(new FactoryModuleBuilder().build(CreateGroup.Factory.class));
   }