diff --git a/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixNode.java b/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixNode.java new file mode 100644 index 0000000000..3009c9a4e8 --- /dev/null +++ b/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixNode.java @@ -0,0 +1,59 @@ +/** + * 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.hadoop.ozone.util; + +import java.util.HashMap; + +/** + * Wrapper class for Radix tree node representing Ozone prefix path segment + * separated by "/". + */ +public class RadixNode { + + public RadixNode(String name) { + this.name = name; + this.children = new HashMap<>(); + } + + public String getName() { + return name; + } + + public boolean hasChildren() { + return children.isEmpty(); + } + + public HashMap getChildren() { + return children; + } + + public void setValue(T v) { + this.value = v; + } + + public T getValue() { + return value; + } + + private HashMap children; + + private String name; + + // TODO: k/v pairs for more metadata as needed + private T value; +} diff --git a/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixTree.java b/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixTree.java new file mode 100644 index 0000000000..72e9ab3f5e --- /dev/null +++ b/hadoop-ozone/common/src/main/java/org/apache/hadoop/ozone/util/RadixTree.java @@ -0,0 +1,214 @@ +/** + * 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.hadoop.ozone.util; + +import com.google.common.annotations.VisibleForTesting; +import org.apache.hadoop.ozone.OzoneConsts; + +import java.util.ArrayList; +import java.util.HashMap; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.List; + +/** + * Wrapper class for handling Ozone prefix path lookup of ACL APIs + * with radix tree. + */ +public class RadixTree { + + /** + * create a empty radix tree with root only. + */ + public RadixTree() { + root = new RadixNode(PATH_DELIMITER); + } + + /** + * If the Radix tree contains root only. + * @return true if the radix tree contains root only. + */ + public boolean isEmpty() { + return root.hasChildren(); + } + + /** + * Insert prefix tree node without value, value can be ACL or other metadata + * of the prefix path. + * @param path + */ + public void insert(String path) { + insert(path, null); + } + + /** + * Insert prefix tree node with value, value can be ACL or other metadata + * of the prefix path. + * @param path + * @param val + */ + public void insert(String path, T val) { + // all prefix path inserted should end with "/" + RadixNode n = root; + Path p = Paths.get(path); + for (int level = 0; level < p.getNameCount(); level++) { + HashMap child = n.getChildren(); + String component = p.getName(level).toString(); + if (child.containsKey(component)) { + n = child.get(component); + } else { + RadixNode tmp = new RadixNode(component); + child.put(component, tmp); + n = tmp; + } + } + if (val != null) { + n.setValue(val); + } + } + + /** + * Get the last node in the exact prefix path that matches in the tree. + * @param path - prefix path + * @return last node in the prefix tree or null if non exact prefix matchl + */ + public RadixNode getLastNodeInPrefixPath(String path) { + List> lpp = getLongestPrefixPath(path); + Path p = Paths.get(path); + if (lpp.size() != p.getNameCount() + 1) { + return null; + } else { + return lpp.get(p.getNameCount()); + } + } + + /** + * Remove prefix path. + * @param path + */ + public void removePrefixPath(String path) { + Path p = Paths.get(path); + removePrefixPathInternal(root, p, 0); + } + + /** + * Recursively remove non-overlapped part of the prefix path from radix tree. + * @param current current radix tree node. + * @param path prefix path to be removed. + * @param level current recursive level. + * @return true if current radix node can be removed. + * (not overlapped with other path), + * false otherwise. + */ + private boolean removePrefixPathInternal(RadixNode current, + Path path, int level) { + // last component is processed + if (level == path.getNameCount()) { + return current.hasChildren(); + } + + // not last component, recur for next component + String name = path.getName(level).toString(); + RadixNode node = current.getChildren().get(name); + if (node == null) { + return false; + } + + if (removePrefixPathInternal(node, path, level+1)) { + current.getChildren().remove(name); + return current.hasChildren(); + } + return false; + } + + /** + * Get the longest prefix path. + * @param path - prefix path. + * @return longest prefix path as list of RadixNode. + */ + public List> getLongestPrefixPath(String path) { + RadixNode n = root; + Path p = Paths.get(path); + int level = 0; + List> result = new ArrayList<>(); + result.add(root); + while (level < p.getNameCount()) { + HashMap children = n.getChildren(); + if (children.isEmpty()) { + break; + } + String component = p.getName(level).toString(); + if (children.containsKey(component)) { + n = children.get(component); + result.add(n); + level++; + } else { + break; + } + } + return result; + } + + @VisibleForTesting + /** + * Convert radix path to string format for output. + * @param path - radix path represented by list of radix nodes. + * @return radix path as string separated by "/". + * Note: the path will always be normalized with and ending "/". + */ + public static String radixPathToString(List> path) { + StringBuilder sb = new StringBuilder(); + for (RadixNode n : path) { + sb.append(n.getName()); + sb.append(n.getName().equals(PATH_DELIMITER) ? "" : PATH_DELIMITER); + } + return sb.toString(); + } + + /** + * Get the longest prefix path. + * @param path - prefix path. + * @return longest prefix path as String separated by "/". + */ + public String getLongestPrefix(String path) { + RadixNode n = root; + Path p = Paths.get(path); + int level = 0; + while (level < p.getNameCount()) { + HashMap children = n.getChildren(); + if (children.isEmpty()) { + break; + } + String component = p.getName(level).toString(); + if (children.containsKey(component)) { + n = children.get(component); + level++; + } else { + break; + } + } + return level >= 1 ? + Paths.get(root.getName()).resolve(p.subpath(0, level)).toString() : + root.getName(); + } + + // root of a radix tree has a name of "/" and may optionally has it value. + private RadixNode root; + + private final static String PATH_DELIMITER = OzoneConsts.OZONE_URI_DELIMITER; +} diff --git a/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/TestRadixTree.java b/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/TestRadixTree.java new file mode 100644 index 0000000000..ceed5346f8 --- /dev/null +++ b/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/TestRadixTree.java @@ -0,0 +1,129 @@ +/* + * 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.hadoop.ozone.util; + +import org.junit.BeforeClass; +import org.junit.Test; + +import java.util.List; + +import static org.junit.Assert.assertEquals; + +/** + * Test Ozone Radix tree operations. + */ +public class TestRadixTree { + + final static RadixTree ROOT = new RadixTree<>(); + + @BeforeClass + public static void setupRadixTree() { + // Test prefix paths with an empty tree + assertEquals(true, ROOT.isEmpty()); + assertEquals("/", ROOT.getLongestPrefix("/a/b/c")); + assertEquals("/", RadixTree.radixPathToString( + ROOT.getLongestPrefixPath("/a/g"))); + // Build Radix tree below for testing. + // a + // | + // b + // / \ + // c e + // / \ / \ \ + // d f g dir1 dir2(1000) + // | + // g + // | + // h + ROOT.insert("/a/b/c/d"); + ROOT.insert("/a/b/c/d/g/h"); + ROOT.insert("/a/b/c/f"); + ROOT.insert("/a/b/e/g"); + ROOT.insert("/a/b/e/dir1"); + ROOT.insert("/a/b/e/dir2", 1000); + } + + /** + * Tests if insert and build prefix tree is correct. + */ + @Test + public void testGetLongestPrefix() { + assertEquals("/a/b/c", ROOT.getLongestPrefix("/a/b/c")); + assertEquals("/a/b", ROOT.getLongestPrefix("/a/b")); + assertEquals("/a", ROOT.getLongestPrefix("/a")); + assertEquals("/a/b/e/g", ROOT.getLongestPrefix("/a/b/e/g/h")); + + assertEquals("/", ROOT.getLongestPrefix("/d/b/c")); + assertEquals("/a/b/e", ROOT.getLongestPrefix("/a/b/e/dir3")); + assertEquals("/a/b/c/d", ROOT.getLongestPrefix("/a/b/c/d/p")); + + assertEquals("/a/b/c/f", ROOT.getLongestPrefix("/a/b/c/f/p")); + } + + @Test + public void testGetLongestPrefixPath() { + List> lpp = + ROOT.getLongestPrefixPath("/a/b/c/d/g/p"); + RadixNode lpn = lpp.get(lpp.size()-1); + assertEquals("g", lpn.getName()); + lpn.setValue(100); + + + List> lpq = + ROOT.getLongestPrefixPath("/a/b/c/d/g/q"); + RadixNode lqn = lpp.get(lpq.size()-1); + System.out.print(RadixTree.radixPathToString(lpq)); + assertEquals(lpn, lqn); + assertEquals("g", lqn.getName()); + assertEquals(100, (int)lqn.getValue()); + + + assertEquals("/a/", RadixTree.radixPathToString( + ROOT.getLongestPrefixPath("/a/g"))); + + } + + @Test + public void testGetLastNoeInPrefixPath() { + assertEquals(null, ROOT.getLastNodeInPrefixPath("/a/g")); + RadixNode ln = ROOT.getLastNodeInPrefixPath("/a/b/e/dir1"); + assertEquals("dir1", ln.getName()); + } + + @Test + public void testRemovePrefixPath() { + + // Remove, test and restore + // Remove partially overlapped path + ROOT.removePrefixPath("/a/b/c/d/g/h"); + assertEquals("/a/b/c", ROOT.getLongestPrefix("a/b/c/d")); + ROOT.insert("/a/b/c/d/g/h"); + + // Remove fully overlapped path + ROOT.removePrefixPath("/a/b/c/d"); + assertEquals("/a/b/c/d", ROOT.getLongestPrefix("a/b/c/d")); + ROOT.insert("/a/b/c/d"); + + // Remove non existing path + ROOT.removePrefixPath("/d/a"); + assertEquals("/a/b/c/d", ROOT.getLongestPrefix("a/b/c/d")); + } + + +} diff --git a/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/package-info.java b/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/package-info.java new file mode 100644 index 0000000000..a6acd30d77 --- /dev/null +++ b/hadoop-ozone/common/src/test/java/org/apache/hadoop/ozone/util/package-info.java @@ -0,0 +1,21 @@ +/** + * 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.hadoop.ozone.util; +/** + * Unit tests of generic ozone utils. + */