20 #include <ripple/basics/PerfLog.h>
21 #include <ripple/basics/random.h>
22 #include <ripple/beast/unit_test.h>
23 #include <ripple/beast/utility/Journal.h>
24 #include <ripple/json/json_reader.h>
25 #include <ripple/protocol/jss.h>
26 #include <ripple/rpc/impl/Handler.h>
32 #include <test/jtx/Env.h>
43 using path = boost::filesystem::path;
61 using namespace boost::filesystem;
68 if (!exists(dir) || !is_directory(dir) || !is_empty(dir))
84 using namespace boost::filesystem;
85 return temp_directory_path() /
"perf_log_test_dir";
91 return logDir() /
"perf_log.txt";
115 using namespace boost::filesystem;
128 secondSize = file_size(
path);
129 }
while (firstSize >= secondSize);
134 }
while (secondSize >= file_size(
path));
170 cur.isMember(jss::job) ? cur[jss::job].asString()
171 : cur[jss::method].asString());
179 [](
Cur const& lhs,
Cur const& rhs) {
180 if (lhs.dur != rhs.dur)
181 return (rhs.dur < lhs.dur);
182 return (lhs.name < rhs.name);
191 using namespace boost::filesystem;
196 BEAST_EXPECT(!exists(fixture.logFile()));
200 BEAST_EXPECT(fixture.stopSignaled ==
false);
201 BEAST_EXPECT(exists(fixture.logFile()));
208 if (!BEAST_EXPECT(!exists(fixture.logDir())))
215 fixture.logDir().c_str(), std::ios::out | std::ios::app);
216 if (!BEAST_EXPECT(nastyFile))
223 BEAST_EXPECT(fixture.stopSignaled ==
false);
225 BEAST_EXPECT(fixture.stopSignaled ==
true);
235 remove(fixture.logDir());
242 if (!BEAST_EXPECT(!exists(fixture.logDir())))
247 boost::system::error_code ec;
248 boost::filesystem::create_directories(fixture.logDir(), ec);
249 if (!BEAST_EXPECT(!ec))
252 auto fileWriteable = [](boost::filesystem::path
const& p) ->
bool {
253 return std::ofstream{p.c_str(), std::ios::out | std::ios::app}
257 if (!BEAST_EXPECT(fileWriteable(fixture.logFile())))
260 boost::filesystem::permissions(
262 perms::remove_perms | perms::owner_write | perms::others_write |
267 if (fileWriteable(fixture.logFile()))
269 log <<
"Unable to write protect file. Test skipped."
276 BEAST_EXPECT(fixture.stopSignaled ==
false);
278 BEAST_EXPECT(fixture.stopSignaled ==
true);
288 boost::filesystem::permissions(
290 perms::add_perms | perms::owner_write | perms::others_write |
301 auto perfLog{fixture.perfLog(withFile)};
313 ids.
reserve(labels.size() * 2);
330 for (
int labelIndex = 0; labelIndex < labels.size(); ++labelIndex)
332 for (
int idIndex = 0; idIndex < 2; ++idIndex)
336 labels[labelIndex], ids[(labelIndex * 2) + idIndex]);
342 BEAST_EXPECT(countersJson.size() == labels.size() + 1);
343 for (
auto& label : labels)
347 BEAST_EXPECT(counter[jss::duration_us] ==
"0");
348 BEAST_EXPECT(counter[jss::errored] ==
"0");
349 BEAST_EXPECT(counter[jss::finished] ==
"0");
350 BEAST_EXPECT(counter[jss::started] ==
"2");
353 Json::Value const& total{countersJson[jss::total]};
354 BEAST_EXPECT(total[jss::duration_us] ==
"0");
355 BEAST_EXPECT(total[jss::errored] ==
"0");
356 BEAST_EXPECT(total[jss::finished] ==
"0");
365 BEAST_EXPECT(currents.size() == labels.size() * 2);
368 for (
int i = 0; i < currents.size(); ++i)
370 BEAST_EXPECT(currents[i].name == labels[i / 2]);
371 BEAST_EXPECT(prevDur > currents[i].dur);
372 prevDur = currents[i].dur;
379 for (
int labelIndex = labels.size() - 1; labelIndex > 0; --labelIndex)
382 perfLog->rpcFinish(labels[labelIndex], ids[(labelIndex * 2) + 1]);
384 perfLog->rpcError(labels[labelIndex], ids[(labelIndex * 2) + 0]);
386 perfLog->rpcFinish(labels[0], ids[0 + 1]);
389 auto validateFinalCounters = [
this, &labels](
392 Json::Value const& jobQueue = countersJson[jss::job_queue];
394 BEAST_EXPECT(jobQueue.
size() == 0);
398 BEAST_EXPECT(rpc.
size() == labels.size() + 1);
406 BEAST_EXPECT(first[jss::duration_us] !=
"0");
407 BEAST_EXPECT(first[jss::errored] ==
"0");
408 BEAST_EXPECT(first[jss::finished] ==
"1");
409 BEAST_EXPECT(first[jss::started] ==
"2");
414 for (
int i = 1; i < labels.size(); ++i)
419 BEAST_EXPECT(dur != 0 && dur < prevDur);
421 BEAST_EXPECT(counter[jss::errored] ==
"1");
422 BEAST_EXPECT(counter[jss::finished] ==
"1");
423 BEAST_EXPECT(counter[jss::started] ==
"2");
428 BEAST_EXPECT(total[jss::duration_us] !=
"0");
430 jsonToUint64(total[jss::errored]) == labels.size() - 1);
431 BEAST_EXPECT(
jsonToUint64(total[jss::finished]) == labels.size());
433 jsonToUint64(total[jss::started]) == labels.size() * 2);
436 auto validateFinalCurrent = [
this,
439 Json::Value const& job_queue = currentJson[jss::jobs];
440 BEAST_EXPECT(job_queue.
isArray());
441 BEAST_EXPECT(job_queue.
size() == 0);
444 Json::Value const& methods = currentJson[jss::methods];
445 BEAST_EXPECT(methods.
size() == 1);
446 BEAST_EXPECT(methods.
isArray());
449 BEAST_EXPECT(only.
size() == 2);
451 BEAST_EXPECT(only[jss::duration_us] !=
"0");
452 BEAST_EXPECT(only[jss::method] == labels[0]);
456 validateFinalCounters(
perfLog->countersJson());
457 validateFinalCurrent(
perfLog->currentJson());
465 auto const fullPath = fixture.logFile();
469 BEAST_EXPECT(!exists(fullPath));
483 lastLine = std::move(line);
493 validateFinalCounters(parsedLastLine[jss::counters]);
494 validateFinalCurrent(parsedLastLine[jss::current_activities]);
506 auto perfLog{fixture.perfLog(withFile)};
517 : type(t), typeName(std::move(name))
526 for (
auto const& job : jobTypes)
535 for (
int i = 0; i < jobs.
size(); ++i)
537 perfLog->jobQueue(jobs[i].type);
539 perfLog->countersJson()[jss::job_queue]};
541 BEAST_EXPECT(jq_counters.size() == i + 2);
542 for (
int j = 0; j <= i; ++j)
546 Json::Value const& counter{jq_counters[jobs[j].typeName]};
547 BEAST_EXPECT(counter.size() == 5);
548 BEAST_EXPECT(counter[jss::queued] ==
"1");
549 BEAST_EXPECT(counter[jss::started] ==
"0");
550 BEAST_EXPECT(counter[jss::finished] ==
"0");
551 BEAST_EXPECT(counter[jss::queued_duration_us] ==
"0");
552 BEAST_EXPECT(counter[jss::running_duration_us] ==
"0");
557 BEAST_EXPECT(total.size() == 5);
558 BEAST_EXPECT(
jsonToUint64(total[jss::queued]) == i + 1);
559 BEAST_EXPECT(total[jss::started] ==
"0");
560 BEAST_EXPECT(total[jss::finished] ==
"0");
561 BEAST_EXPECT(total[jss::queued_duration_us] ==
"0");
562 BEAST_EXPECT(total[jss::running_duration_us] ==
"0");
568 BEAST_EXPECT(
current.size() == 2);
569 BEAST_EXPECT(
current.isMember(jss::jobs));
570 BEAST_EXPECT(
current[jss::jobs].size() == 0);
571 BEAST_EXPECT(
current.isMember(jss::methods));
572 BEAST_EXPECT(
current[jss::methods].size() == 0);
584 for (
int i = 0; i < jobs.
size(); ++i)
587 jobs[i].type,
microseconds{i + 1}, steady_clock::now(), i * 2);
592 perfLog->countersJson()[jss::job_queue]};
593 for (
int j = 0; j < jobs.
size(); ++j)
595 Json::Value const& counter{jq_counters[jobs[j].typeName]};
600 BEAST_EXPECT(counter[jss::started] ==
"2");
601 BEAST_EXPECT(queued_dur_us == j + 1);
605 BEAST_EXPECT(counter[jss::started] ==
"1");
606 BEAST_EXPECT(queued_dur_us == j + 1);
610 BEAST_EXPECT(counter[jss::started] ==
"0");
611 BEAST_EXPECT(queued_dur_us == 0);
614 BEAST_EXPECT(counter[jss::queued] ==
"1");
615 BEAST_EXPECT(counter[jss::finished] ==
"0");
616 BEAST_EXPECT(counter[jss::running_duration_us] ==
"0");
622 BEAST_EXPECT(
jsonToUint64(total[jss::started]) == (i * 2) + 1);
623 BEAST_EXPECT(total[jss::finished] ==
"0");
628 (((i * i) + 3 * i + 2) / 2));
629 BEAST_EXPECT(total[jss::running_duration_us] ==
"0");
644 BEAST_EXPECT(currents.size() == (i + 1) * 2);
647 for (
int j = 0; j <= i; ++j)
649 BEAST_EXPECT(currents[j * 2].name == jobs[j].typeName);
650 BEAST_EXPECT(prevDur > currents[j * 2].dur);
651 prevDur = currents[j * 2].dur;
653 BEAST_EXPECT(currents[(j * 2) + 1].name == jobs[j].typeName);
654 BEAST_EXPECT(prevDur > currents[(j * 2) + 1].dur);
655 prevDur = currents[(j * 2) + 1].dur;
660 for (
int i = jobs.
size() - 1; i >= 0; --i)
664 int const finished = ((jobs.
size() - i) * 2) - 1;
670 perfLog->countersJson()[jss::job_queue]};
671 for (
int j = 0; j < jobs.
size(); ++j)
673 Json::Value const& counter{jq_counters[jobs[j].typeName]};
678 BEAST_EXPECT(counter[jss::finished] ==
"0");
679 BEAST_EXPECT(running_dur_us == 0);
683 BEAST_EXPECT(counter[jss::finished] ==
"1");
684 BEAST_EXPECT(running_dur_us == ((jobs.
size() - j) * 2) - 1);
688 BEAST_EXPECT(counter[jss::finished] ==
"2");
689 BEAST_EXPECT(running_dur_us == ((jobs.
size() - j) * 4) - 1);
694 BEAST_EXPECT(queued_dur_us == j + 1);
695 BEAST_EXPECT(counter[jss::queued] ==
"1");
696 BEAST_EXPECT(counter[jss::started] ==
"2");
704 BEAST_EXPECT(
jsonToUint64(total[jss::finished]) == finished);
708 int const queuedDur = ((jobs.
size() * (jobs.
size() + 1)) / 2);
710 jsonToUint64(total[jss::queued_duration_us]) == queuedDur);
713 int const runningDur = ((finished * (finished + 1)) / 2);
727 BEAST_EXPECT(currents.size() == i * 2);
730 for (
int j = 0; j < i; ++j)
732 BEAST_EXPECT(currents[j * 2].name == jobs[j].typeName);
733 BEAST_EXPECT(prevDur > currents[j * 2].dur);
734 prevDur = currents[j * 2].dur;
736 BEAST_EXPECT(currents[(j * 2) + 1].name == jobs[j].typeName);
737 BEAST_EXPECT(prevDur > currents[(j * 2) + 1].dur);
738 prevDur = currents[(j * 2) + 1].dur;
743 auto validateFinalCounters = [
this,
748 BEAST_EXPECT(rpc.
size() == 0);
751 Json::Value const& jobQueue = countersJson[jss::job_queue];
752 for (
int i = jobs.
size() - 1; i >= 0; --i)
754 Json::Value const& counter{jobQueue[jobs[i].typeName]};
757 BEAST_EXPECT(running_dur_us == ((jobs.
size() - i) * 4) - 1);
761 BEAST_EXPECT(queued_dur_us == i + 1);
763 BEAST_EXPECT(counter[jss::queued] ==
"1");
764 BEAST_EXPECT(counter[jss::started] ==
"2");
765 BEAST_EXPECT(counter[jss::finished] ==
"2");
770 const int finished = jobs.
size() * 2;
772 BEAST_EXPECT(
jsonToUint64(total[jss::started]) == finished);
773 BEAST_EXPECT(
jsonToUint64(total[jss::finished]) == finished);
777 int const queuedDur = ((jobs.
size() * (jobs.
size() + 1)) / 2);
779 jsonToUint64(total[jss::queued_duration_us]) == queuedDur);
782 int const runningDur = ((finished * (finished + 1)) / 2);
784 jsonToUint64(total[jss::running_duration_us]) == runningDur);
787 auto validateFinalCurrent = [
this](
Json::Value const& currentJson) {
791 BEAST_EXPECT(j.
size() == 0);
794 Json::Value const& methods = currentJson[jss::methods];
795 BEAST_EXPECT(methods.
size() == 0);
796 BEAST_EXPECT(methods.
isArray());
800 validateFinalCounters(
perfLog->countersJson());
801 validateFinalCurrent(
perfLog->currentJson());
810 auto const fullPath = fixture.logFile();
814 BEAST_EXPECT(!exists(fullPath));
828 lastLine = std::move(line);
838 validateFinalCounters(parsedLastLine[jss::counters]);
839 validateFinalCurrent(parsedLastLine[jss::current_activities]);
853 auto perfLog{fixture.perfLog(withFile)};
863 auto iter{jobTypes.begin()};
866 jobType = iter->second.type();
867 jobTypeName = iter->second.name();
874 auto verifyCounters = [
this, jobTypeName](
880 BEAST_EXPECT(countersJson.
isObject());
881 BEAST_EXPECT(countersJson.
size() == 2);
883 BEAST_EXPECT(countersJson.
isMember(jss::rpc));
884 BEAST_EXPECT(countersJson[jss::rpc].isObject());
885 BEAST_EXPECT(countersJson[jss::rpc].size() == 0);
887 BEAST_EXPECT(countersJson.
isMember(jss::job_queue));
888 BEAST_EXPECT(countersJson[jss::job_queue].isObject());
889 BEAST_EXPECT(countersJson[jss::job_queue].size() == 1);
892 countersJson[jss::job_queue][jobTypeName]};
894 BEAST_EXPECT(job.isObject());
896 BEAST_EXPECT(
jsonToUint64(job[jss::started]) == started);
897 BEAST_EXPECT(
jsonToUint64(job[jss::finished]) == finished);
900 jsonToUint64(job[jss::queued_duration_us]) == queued_us);
902 jsonToUint64(job[jss::running_duration_us]) == running_us);
907 auto verifyEmptyCurrent = [
this](
Json::Value const& currentJson) {
908 BEAST_EXPECT(currentJson.isObject());
909 BEAST_EXPECT(currentJson.size() == 2);
911 BEAST_EXPECT(currentJson.isMember(jss::jobs));
912 BEAST_EXPECT(currentJson[jss::jobs].isArray());
913 BEAST_EXPECT(currentJson[jss::jobs].size() == 0);
915 BEAST_EXPECT(currentJson.isMember(jss::methods));
916 BEAST_EXPECT(currentJson[jss::methods].isArray());
917 BEAST_EXPECT(currentJson[jss::methods].size() == 0);
923 verifyCounters(
perfLog->countersJson(), 1, 0, 11, 0);
924 verifyEmptyCurrent(
perfLog->currentJson());
929 verifyCounters(
perfLog->countersJson(), 2, 0, 24, 0);
930 verifyEmptyCurrent(
perfLog->currentJson());
935 verifyCounters(
perfLog->countersJson(), 2, 1, 24, 17);
936 verifyEmptyCurrent(
perfLog->currentJson());
941 verifyCounters(
perfLog->countersJson(), 2, 2, 24, 36);
942 verifyEmptyCurrent(
perfLog->currentJson());
951 auto const fullPath = fixture.logFile();
955 BEAST_EXPECT(!exists(fullPath));
969 lastLine = std::move(line);
979 verifyCounters(parsedLastLine[jss::counters], 2, 2, 24, 36);
980 verifyEmptyCurrent(parsedLastLine[jss::current_activities]);
990 using namespace boost::filesystem;
993 BEAST_EXPECT(!exists(fixture.logDir()));
995 auto perfLog{fixture.perfLog(withFile)};
997 BEAST_EXPECT(fixture.stopSignaled ==
false);
1000 BEAST_EXPECT(!exists(fixture.logDir()));
1004 BEAST_EXPECT(exists(fixture.logFile()));
1005 BEAST_EXPECT(file_size(fixture.logFile()) == 0);
1013 decltype(file_size(fixture.logFile())) firstFileSize{0};
1016 BEAST_EXPECT(!exists(fixture.logDir()));
1020 firstFileSize = file_size(fixture.logFile());
1021 BEAST_EXPECT(firstFileSize > 0);
1032 BEAST_EXPECT(!exists(fixture.logDir()));
1036 BEAST_EXPECT(file_size(fixture.logFile()) > firstFileSize);