CreateTTLTest.java
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.zookeeper.server;
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.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicLong;
import org.apache.zookeeper.AsyncCallback;
import org.apache.zookeeper.CreateMode;
import org.apache.zookeeper.CreateOptions;
import org.apache.zookeeper.KeeperException;
import org.apache.zookeeper.KeeperException.Code;
import org.apache.zookeeper.Op;
import org.apache.zookeeper.OpResult;
import org.apache.zookeeper.TestableZooKeeper;
import org.apache.zookeeper.ZooDefs;
import org.apache.zookeeper.data.Stat;
import org.apache.zookeeper.metrics.MetricsUtils;
import org.apache.zookeeper.proto.CreateResponse;
import org.apache.zookeeper.proto.CreateTTLRequest;
import org.apache.zookeeper.proto.ReplyHeader;
import org.apache.zookeeper.proto.RequestHeader;
import org.apache.zookeeper.test.ClientBase;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestInfo;
public class CreateTTLTest extends ClientBase {
private TestableZooKeeper zk;
private static final Collection<String> disabledTests = Collections.singleton("testDisabled");
@Override
public void setUp() throws Exception {
// to be able to get the test method name a testInfo object is needed
// to override the parent's setUp method we need this empty method
}
@BeforeEach
public void setUp(TestInfo testInfo) throws Exception {
System.setProperty(
EphemeralType.EXTENDED_TYPES_ENABLED_PROPERTY,
disabledTests.contains(testInfo.getTestMethod().get().getName()) ? "false" : "true");
super.setUpWithServerId(254);
zk = createClient();
}
@AfterEach
@Override
public void tearDown() throws Exception {
System.clearProperty(EphemeralType.EXTENDED_TYPES_ENABLED_PROPERTY);
super.tearDown();
zk.close();
}
@Test
public void testCreate() throws KeeperException, InterruptedException {
Stat stat = new Stat();
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL, stat, 100);
assertEquals(0, stat.getEphemeralOwner());
final AtomicLong fakeElapsed = new AtomicLong(0);
ContainerManager containerManager = newContainerManager(fakeElapsed);
containerManager.checkContainers();
assertNotNull(zk.exists("/foo", false), "Ttl node should not have been deleted yet");
fakeElapsed.set(1000);
containerManager.checkContainers();
assertNull(zk.exists("/foo", false), "Ttl node should have been deleted");
// validate deleted TTL nodes count
Map<String, Object> metrics = MetricsUtils.currentServerMetrics();
assertTrue((long) metrics.get("ttl_node_deleted_count") >= 1);
}
@Test
public void testBadTTLs() throws InterruptedException, KeeperException {
RequestHeader h = new RequestHeader(1, ZooDefs.OpCode.createTTL);
String path = "/bad_ttl";
CreateTTLRequest request = new CreateTTLRequest(path, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL.toFlag(), -100);
CreateResponse response = new CreateResponse();
ReplyHeader r = zk.submitRequest(h, request, response, null);
assertEquals(r.getErr(), Code.BADARGUMENTS.intValue(), "An invalid CreateTTLRequest should throw BadArguments");
assertNull(zk.exists(path, false), "An invalid CreateTTLRequest should not result in znode creation");
request = new CreateTTLRequest(path, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL.toFlag(),
EphemeralType.TTL.maxValue()
+ 1);
response = new CreateResponse();
r = zk.submitRequest(h, request, response, null);
assertEquals(r.getErr(), Code.BADARGUMENTS.intValue(), "An invalid CreateTTLRequest should throw BadArguments");
assertNull(zk.exists(path, false), "An invalid CreateTTLRequest should not result in znode creation");
}
@Test
public void testMaxTTLs() throws InterruptedException, KeeperException {
RequestHeader h = new RequestHeader(1, ZooDefs.OpCode.createTTL);
String path = "/bad_ttl";
CreateTTLRequest request = new CreateTTLRequest(path, new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL.toFlag(), EphemeralType.TTL.maxValue());
CreateResponse response = new CreateResponse();
ReplyHeader r = zk.submitRequest(h, request, response, null);
assertEquals(r.getErr(), Code.OK.intValue(), "EphemeralType.getMaxTTL() should succeed");
assertNotNull(zk.exists(path, false), "Node should exist");
}
@Test
public void testCreateSequential() throws KeeperException, InterruptedException {
Stat stat = new Stat();
String path = zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_SEQUENTIAL_WITH_TTL, stat, 100);
assertEquals(0, stat.getEphemeralOwner());
final AtomicLong fakeElapsed = new AtomicLong(0);
ContainerManager containerManager = newContainerManager(fakeElapsed);
containerManager.checkContainers();
assertNotNull(zk.exists(path, false), "Ttl node should not have been deleted yet");
fakeElapsed.set(1000);
containerManager.checkContainers();
assertNull(zk.exists(path, false), "Ttl node should have been deleted");
}
@Test
public void testCreateAsync() throws KeeperException, InterruptedException {
AsyncCallback.Create2Callback callback = (rc, path, ctx, name, stat) -> {
// NOP
};
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL, callback, null, 100);
final AtomicLong fakeElapsed = new AtomicLong(0);
ContainerManager containerManager = newContainerManager(fakeElapsed);
containerManager.checkContainers();
assertNotNull(zk.exists("/foo", false), "Ttl node should not have been deleted yet");
fakeElapsed.set(1000);
containerManager.checkContainers();
assertNull(zk.exists("/foo", false), "Ttl node should have been deleted");
}
@Test
public void testModifying() throws KeeperException, InterruptedException {
Stat stat = new Stat();
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL, stat, 100);
assertEquals(0, stat.getEphemeralOwner());
final AtomicLong fakeElapsed = new AtomicLong(0);
ContainerManager containerManager = newContainerManager(fakeElapsed);
containerManager.checkContainers();
assertNotNull(zk.exists("/foo", false), "Ttl node should not have been deleted yet");
for (int i = 0; i < 10; ++i) {
fakeElapsed.set(50);
zk.setData("/foo", new byte[i + 1], -1);
containerManager.checkContainers();
assertNotNull(zk.exists("/foo", false), "Ttl node should not have been deleted yet");
}
fakeElapsed.set(200);
containerManager.checkContainers();
assertNull(zk.exists("/foo", false), "Ttl node should have been deleted");
}
@Test
public void testMulti() throws KeeperException, InterruptedException {
CreateOptions options = CreateOptions
.newBuilder(ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL)
.withTtl(100)
.build();
CreateOptions sequentialOptions = CreateOptions
.newBuilder(ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_SEQUENTIAL_WITH_TTL)
.withTtl(200)
.build();
Op createTtl = Op.create("/a", new byte[0], options.getAcl(), options.getCreateMode(), options.getTtl());
Op createTtl2 = Op.create("/a2", new byte[0], options);
Op createTtlSequential = Op.create("/b", new byte[0], sequentialOptions.getAcl(), sequentialOptions.getCreateMode(), sequentialOptions.getTtl());
Op createTtlSequential2 = Op.create("/b2", new byte[0], sequentialOptions);
Op createNonTtl = Op.create("/c", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT);
List<OpResult> results = zk.multi(Arrays.asList(createTtl, createTtl2, createTtlSequential, createTtlSequential2, createNonTtl));
String sequentialPath = ((OpResult.CreateResult) results.get(2)).getPath();
String sequentialPath2 = ((OpResult.CreateResult) results.get(3)).getPath();
final AtomicLong fakeElapsed = new AtomicLong(0);
ContainerManager containerManager = newContainerManager(fakeElapsed);
containerManager.checkContainers();
assertNotNull(zk.exists("/a", false), "node should not have been deleted yet");
assertNotNull(zk.exists("/a2", false), "node should not have been deleted yet");
assertNotNull(zk.exists(sequentialPath, false), "node should not have been deleted yet");
assertNotNull(zk.exists(sequentialPath2, false), "node should not have been deleted yet");
assertNotNull(zk.exists("/c", false), "node should never be deleted");
fakeElapsed.set(110);
containerManager.checkContainers();
assertNull(zk.exists("/a", false), "node should have been deleted");
assertNull(zk.exists("/a2", false), "node should have been deleted");
assertNotNull(zk.exists(sequentialPath, false), "node should not have been deleted yet");
assertNotNull(zk.exists(sequentialPath2, false), "node should not have been deleted yet");
assertNotNull(zk.exists("/c", false), "node should never be deleted");
fakeElapsed.set(210);
containerManager.checkContainers();
assertNull(zk.exists("/a", false), "node should have been deleted");
assertNull(zk.exists("/a2", false), "node should have been deleted");
assertNull(zk.exists(sequentialPath, false), "node should have been deleted");
assertNull(zk.exists(sequentialPath2, false), "node should have been deleted");
assertNotNull(zk.exists("/c", false), "node should never be deleted");
}
@Test
public void testBadUsage() throws KeeperException, InterruptedException {
for (CreateMode createMode : CreateMode.values()) {
try {
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, createMode, new Stat(), createMode.isTTL() ? 0 : 100);
fail("should have thrown IllegalArgumentException");
} catch (IllegalArgumentException dummy) {
// correct
}
}
for (CreateMode createMode : CreateMode.values()) {
AsyncCallback.Create2Callback callback = (rc, path, ctx, name, stat) -> {
// NOP
};
try {
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, createMode, callback, null, createMode.isTTL() ? 0 : 100);
fail("should have thrown IllegalArgumentException");
} catch (IllegalArgumentException dummy) {
// correct
}
}
try {
Op op = Op.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL, 0);
zk.multi(Collections.singleton(op));
fail("should have thrown IllegalArgumentException");
} catch (IllegalArgumentException dummy) {
// correct
}
try {
Op op = Op.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_SEQUENTIAL_WITH_TTL, 0);
zk.multi(Collections.singleton(op));
fail("should have thrown IllegalArgumentException");
} catch (IllegalArgumentException dummy) {
// correct
}
}
@Test
public void testDisabled() throws KeeperException, InterruptedException {
assertThrows(KeeperException.UnimplementedException.class, () -> {
// note, setUp() enables this test based on the test name
zk.create("/foo", new byte[0], ZooDefs.Ids.OPEN_ACL_UNSAFE, CreateMode.PERSISTENT_WITH_TTL, new Stat(), 100);
});
}
private ContainerManager newContainerManager(final AtomicLong fakeElapsed) {
return new ContainerManager(serverFactory.getZooKeeperServer().getZKDatabase(), serverFactory.getZooKeeperServer().firstProcessor, 1, 100) {
@Override
protected long getElapsed(DataNode node) {
return fakeElapsed.get();
}
};
}
}