blob: b51440444506f295608e8a7b03ade5a7657c2477 [file] [log] [blame]
/** @file
Unit tests for PriorityQueue
@section license License
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.
*/
#include <cstdint>
#include "tscore/PriorityQueue.h"
#include "catch.hpp"
using namespace std;
class N
{
public:
N(uint32_t w, string c) : weight(w), content(std::move(c)) {}
bool
operator<(const N &n) const
{
return weight < n.weight;
}
uint32_t weight;
string content;
};
using Entry = PriorityQueueEntry<N *>;
using PQ = PriorityQueue<N *>;
// Push, top, and pop a entry
TEST_CASE("PriorityQueue1", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(6, "A");
Entry *entry_a = new Entry(a);
pq->push(entry_a);
REQUIRE(pq->top() == entry_a);
pq->pop();
REQUIRE(pq->top() == nullptr);
delete pq;
delete a;
delete entry_a;
}
// Increase weight
TEST_CASE("PriorityQueue2", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(10, "A");
N *b = new N(20, "B");
N *c = new N(30, "C");
Entry *entry_a = new Entry(a);
Entry *entry_b = new Entry(b);
Entry *entry_c = new Entry(c);
pq->push(entry_a);
pq->push(entry_b);
pq->push(entry_c);
REQUIRE(pq->top() == entry_a);
a->weight = 40;
pq->update(entry_a);
REQUIRE(pq->top() == entry_b);
b->weight = 50;
pq->update(entry_b, true);
REQUIRE(pq->top() == entry_c);
delete pq;
delete a;
delete b;
delete c;
delete entry_a;
delete entry_b;
delete entry_c;
}
// Decrease weight
TEST_CASE("PriorityQueue3", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(10, "A");
N *b = new N(20, "B");
N *c = new N(30, "C");
Entry *entry_a = new Entry(a);
Entry *entry_b = new Entry(b);
Entry *entry_c = new Entry(c);
pq->push(entry_a);
pq->push(entry_b);
pq->push(entry_c);
REQUIRE(pq->top() == entry_a);
b->weight = 5;
pq->update(entry_b);
REQUIRE(pq->top() == entry_b);
c->weight = 3;
pq->update(entry_c, false);
REQUIRE(pq->top() == entry_c);
delete pq;
delete a;
delete b;
delete c;
delete entry_a;
delete entry_b;
delete entry_c;
}
// Push, top, and pop 9 entries
TEST_CASE("PriorityQueue4", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(6, "A");
N *b = new N(1, "B");
N *c = new N(9, "C");
N *d = new N(8, "D");
N *e = new N(4, "E");
N *f = new N(3, "F");
N *g = new N(2, "G");
N *h = new N(7, "H");
N *i = new N(5, "I");
Entry *entry_a = new Entry(a);
Entry *entry_b = new Entry(b);
Entry *entry_c = new Entry(c);
Entry *entry_d = new Entry(d);
Entry *entry_e = new Entry(e);
Entry *entry_f = new Entry(f);
Entry *entry_g = new Entry(g);
Entry *entry_h = new Entry(h);
Entry *entry_i = new Entry(i);
pq->push(entry_a);
pq->push(entry_b);
pq->push(entry_c);
pq->push(entry_d);
pq->push(entry_e);
pq->push(entry_f);
pq->push(entry_g);
pq->push(entry_h);
pq->push(entry_i);
REQUIRE(pq->top() == entry_b); // 1
pq->pop();
REQUIRE(pq->top() == entry_g); // 2
pq->pop();
REQUIRE(pq->top() == entry_f); // 3
pq->pop();
REQUIRE(pq->top() == entry_e); // 4
pq->pop();
REQUIRE(pq->top() == entry_i); // 5
pq->pop();
REQUIRE(pq->top() == entry_a); // 6
pq->pop();
REQUIRE(pq->top() == entry_h); // 7
pq->pop();
REQUIRE(pq->top() == entry_d); // 8
pq->pop();
REQUIRE(pq->top() == entry_c); // 9
pq->pop();
REQUIRE(pq->top() == nullptr);
delete pq;
delete a;
delete b;
delete c;
delete d;
delete e;
delete f;
delete g;
delete h;
delete i;
delete entry_a;
delete entry_b;
delete entry_c;
delete entry_d;
delete entry_e;
delete entry_f;
delete entry_g;
delete entry_h;
delete entry_i;
}
// Push, top, pop, and update 9 entries
TEST_CASE("PriorityQueue5", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(6, "A");
N *b = new N(1, "B");
N *c = new N(9, "C");
N *d = new N(8, "D");
N *e = new N(4, "E");
N *f = new N(3, "F");
N *g = new N(2, "G");
N *h = new N(7, "H");
N *i = new N(5, "I");
Entry *entry_a = new Entry(a);
Entry *entry_b = new Entry(b);
Entry *entry_c = new Entry(c);
Entry *entry_d = new Entry(d);
Entry *entry_e = new Entry(e);
Entry *entry_f = new Entry(f);
Entry *entry_g = new Entry(g);
Entry *entry_h = new Entry(h);
Entry *entry_i = new Entry(i);
pq->push(entry_a);
pq->push(entry_b);
pq->push(entry_c);
pq->push(entry_d);
pq->push(entry_e);
pq->push(entry_f);
pq->push(entry_g);
pq->push(entry_h);
pq->push(entry_i);
// Pop head and push it back again
REQUIRE(pq->top() == entry_b); // 1
pq->pop();
b->weight += 100;
pq->push(entry_b);
// Update weight
a->weight += 100;
pq->update(entry_a);
c->weight += 100;
pq->update(entry_d);
e->weight += 100;
pq->update(entry_e);
g->weight += 100;
pq->update(entry_g);
// Check
REQUIRE(pq->top() == entry_f); // 3
pq->pop();
REQUIRE(pq->top() == entry_i); // 5
pq->pop();
REQUIRE(pq->top() == entry_h); // 7
pq->pop();
REQUIRE(pq->top() == entry_d); // 8
pq->pop();
REQUIRE(pq->top() == entry_b); // 101
pq->pop();
REQUIRE(pq->top() == entry_g); // 102
pq->pop();
REQUIRE(pq->top() == entry_e); // 104
pq->pop();
REQUIRE(pq->top() == entry_a); // 106
pq->pop();
REQUIRE(pq->top() == entry_c); // 109
pq->pop();
REQUIRE(pq->top() == nullptr);
delete pq;
delete a;
delete b;
delete c;
delete d;
delete e;
delete f;
delete g;
delete h;
delete i;
delete entry_a;
delete entry_b;
delete entry_c;
delete entry_d;
delete entry_e;
delete entry_f;
delete entry_g;
delete entry_h;
delete entry_i;
}
// Test erase method
TEST_CASE("PriorityQueue6", "[libts][PriorityQueue]")
{
PQ *pq = new PQ();
N *a = new N(10, "A");
N *b = new N(20, "B");
N *c = new N(30, "C");
Entry *entry_a = new Entry(a);
Entry *entry_b = new Entry(b);
Entry *entry_c = new Entry(c);
pq->push(entry_a);
pq->push(entry_b);
pq->push(entry_c);
uint32_t index;
REQUIRE(pq->top() == entry_a);
index = entry_a->index;
pq->erase(entry_a);
REQUIRE(entry_a->index == index);
REQUIRE(pq->top() == entry_b);
index = entry_c->index;
pq->erase(entry_c);
REQUIRE(entry_c->index == index);
REQUIRE(pq->top() == entry_b);
index = entry_b->index;
pq->erase(entry_b);
REQUIRE(entry_b->index == index);
REQUIRE(pq->top() == nullptr);
REQUIRE(pq->empty());
delete pq;
delete a;
delete b;
delete c;
delete entry_a;
delete entry_b;
delete entry_c;
PQ *pq2 = new PQ();
N *w = new N(10, "W");
N *x = new N(20, "X");
N *y = new N(30, "Y");
N *z = new N(40, "Z");
Entry *entry_w = new Entry(w);
Entry *entry_x = new Entry(x);
Entry *entry_y = new Entry(y);
Entry *entry_z = new Entry(z);
pq2->push(entry_z);
pq2->push(entry_y);
pq2->push(entry_x);
pq2->push(entry_w);
REQUIRE(pq2->top() == entry_w);
pq2->erase(entry_x);
REQUIRE(pq2->top() == entry_w);
// The following two cases should test that erase preserves the index
pq2->erase(entry_y);
REQUIRE(pq2->top() == entry_w);
pq2->erase(entry_z);
REQUIRE(pq2->top() == entry_w);
delete pq2;
delete w;
delete x;
delete y;
delete z;
delete entry_w;
delete entry_x;
delete entry_y;
delete entry_z;
}
// Test erase and pop method to ensure the index entries are updated (TS-4915)
TEST_CASE("PriorityQueue7", "[libts][PriorityQueue]")
{
PQ *pq2 = new PQ();
N *x = new N(20, "X");
N *y = new N(30, "Y");
N *z = new N(40, "Z");
Entry *entry_x = new Entry(x);
Entry *entry_y = new Entry(y);
Entry *entry_z = new Entry(z);
pq2->push(entry_z);
pq2->push(entry_y);
pq2->push(entry_x);
REQUIRE(pq2->top() == entry_x);
pq2->pop();
REQUIRE(pq2->top() == entry_y);
pq2->erase(entry_y);
REQUIRE(pq2->top() == entry_z);
delete pq2;
delete x;
delete y;
delete z;
delete entry_x;
delete entry_y;
delete entry_z;
}
// Test erase and pop method to ensure the index entries are correctly
TEST_CASE("PriorityQueue8", "[libts][PriorityQueue]")
{
PQ *pq1 = new PQ();
PQ *pq2 = new PQ();
N *x = new N(20, "X");
N *y = new N(30, "Y");
N *z = new N(40, "Z");
Entry *entry_x = new Entry(x);
Entry *entry_y = new Entry(y);
Entry *entry_z = new Entry(z);
pq2->push(entry_z);
pq2->push(entry_y);
pq2->push(entry_x);
x->weight = 40;
y->weight = 30;
z->weight = 20;
pq1->push(pq2->top());
pq2->pop();
REQUIRE(pq1->top()->index == 0);
pq1->push(pq2->top());
pq2->pop();
REQUIRE(pq1->top()->index == 0);
pq1->push(pq2->top());
pq2->pop();
REQUIRE(pq1->top()->index == 0);
delete pq1;
delete pq2;
delete x;
delete y;
delete z;
delete entry_x;
delete entry_y;
delete entry_z;
}
TEST_CASE("PriorityQueue9", "[libts][PriorityQueue]")
{
PQ *pq1 = new PQ();
N *x = new N(20, "X");
N *y = new N(30, "Y");
Entry *X = new Entry(x);
Entry *Y = new Entry(y);
REQUIRE(X->index == 0);
REQUIRE(Y->index == 0);
pq1->push(X);
pq1->erase(Y);
REQUIRE(pq1->top() == X);
delete x;
delete y;
delete X;
delete Y;
delete pq1;
}