13 #ifndef HELIB_PARTIALMATCH_H
14 #define HELIB_PARTIALMATCH_H
19 #include <helib/Matrix.h>
20 #include <helib/PolyMod.h>
38 template <
typename TXT>
43 if (query.
dims(0) != 1)
45 if (query.
dims(1) != database.dims(1))
47 "Database and query must have same number of columns");
52 std::vector<long> columns(database.dims(0), 0l);
59 .apply([&](
auto& entry) {
mapTo01(ea, entry); })
60 .apply([](
auto& entry) { entry.negate(); })
61 .apply([](
auto& entry) { entry.addConstant(NTL::ZZX(1l)); });
77 template <
typename TXT>
82 if (query.
dims(0) != 1)
84 if (query.
dims(1) != database.
dims(1))
86 "Database and query must have same number of columns");
91 std::vector<long> columns(database.
dims(0), 0l);
98 .apply([&](
auto& entry) {
mapTo01(ea, entry); })
99 .apply([](
auto& entry) { entry.negate(); })
100 .apply([](
auto& entry) { entry.addConstant(NTL::ZZX(1l)); });
118 template <
typename TXT>
120 const std::vector<std::vector<long>> index_sets,
121 const std::vector<long>& offsets,
125 assertEq<InvalidArgument>(index_sets.size(),
127 "index_sets and offsets must have matching size");
128 assertEq<InvalidArgument>(index_sets.size(),
130 "index_sets and weights must have matching size");
131 auto ones(mask(0, 0));
133 ones.addConstant(NTL::ZZX(1L));
135 for (std::size_t i = 0; i < index_sets.size(); ++i) {
136 const auto& index_set = index_sets.at(i);
137 const auto& weight_set = weights.at(i);
138 long offset = offsets.at(i);
140 assertEq<InvalidArgument>(
143 "found mismatch between index set size and weight set size");
145 assertEq<InvalidArgument>(weight_set.dims(1),
147 "all weight sets must be column vectors");
152 factor.
apply([&](
auto& entry) { entry.addConstant(NTL::ZZX(offset)); });
153 result.template entrywiseOperation<TXT>(
155 [](
auto& lhs,
const auto& rhs) -> decltype(
auto) {
176 for (
long i = 0; i < long(coeffs.size()) && input != 0; ++i) {
177 coeffs[i] = input % p;
199 return std::make_shared<ColNumber>(cl);
209 virtual std::string
eval()
const = 0;
225 std::string
eval()
const override {
return std::to_string(column); }
252 std::string
eval()
const override
254 return lhs->eval() +
" " + rhs->eval() +
" &&";
284 std::string
eval()
const override
286 return lhs->eval() +
" " + rhs->eval() +
" ||";
311 return std::make_shared<And>(lhs, rhs);
324 return std::make_shared<Or>(lhs, rhs);
337 std::vector<std::vector<long>>
Fs;
350 std::vector<Matrix<long>>
taus;
366 Query_t(
const std::vector<std::vector<long>>& index_sets,
367 const std::vector<long>& offsets,
369 const bool isThereAnOR) :
381 Query_t(std::vector<std::vector<long>>&& index_sets,
382 std::vector<long>&& offsets,
398 using vecvec = std::vector<std::vector<long>>;
418 vecvec expr = expandOr(query_str);
419 bool containsOR =
false;
421 vecvec Fs(expr.size());
423 std::vector<long> v(columns);
424 std::iota(v.begin(), v.end(), 0);
425 std::fill(Fs.begin(), Fs.end(), v);
427 std::vector<long> mus(expr.size(), 1);
428 std::vector<Matrix<long>> taus;
429 taus.reserve(expr.size());
432 for (
long i = 0; i < long(expr.size()); ++i) {
435 containsOR = (expr[i].size() > 1) ?
true :
false;
436 for (
long j = 0; j < long(expr[i].size()); ++j)
437 M(expr[i][j], 0) = 1;
438 taus.push_back(std::move(M));
441 return Query_t(std::move(Fs), std::move(mus), std::move(taus), containsOR);
445 std::string query_str;
447 void printStack(std::stack<vecvec> stack)
449 while (!stack.empty()) {
450 printVecVec(stack.top());
455 void printVecVec(
const vecvec& vv)
457 for (
const auto& v : vv) {
459 for (
const auto& e : v) {
460 std::cout << e <<
" ";
467 bool isNumber(
const std::string& s)
const
470 return std::all_of(s.begin(), s.end(), ::isdigit);
473 vecvec expandOr(
const std::string& s)
const
475 std::stack<vecvec> convertStack;
477 std::istringstream input{s};
478 std::ostringstream output{};
482 while (input >> symbol) {
483 if (!symbol.compare(
"&&")) {
485 auto op = convertStack.top();
487 auto& top = convertStack.top();
488 top.insert(top.end(), op.begin(), op.end());
489 }
else if (!symbol.compare(
"||")) {
491 auto op1 = convertStack.top();
493 auto op2 = convertStack.top();
497 prod.reserve(op1.size() * op2.size());
498 for (
const auto& i : op1)
499 for (
const auto& j : op2) {
501 x.insert(x.end(), j.begin(), j.end());
502 prod.push_back(std::move(x));
505 convertStack.push(std::move(prod));
509 "String is not a number: '" + symbol +
"'");
510 convertStack.emplace(vecvec(1, {std::stol(symbol)}));
515 assertEq<LogicError>(1UL,
517 "Size of stack after expandOr should be 1");
519 return std::move(convertStack.top());
529 template <
typename TXT>
557 context(std::shared_ptr<const
helib::
Context>(&c, [](auto UNUSED p) {}))
571 template <
typename TXT2>
572 Matrix<TXT2>
contains(
const Query_t& lookup_query,
573 const Matrix<TXT2>& query_data)
const;
585 template <
typename TXT2>
586 Matrix<TXT2>
getScore(
const Query_t& weighted_query,
587 const Matrix<TXT2>& query_data)
const;
598 std::shared_ptr<const Context> context;
601 template <
typename TXT>
602 template <
typename TXT2>
607 auto result = getScore<TXT2>(lookup_query, query_data);
611 result.apply([&](
auto& txt) {
612 txt.power(context->alMod.getPPowR() - 1);
620 template <
typename TXT>
621 template <
typename TXT2>
An object that mimics the functionality of the Ctxt object, and acts as a convenient entry point for ...
Definition: Ptxt.h:280
An object used to construct a Query_t object from a logical expression.
Definition: partialMatch.h:395
Database(const Matrix< TXT > &M, std::shared_ptr< const Context > c)
Constructor.
Definition: partialMatch.h:542
Matrix< TXT > calculateMasks(const EncryptedArray &ea, Matrix< TXT > query, const Matrix< Ptxt< BGV >> &database)
Given a query set and a database, calculates a mask of {0,1} where 1 signifies a matching element and...
Definition: partialMatch.h:39
Matrix< TXT > calculateScores(const std::vector< std::vector< long >> index_sets, const std::vector< long > &offsets, const std::vector< Matrix< long >> &weights, const Matrix< TXT > &mask)
Given a mask and information about the query to be performed, calculates a score for each matching el...
Definition: partialMatch.h:119
void assertTrue(const T &value, const std::string &message)
Definition: assertions.h:61
QueryBuilder(const QueryExpr &expr)
Constructor.
Definition: partialMatch.h:406
Matrix< TXT2 > getScore(const Query_t &weighted_query, const Matrix< TXT2 > &query_data) const
Function for performing a weighted partial match given a query expression and query data.
Definition: partialMatch.h:622
Query_t(std::vector< std::vector< long >> &&index_sets, std::vector< long > &&offsets, std::vector< Matrix< long >> &&weights, bool isThereAnOR)
Constructor.
Definition: partialMatch.h:381
std::string eval() const override
Function for returning the logical AND expression in reverse polish notation where the AND operation ...
Definition: partialMatch.h:252
long getOrdP() const
The order of p in (Z/mZ)^*.
Definition: PAlgebra.h:171
Base structure for logical expressions.
Definition: partialMatch.h:208
An object representing a column of a database as an expression which inherits from Expr.
Definition: partialMatch.h:219
std::shared_ptr< And > operator&&(const QueryExpr &lhs, const QueryExpr &rhs)
Overloaded operator for creating a shared pointer to an AND expression.
Definition: partialMatch.h:308
And(const QueryExpr &l, const QueryExpr &r)
Constructor.
Definition: partialMatch.h:262
std::size_t dims(int i) const
Definition: Matrix.h:225
A simple wrapper for a smart pointer to an EncryptedArrayBase. This is the interface that higher-leve...
Definition: EncryptedArray.h:1233
long getP() const
Returns p.
Definition: PAlgebra.h:165
std::vector< long > mus
std::vector of offsets. Each offset is a constant value. There should be a single offset for each ind...
Definition: partialMatch.h:343
Structure containing all information required for an HE query.
Definition: partialMatch.h:332
std::vector< Matrix< long > > taus
std::vector of a set of weights. Each weight set corresponds to a single index set where each individ...
Definition: partialMatch.h:350
ColNumber(int c)
Constructor.
Definition: partialMatch.h:231
std::string eval() const override
Function for returning the logical OR expression in reverse polish notation where the OR operation is...
Definition: partialMatch.h:284
Query_t build(long columns) const
Function for building the Query_t object from the expression.
Definition: partialMatch.h:414
std::string eval() const override
Function for returning the column number of the object.
Definition: partialMatch.h:225
An object representing the logical OR expression which inherits from Expr.
Definition: partialMatch.h:275
std::shared_ptr< ColNumber > makeQueryExpr(int cl)
Utility function for creating a shared pointer to a specified column in a query.
Definition: partialMatch.h:197
long columns()
Returns number of columns in the database.
Definition: partialMatch.h:594
Inherits from Exception and std::invalid_argument.
Definition: exceptions.h:140
std::vector< std::vector< long > > Fs
std::vector of index sets. These index sets specify the indexes of the columns in each column subset.
Definition: partialMatch.h:337
An object representing the logical AND expression which inherits from Expr.
Definition: partialMatch.h:243
Definition: apiAttributes.h:21
const std::vector< T > & data() const
Definition: Matrix.h:465
Or(const QueryExpr &l, const QueryExpr &r)
Constructor.
Definition: partialMatch.h:294
Query_t(const std::vector< std::vector< long >> &index_sets, const std::vector< long > &offsets, const std::vector< Matrix< long >> &weights, const bool isThereAnOR)
Constructor.
Definition: partialMatch.h:366
PolyMod partialMatchEncode(uint32_t input, const Context &context)
Given a value, encode the value across the coefficients of a polynomial.
Definition: partialMatch.h:171
Tensor< T, 2 > & transpose()
Definition: Matrix.h:393
Database(const Matrix< TXT > &M, const Context &c)
Constructor.
Definition: partialMatch.h:555
bool containsOR
Flag indicating if the query contains a logical OR operation. This is used for optimization purposes.
Definition: partialMatch.h:356
Maintaining the parameters.
Definition: Context.h:121
std::shared_ptr< Or > operator||(const QueryExpr &lhs, const QueryExpr &rhs)
Overloaded operator for creating a shared pointer to an OR expression.
Definition: partialMatch.h:321
Matrix< TXT2 > contains(const Query_t &lookup_query, const Matrix< TXT2 > &query_data) const
Function for performing a database lookup given a query expression and query data.
Definition: partialMatch.h:603
virtual std::string eval() const =0
An object that contains an NTL::ZZX polynomial along with a coefficient modulus p2r and a polynomial ...
Definition: PolyMod.h:47
void mapTo01(const EncryptedArray &ea, Ctxt &ctxt)
Definition: eqtesting.cpp:35
std::shared_ptr< PolyModRing > slotRing
The structure of a single slot of the plaintext space.
Definition: Context.h:145
def prod(iterable)
Definition: encode.py:67
std::shared_ptr< Expr > QueryExpr
An alias for a shared pointer to an Expr object.
Definition: partialMatch.h:189
Tensor< T, N > columns(const std::vector< long > &js) const
Definition: Matrix.h:272
PAlgebra zMStar
The structure of Zm*.
Definition: Context.h:131
Tensor< T, N > & apply(std::function< void(T &x)> fn)
Definition: Matrix.h:372
An object representing a database which is a HElib::Matrix<TXT>.
Definition: partialMatch.h:531