Skip to content

Commit

Permalink
Add map_top_n_keys Presto function (facebookincubator#10271)
Browse files Browse the repository at this point in the history
Summary:
Pull Request resolved: facebookincubator#10271

This function returns the respective items within the input vector with the top N keys in descending order.

map_top_n_keys is  a Presto function defined here

https://www.internalfb.com/code/fbsource/[2d472d9e8215dd5f1a792f38b3d8c2dbba320698]/fbcode/github/presto-trunk/presto-main/src/main/java/com/facebook/presto/operator/scalar/sql/MapSqlFunctions.java?lines=60

Differential Revision: D58621202
  • Loading branch information
Daniel Hunte authored and facebook-github-bot committed Jun 26, 2024
1 parent 7add4bf commit 877520a
Show file tree
Hide file tree
Showing 6 changed files with 205 additions and 0 deletions.
9 changes: 9 additions & 0 deletions velox/docs/functions/presto/map.rst
Original file line number Diff line number Diff line change
Expand Up @@ -114,6 +114,15 @@ Map Functions
SELECT map_top_n(map(ARRAY['a', 'b', 'c'], ARRAY[2, 3, 1]), 2) --- {'b' -> 3, 'a' -> 2}
SELECT map_top_n(map(ARRAY['a', 'b', 'c'], ARRAY[NULL, 3, NULL]), 2) --- {'b' -> 3, 'a' -> NULL}

.. function:: map_top_n_keys(map(K,V), n) -> array(K)

Constructs an array of the top N keys. Keys should be orderable.

``n`` must be a non-negative BIGINT value.::

SELECT map_top_n_keys(map(ARRAY['a', 'b', 'c'], ARRAY[1, 2, 3]), 2) --- ['c', 'b']
SELECT map_top_n_keys(map(ARRAY['a', 'b', 'c'], ARRAY[1, 2, 3]), 0) --- []

.. function:: map_keys(x(K,V)) -> array(K)

Returns all the keys in the map ``x``.
Expand Down
1 change: 1 addition & 0 deletions velox/functions/prestosql/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ add_library(
MapEntries.cpp
MapFromEntries.cpp
MapKeysAndValues.cpp
MapTopNKeys.cpp
MapZipWith.cpp
Not.cpp
Reduce.cpp
Expand Down
61 changes: 61 additions & 0 deletions velox/functions/prestosql/MapTopNImpl.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* 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.
*/
#pragma once

#include "velox/expression/ComplexViewTypes.h"
#include "velox/functions/Udf.h"

namespace facebook::velox::functions {

template <typename TExec, typename Compare>
struct MapTopNImpl {
VELOX_DEFINE_FUNCTION_TYPES(TExec);

void call(
out_type<Array<Orderable<T1>>>& out,
const arg_type<Map<Orderable<T1>, Orderable<T2>>>& inputMap,
int64_t n) {
VELOX_USER_CHECK_GE(n, 0, "n must be greater than or equal to 0")

if (n == 0) {
return;
}
using It = typename arg_type<Map<Orderable<T1>, Orderable<T2>>>::Iterator;
Compare comparator;
std::priority_queue<It, std::vector<It>, Compare> topEntries(comparator);

for (auto it = inputMap.begin(); it != inputMap.end(); ++it) {
if (topEntries.size() < n) {
topEntries.push(it);
} else if (comparator(it, topEntries.top())) {
topEntries.pop();
topEntries.push(it);
}
}
std::vector<It> result;
result.reserve(topEntries.size());
while (!topEntries.empty()) {
result.push_back(topEntries.top());
topEntries.pop();
}
// Reverse the order of the result to be in descending order.
for (size_t i = result.size() - 1; i >= 0; i--) {
out.push_back(result[i]->first);
}
}
};

} // namespace facebook::velox::functions
44 changes: 44 additions & 0 deletions velox/functions/prestosql/MapTopNKeys.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* 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.
*/
#pragma once

#include "velox/expression/ComplexViewTypes.h"
#include "velox/functions/Udf.h"
#include "velox/functions/prestosql/MapTopNImpl.h"

namespace facebook::velox::functions {

template <typename TExec>
struct CompareKeys {
VELOX_DEFINE_FUNCTION_TYPES(TExec);

using It = typename arg_type<Map<Orderable<T1>, Orderable<T2>>>::Iterator;

bool operator()(const It& l, const It& r) const {
static const CompareFlags flags{
false /*nullsFirst*/,
true /*ascending*/,
false /*equalsOnly*/,
CompareFlags::NullHandlingMode::kNullAsIndeterminate};
return l->first.compare(r->first, flags) > 0;
}
};

// Returns an array with the top N keys in descending order.
template <typename TExec>
struct MapTopNKeysFunction : MapTopNImpl<TExec, CompareKeys<TExec>> {};

} // namespace facebook::velox::functions
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
#include "velox/functions/prestosql/MapRemoveNullValues.h"
#include "velox/functions/prestosql/MapSubset.h"
#include "velox/functions/prestosql/MapTopN.h"
#include "velox/functions/prestosql/MapTopNKeys.h"
#include "velox/functions/prestosql/MultimapFromEntries.h"

namespace facebook::velox::functions {
Expand Down Expand Up @@ -104,6 +105,12 @@ void registerMapFunctions(const std::string& prefix) {
Map<Orderable<T1>, Orderable<T2>>,
int64_t>({prefix + "map_top_n"});

registerFunction<
MapTopNKeysFunction,
Array<Orderable<T1>>,
Map<Orderable<T1>, Orderable<T2>>,
int64_t>({prefix + "map_top_n_keys"});

registerMapSubset(prefix);

registerMapRemoveNullValues(prefix);
Expand Down
83 changes: 83 additions & 0 deletions velox/functions/prestosql/tests/MapTopNKeysTest.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* 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.
*/
#include "velox/common/base/tests/GTestUtils.h"
#include "velox/functions/prestosql/tests/utils/FunctionBaseTest.h"

using namespace facebook::velox::test;

namespace facebook::velox::functions {
namespace {

class MapTopNKeysTest : public test::FunctionBaseTest {};

TEST_F(MapTopNKeysTest, emptyMap) {
RowVectorPtr input = makeRowVector({
makeMapVectorFromJson<int32_t, int64_t>({
"{}",
}),
});

assertEqualVectors(
evaluate("map_top_n_keys(c0, 3)", input),
makeArrayVectorFromJson<int32_t>({
"[]",
}));
}

TEST_F(MapTopNKeysTest, multipleMaps) {
RowVectorPtr input = makeRowVector({
makeMapVectorFromJson<int32_t, int64_t>({
"{3:1, 2:1, 5:1, 4:1, 1:1}",
"{3:1, 2:1, 1:1}",
"{2:1, 1:1}",
}),
});

assertEqualVectors(
evaluate("map_top_n_keys(c0, 3)", input),
makeArrayVectorFromJson<int32_t>({
"[5, 4, 3]",
"[3, 2, 1]",
"[2, 1]",
}));
}

TEST_F(MapTopNKeysTest, nIsZero) {
RowVectorPtr input = makeRowVector({
makeMapVectorFromJson<int32_t, int64_t>({
"{2:1, 1:1}",
}),
});

assertEqualVectors(
evaluate("map_top_n_keys(c0, 0)", input),
makeArrayVectorFromJson<int32_t>({"[]"}));
}

TEST_F(MapTopNKeysTest, nIsNegative) {
RowVectorPtr input = makeRowVector({
makeMapVectorFromJson<int32_t, int64_t>({
"{2:1, 1:1}",
}),
});

VELOX_ASSERT_THROW(
evaluate("map_top_n_keys(c0, -1)", input),
"n must be greater than or equal to 0");
}

} // namespace
} // namespace facebook::velox::functions

0 comments on commit 877520a

Please sign in to comment.