Skip to content

Commit

Permalink
ZOOKEEPER-4219: Quota checks break setData in multi transactions (#108)
Browse files Browse the repository at this point in the history
Without this patch, a multi() transaction such as the one implemented
in ZooKeeperQuotaTest.testMultiCreateThenSetDataShouldWork fails with
MarshallingError when 'enforceQuota' is enabled.

This happens whenever the node has an associated quota, whether it was
exceeded or not.

This is due to the server encountering null while trying to access a
database node by path--whereas that node only exists as a ChangeRecord
in the server's 'outstandingChanges' list:

    java.lang.NullPointerException
        at org.apache.zookeeper.server.ZooKeeperServer.checkQuota(ZooKeeperServer.java:2048)
        at org.apache.zookeeper.server.PrepRequestProcessor.pRequest2Txn(PrepRequestProcessor.java:397)

The patch adds an additional 'lastData' parameter to the quota
checking function, and passes the data from the ChangeRecord during
'setData' operations.

Author: Damien Diederen <[email protected]>

Reviewers: Enrico Olivelli <[email protected]>, Mate Szalay-Beko <[email protected]>, Norbert Kalmar <[email protected]>

Closes apache#1611 from ztzg/ZOOKEEPER-4219-quota-multi-setdata

Co-authored-by: Damien Diederen <[email protected]>
  • Loading branch information
anuragmadnawat1 and ztzg authored Nov 2, 2022
1 parent 52ab212 commit fce79f4
Show file tree
Hide file tree
Showing 3 changed files with 59 additions and 12 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -394,7 +394,7 @@ protected void pRequest2Txn(int type, long zxid, Request request, Record record,
validatePath(path, request.sessionId);
nodeRecord = getRecordForPath(path);
zks.checkACL(request.cnxn, nodeRecord.acl, ZooDefs.Perms.WRITE, request.authInfo, path, null);
zks.checkQuota(path, setDataRequest.getData(), OpCode.setData);
zks.checkQuota(path, nodeRecord.data, setDataRequest.getData(), OpCode.setData);
int newVersion = checkAndIncVersion(nodeRecord.stat.getVersion(), setDataRequest.getVersion(), path);
request.setTxn(new SetDataTxn(path, setDataRequest.getData(), newVersion));
nodeRecord = nodeRecord.duplicate(request.getHdr().getZxid());
Expand Down Expand Up @@ -698,7 +698,7 @@ private void pRequest2TxnCreate(int type, Request request, Record record, boolea
throw new KeeperException.NoChildrenForEphemeralsException(path);
}
int newCversion = parentRecord.stat.getCversion() + 1;
zks.checkQuota(path, data, OpCode.create);
zks.checkQuota(path, null, data, OpCode.create);
if (type == OpCode.createContainer) {
request.setTxn(new CreateContainerTxn(path, data, listACL, newCversion));
} else if (type == OpCode.createTTL) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2020,14 +2020,15 @@ public void checkACL(ServerCnxn cnxn, List<ACL> acl, int perm, List<Id> ids, Str
* check a path whether exceeded the quota.
*
* @param path
* the path of the node
* the path of the node, used for the quota prefix check
* @param lastData
* the current node data, {@code null} for none
* @param data
* the data of the path
* the data to be set, or {@code null} for none
* @param type
* currently, create and setData need to check quota
*/

public void checkQuota(String path, byte[] data, int type) throws KeeperException.QuotaExceededException {
public void checkQuota(String path, byte[] lastData, byte[] data, int type) throws KeeperException.QuotaExceededException {
if (!enforceQuota) {
return;
}
Expand All @@ -2043,11 +2044,6 @@ public void checkQuota(String path, byte[] data, int type) throws KeeperExceptio
checkQuota(lastPrefix, dataBytes, 1);
break;
case OpCode.setData:
DataNode node = zkDatabase.getDataTree().getNode(path);
byte[] lastData;
synchronized (node) {
lastData = node.getData();
}
checkQuota(lastPrefix, dataBytes - (lastData == null ? 0 : lastData.length), 0);
break;
default:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,17 @@

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.List;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.KeeperException.QuotaExceededException;
import org.apache.zookeeper.Op;
import org.apache.zookeeper.Quotas;
import org.apache.zookeeper.StatsTrack;
import org.apache.zookeeper.ZooDefs.Ids;
Expand Down Expand Up @@ -396,6 +400,53 @@ public void testSetQuotaWhenExceedBothBytesAndCountHardQuota() throws Exception
}
}

@Test
public void testMultiCreateThenSetDataShouldWork() throws Exception {
final String path = "/a";
final String subPath = "/a/b";

zk.create(path, null, Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);

final byte[] data13b = "Hello, World!".getBytes(StandardCharsets.UTF_8);

final StatsTrack st = new StatsTrack();
st.setByteHardLimit(data13b.length);
SetQuotaCommand.createQuota(zk, path, st);

final List<Op> ops = Arrays.asList(
Op.create(subPath, null, Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT),
Op.setData(subPath, data13b, -1));

zk.multi(ops);
}

@Test
public void testMultiCreateThenSetDataShouldFail() throws Exception {
final String path = "/a";
final String subPath = "/a/b";

zk.create(path, null, Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);

final byte[] data13b = "Hello, World!".getBytes(StandardCharsets.UTF_8);

final StatsTrack st = new StatsTrack();
st.setByteHardLimit(data13b.length - 1);
SetQuotaCommand.createQuota(zk, path, st);

final List<Op> ops = Arrays.asList(
Op.create(subPath, null, Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT),
Op.setData(subPath, data13b, -1));

try {
zk.multi(ops);
fail("should fail transaction when hard quota is exceeded");
} catch (QuotaExceededException e) {
//expected
}

assertNull(zk.exists(subPath, null));
}

@Test
public void testDeleteBytesQuota() throws Exception {

Expand Down Expand Up @@ -501,4 +552,4 @@ public void testListQuota() throws Exception {
}
}
}
}
}

0 comments on commit fce79f4

Please sign in to comment.