2023-03-09 08:09:50 +08:00
|
|
|
/*
|
|
|
|
* ResolverBug.cpp
|
|
|
|
*
|
|
|
|
* This source file is part of the FoundationDB open source project
|
|
|
|
*
|
|
|
|
* Copyright 2013-2023 Apple Inc. and the FoundationDB project authors
|
|
|
|
*
|
|
|
|
* 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 "flow/ProcessEvents.h"
|
|
|
|
#include "fdbserver/workloads/workloads.actor.h"
|
|
|
|
#include "fdbserver/ResolverBug.h"
|
|
|
|
#include "fdbserver/ServerDBInfo.actor.h"
|
|
|
|
|
|
|
|
#include "flow/actorcompiler.h" // has to be last include
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
struct ResolverBugWorkload : TestWorkload {
|
|
|
|
constexpr static auto NAME = "ResolverBug";
|
|
|
|
bool disableFailureInjections;
|
|
|
|
ResolverBug resolverBug;
|
|
|
|
Standalone<VectorRef<KeyValueRef>> cycleOptions;
|
|
|
|
KeyRef controlKey = "workload_control"_sr;
|
2023-03-10 04:05:00 +08:00
|
|
|
Promise<Void> bugFound;
|
2023-03-09 08:09:50 +08:00
|
|
|
|
|
|
|
ResolverBugWorkload(WorkloadContext const& wcx) : TestWorkload(wcx) {
|
|
|
|
disableFailureInjections = getOption(options, "disableFailureInjections"_sr, true);
|
|
|
|
resolverBug.ignoreTooOldProbability = getOption(options, "ignoreTooOldProbability"_sr, 0.0);
|
|
|
|
resolverBug.ignoreWriteSetProbability = getOption(options, "ignoreWriteSetProbability"_sr, 0.0);
|
|
|
|
resolverBug.ignoreReadSetProbability = getOption(options, "ignoreReadSetProbability"_sr, 0.0);
|
|
|
|
|
|
|
|
for (auto& o : options) {
|
|
|
|
if (o.key.startsWith("cycle_"_sr)) {
|
|
|
|
KeyValueRef option;
|
|
|
|
option.key = o.key.removePrefix("cycle_"_sr);
|
|
|
|
option.value = o.value;
|
|
|
|
cycleOptions.push_back_deep(cycleOptions.arena(), option);
|
|
|
|
o.value = ""_sr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clientId == 0) {
|
|
|
|
SimBugInjector().enable();
|
|
|
|
auto bug = SimBugInjector().enable<ResolverBug>(ResolverBugID());
|
|
|
|
*bug = resolverBug;
|
|
|
|
bug->cycleState.resize(clientCount, 0);
|
|
|
|
SimBugInjector().disable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void disableFailureInjectionWorkloads(std::set<std::string>& out) const override {
|
|
|
|
if (disableFailureInjections) {
|
|
|
|
out.insert("all");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Reference<TestWorkload> createCycle() {
|
|
|
|
WorkloadContext wcx;
|
|
|
|
wcx.clientId = clientId;
|
|
|
|
wcx.clientCount = clientCount;
|
|
|
|
wcx.ccr = ccr;
|
|
|
|
wcx.dbInfo = dbInfo;
|
|
|
|
wcx.options = cycleOptions;
|
|
|
|
wcx.sharedRandomNumber = sharedRandomNumber;
|
|
|
|
wcx.defaultTenant = defaultTenant.castTo<TenantName>();
|
|
|
|
return IWorkloadFactory::create("Cycle", wcx);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR static Future<Void> waitForPhase(std::shared_ptr<ResolverBug> bug, int phase) {
|
|
|
|
while (bug->currentPhase != phase) {
|
|
|
|
wait(delay(0.5));
|
|
|
|
}
|
|
|
|
return Void();
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR static Future<Void> waitForPhaseDone(std::shared_ptr<ResolverBug> bug, int phase, int clientCount) {
|
|
|
|
while (std::count(bug->cycleState.begin(), bug->cycleState.end(), phase) != clientCount) {
|
|
|
|
wait(delay(0.5));
|
|
|
|
}
|
|
|
|
return Void();
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ReportTraces {
|
|
|
|
ReportTraces() { g_traceProcessEvents = true; }
|
|
|
|
~ReportTraces() { g_traceProcessEvents = false; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct OnTestFailure {
|
|
|
|
std::shared_ptr<ResolverBug> bug;
|
|
|
|
OnTestFailure(std::shared_ptr<ResolverBug> bug) : bug(bug) {}
|
|
|
|
|
|
|
|
void operator()(StringRef, auto const& data, Error const&) {
|
|
|
|
BaseTraceEvent* trace = std::any_cast<BaseTraceEvent*>(data);
|
|
|
|
if (trace->getSeverity() == SevError) {
|
|
|
|
bug->bugFound = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ACTOR static Future<Void> driveWorkload(std::shared_ptr<ResolverBug> bug, int clientCount) {
|
|
|
|
state ReportTraces _;
|
|
|
|
state OnTestFailure onTestFailure(bug);
|
|
|
|
state ProcessEvents::Event ev("TraceEvent::TestFailure"_sr, onTestFailure);
|
|
|
|
loop {
|
|
|
|
bug->currentPhase = 1;
|
|
|
|
wait(waitForPhaseDone(bug, 1, clientCount));
|
|
|
|
SimBugInjector().enable();
|
|
|
|
bug->currentPhase = 2;
|
|
|
|
wait(waitForPhaseDone(bug, 2, clientCount));
|
|
|
|
SimBugInjector().disable();
|
|
|
|
bug->currentPhase = 3;
|
|
|
|
wait(waitForPhaseDone(bug, 3, clientCount));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ACTOR static Future<Void> _start(ResolverBugWorkload* self, Database cx) {
|
|
|
|
state Reference<TestWorkload> cycle;
|
2023-03-10 04:05:00 +08:00
|
|
|
state std::shared_ptr<ResolverBug> bug = SimBugInjector().get<ResolverBug>(ResolverBugID(), true);
|
|
|
|
loop {
|
2023-03-09 08:09:50 +08:00
|
|
|
wait(waitForPhase(bug, 1));
|
|
|
|
cycle = self->createCycle();
|
|
|
|
wait(cycle->setup(cx));
|
|
|
|
bug->cycleState[self->clientId] = 1;
|
|
|
|
wait(waitForPhase(bug, 2));
|
|
|
|
wait(cycle->start(cx));
|
|
|
|
bug->cycleState[self->clientId] = 2;
|
|
|
|
wait(waitForPhase(bug, 3));
|
|
|
|
wait(success(cycle->check(cx)));
|
|
|
|
bug->cycleState[self->clientId] = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 04:05:00 +08:00
|
|
|
ACTOR static Future<Void> onBug(std::shared_ptr<ResolverBug> bug) {
|
|
|
|
loop {
|
|
|
|
if (bug->bugFound) {
|
|
|
|
TraceEvent("NegativeTestSuccess").log();
|
|
|
|
return Void();
|
|
|
|
}
|
|
|
|
wait(delay(0.5));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<Void> start(const Database& cx) override {
|
|
|
|
std::vector<Future<Void>> futures;
|
|
|
|
auto bug = SimBugInjector().get<ResolverBug>(ResolverBugID(), true);
|
|
|
|
if (clientId == 0) {
|
|
|
|
futures.push_back(driveWorkload(bug, clientCount));
|
|
|
|
}
|
|
|
|
futures.push_back(_start(this, cx->clone()));
|
|
|
|
return onBug(bug) || waitForAll(futures);
|
|
|
|
}
|
2023-03-09 08:15:53 +08:00
|
|
|
Future<bool> check(Database const& cx) override { return true; };
|
2023-03-09 08:09:50 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
void getMetrics(std::vector<PerfMetric>& m) override {}
|
|
|
|
};
|
|
|
|
|
|
|
|
WorkloadFactory<ResolverBugWorkload> workloadFactory;
|
|
|
|
|
|
|
|
} // namespace
|