Skip to content

Commit

Permalink
[Core] Add performance profiler logger
Browse files Browse the repository at this point in the history
This adds a new Profiler class that can be used for logging performance
bottlenecks in LORIS. To use it, add a call to Profiler::checkpoint("message")
and if the profiler is enabled in the configuration module (default, disabled)
it will print to the error logs a timestamp, how long the request has taken
so far, the message, max memory usage, and any new memory usage since the last
checkpoint call.

This can be used to debug where requests are slowing down. The checkpoints added
in this PR are the places that I used to optimize the dataquery module.
  • Loading branch information
Dave MacFarlane committed Oct 2, 2024
1 parent 403fd0c commit f824745
Show file tree
Hide file tree
Showing 9 changed files with 98 additions and 3 deletions.
1 change: 1 addition & 0 deletions SQL/0000-00-03-ConfigTables.sql
Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,7 @@ INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, Label, Or
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'database_log_level', 'Verbosity of database logging', 1, 0, 'log_level', ID, 'Database Log Level', 3 FROM ConfigSettings WHERE Name='logs';
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'request_log_level', 'Verbosity of HTTP request logs', 1, 0, 'log_level', ID, 'HTTP Request Log Level', 3 FROM ConfigSettings WHERE Name='logs';
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'exception_log_level', 'Verbosity of PHP exception logging', 1, 0, 'log_level', ID, 'Exception Log Level', 3 FROM ConfigSettings WHERE Name='logs';
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'profiler_log_level', 'Verbosity of performance profiler logging', 1, 0, 'log_level', ID, 'Profiler Log Level', 3 FROM ConfigSettings WHERE Name='logs';

INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, Label, OrderNumber) VALUES ('aws', 'Settings related to AWS services', 1, 0, 'AWS Settings', 13);
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'AWS_S3_Endpoint', 'Endpoint to use for accessing files stored in S3. Endpoint or region are required for S3 support.', 1, 0, 'text', ID, 'AWS S3 Endpoint', 3 FROM ConfigSettings WHERE Name='aws';
Expand Down
2 changes: 2 additions & 0 deletions SQL/New_patches/2024-profiler_log.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
INSERT INTO ConfigSettings (Name, Description, Visible, AllowMultiple, DataType, Parent, Label, OrderNumber) SELECT 'profiler_log_level', 'Verbosity of performance profiler logging', 1, 0, 'log_level', ID, 'Profiler Log Level', 3 FROM ConfigSettings WHERE Name='logs';

1 change: 1 addition & 0 deletions htdocs/index.php
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
$client = new \NDB_Client;
$client->initialize();

Profiler::checkpoint("Profiler started");
// Middleware that happens on every request. This doesn't include
// any authentication middleware, because that's done dynamically
// based on the module router, depending on if the module is public.
Expand Down
3 changes: 3 additions & 0 deletions modules/dataquery/php/endpoints/queries/query/count.class.inc
Original file line number Diff line number Diff line change
Expand Up @@ -58,9 +58,12 @@ class Count extends \LORIS\Http\Endpoint
public function countResults(\User $user, int $queryID) : ResponseInterface
{
try {
\Profiler::checkpoint("Creating query");
$query = new \LORIS\dataquery\Query($this->loris, $queryID);

\Profiler::checkpoint("Matching candidates");
$candidates = \iterator_to_array($query->matchCandidates($user));
\Profiler::checkpoint("Returning count");
return new \LORIS\Http\Response\JSON\OK(
[
'count' => count($candidates),
Expand Down
8 changes: 8 additions & 0 deletions modules/dataquery/php/query.class.inc
Original file line number Diff line number Diff line change
Expand Up @@ -545,8 +545,10 @@ class Query implements \LORIS\StudyEntities\AccessibleResource,
$criteria = $this->getCriteria();

if ($criteria) {
\Profiler::checkpoint("Matching with criteria");
$candidates = $this->_candidateMatches($criteria);
} else {
\Profiler::checkpoint("Matching all candidates");
$candidates = $this->_allCandidates();
}
return $this->_filterInaccessibleCandidates($candidates, $user);
Expand All @@ -564,6 +566,7 @@ class Query implements \LORIS\StudyEntities\AccessibleResource,
array $candIDs,
\User $user,
) : iterable {
\Profiler::checkpoint("Filtering inaccessible candidates");
if (count($candIDs) == 0) {
return [];
}
Expand Down Expand Up @@ -640,6 +643,7 @@ class Query implements \LORIS\StudyEntities\AccessibleResource,
yield $canddata->CandID;
}
}
\Profiler::checkpoint("Finished filtering candidates");
}

/**
Expand Down Expand Up @@ -678,7 +682,9 @@ class Query implements \LORIS\StudyEntities\AccessibleResource,
}

foreach ($criteria['group'] as $crit) {
\Profiler::checkpoint("Checking groups");
if (isset($crit['group'])) {
\Profiler::checkpoint("Subgroup");
$subresults = $this->_candidateMatches($crit);
if (is_array($subresults)) {
$results[] = $subresults;
Expand Down Expand Up @@ -747,12 +753,14 @@ class Query implements \LORIS\StudyEntities\AccessibleResource,
$visitlist = $query['visits'];
}

\Profiler::checkpoint("Calling engine get matches");
$matches = $engine->getCandidateMatches($term, $visitlist);
if (is_array($matches)) {
$results[] = $matches;
} else {
$results[] = iterator_to_array($matches);
}
\Profiler::checkpoint("Done group");
}
$combinedresult = [];
if (count($results) == 1) {
Expand Down
4 changes: 3 additions & 1 deletion modules/instruments/php/instrumentqueryengine.class.inc
Original file line number Diff line number Diff line change
Expand Up @@ -104,14 +104,16 @@ class InstrumentQueryEngine implements \LORIS\Data\Query\QueryEngine
throw new \DomainException("Field for unknown instrument");
}

$inst = \NDB_BVL_Instrument::factory($this->loris, $testname);
$inst = \NDB_BVL_Instrument::factory($this->loris, $testname);
\Profiler::checkpoint("After NDB_BVL_Instrument::factory for $testname");
$values = $inst->bulkLoadInstanceData(
null,
$term,
);
foreach ($values as $match) {
yield $match->getCandID();
}
\Profiler::checkpoint("Finished bulk load for $testname");
return;
}

Expand Down
10 changes: 10 additions & 0 deletions php/libraries/LogSettings.class.inc
Original file line number Diff line number Diff line change
Expand Up @@ -51,4 +51,14 @@ class LogSettings
{
return $this->levels['exception_log_level'] ?? "none";
}

/**
* Get the current verbosity level for performance profile logging
*
* @return string
*/
function getProfilerLogLevel() : string
{
return $this->levels['profile_log_level'] ?? "none";
}
}
7 changes: 5 additions & 2 deletions php/libraries/NDB_BVL_Instrument.class.inc
Original file line number Diff line number Diff line change
Expand Up @@ -2299,7 +2299,7 @@ abstract class NDB_BVL_Instrument extends NDB_Page
JOIN instrument_data ON (flag.DataID=instrument_data.ID)
JOIN test_names tn ON (tn.ID=flag.TestID)
$where";

\Profiler::checkpoint("Running query (JSON data): " . $query);
$jsondata = $db->pselect($query, $params);
foreach ($jsondata as $row) {
$newinst = clone $this;
Expand All @@ -2316,6 +2316,7 @@ abstract class NDB_BVL_Instrument extends NDB_Page
) ?? $this->defaultInstanceData();
yield $newinst;
}
\Profiler::checkpoint("Finished query");
return;
} else {
$conditions = ['tn.test_name=:tn'];
Expand Down Expand Up @@ -2353,7 +2354,8 @@ abstract class NDB_BVL_Instrument extends NDB_Page
JOIN test_names tn ON (tn.ID=f.TestID)
JOIN session ON (f.SessionID=session.ID)
$where";
$data = $db->pselect($query, $params);
\Profiler::checkpoint("Running query (table instrument): " . $query);
$data = $db->pselect($query, $params);
foreach ($data as $row) {
$newinst = clone $this;

Expand All @@ -2367,6 +2369,7 @@ abstract class NDB_BVL_Instrument extends NDB_Page

yield $newinst;
}
\Profiler::checkpoint("Finished query");
return;
}
}
Expand Down
65 changes: 65 additions & 0 deletions php/libraries/Profiler.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
<?php declare(strict_types=1);

use \Psr\Log\LoggerInterface;

/**
* Class to consistently log profiling related information to LORIS
* logs so that performance issues can be more easily debugged.
*
* @license http://www.gnu.org/licenses/gpl-3.0.txt GPLv3
*/
class Profiler
{
protected \DateTimeImmutable $startTime;
protected ?\DateTimeImmutable $lastCheckpoint;
protected int $lastMemory;
protected LoggerInterface $logger;

/**
* Private constructor that is only called from checkpoint, so
* that a singleton is used.
*/
private function __construct()
{

$this->startTime = new \DateTimeImmutable();
$this->lastMemory = memory_get_peak_usage();

$loglevel = \NDB_Config::singleton()->getLogSettings()->getProfilerLogLevel();
if ($loglevel !== "none") {
$this->logger = new \LORIS\Log\ErrorLogLogger(\Psr\Log\LogLevel::DEBUG);
} else {
$this->logger = new \Psr\Log\NullLogger();
}
}

/**
* Adds a checkpoint to the error logs that displays the timing since
* the initial checkpoint, and any increase in the PHP max memory usage
* since the last checkpoint.
*
* @param string $msg - A message to display with the stats in the log
*
* @return void
*/
public static function checkpoint(string $msg) : void
{
static $profiler = new Profiler();
$currentMemory = memory_get_peak_usage();
$now = new \DateTimeImmutable();
$profiler->logger->debug(
"["
. $now->format("U")
. " ("
. $now->diff($profiler->startTime)->format("%s") . "s)] "
. $msg
. ' (Max memory: '
. $currentMemory
. ', since last checkpoint: '
. ($currentMemory - $profiler->lastMemory)
. ')'
);
$profiler->lastCheckpoint = new \DateTimeImmutable();
$profiler->lastMemory = $currentMemory;
}
}

0 comments on commit f824745

Please sign in to comment.