2019-03-23 17:49:37 +01:00
|
|
|
|
<?php
|
|
|
|
|
namespace Psalm\Plugin;
|
|
|
|
|
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function array_filter;
|
2019-07-05 22:24:00 +02:00
|
|
|
|
use function curl_close;
|
|
|
|
|
use function curl_exec;
|
|
|
|
|
use function curl_getinfo;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function curl_init;
|
|
|
|
|
use function curl_setopt;
|
|
|
|
|
use const CURLINFO_HEADER_OUT;
|
2019-07-05 22:24:00 +02:00
|
|
|
|
use const CURLOPT_HTTPHEADER;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use const CURLOPT_POST;
|
|
|
|
|
use const CURLOPT_POSTFIELDS;
|
2019-07-05 22:24:00 +02:00
|
|
|
|
use const CURLOPT_RETURNTRANSFER;
|
|
|
|
|
use function function_exists;
|
|
|
|
|
use function fwrite;
|
|
|
|
|
use function json_encode;
|
|
|
|
|
use function parse_url;
|
|
|
|
|
use const PHP_EOL;
|
|
|
|
|
use const PHP_URL_SCHEME;
|
|
|
|
|
use Psalm\Codebase;
|
|
|
|
|
use Psalm\SourceControl\SourceControlInfo;
|
|
|
|
|
use const STDERR;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function strlen;
|
|
|
|
|
use function var_export;
|
2019-03-23 17:49:37 +01:00
|
|
|
|
|
2019-03-31 20:02:30 +02:00
|
|
|
|
class Shepherd implements \Psalm\Plugin\Hook\AfterAnalysisInterface
|
2019-03-23 17:49:37 +01:00
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* Called after analysis is complete
|
2019-07-05 22:24:00 +02:00
|
|
|
|
*
|
2019-03-23 17:49:37 +01:00
|
|
|
|
* @param array<int, array{severity: string, line_from: int, line_to: int, type: string, message: string,
|
|
|
|
|
* file_name: string, file_path: string, snippet: string, from: int, to: int,
|
|
|
|
|
* snippet_from: int, snippet_to: int, column_from: int, column_to: int, selected_text: string}> $issues
|
|
|
|
|
*
|
|
|
|
|
* @return void
|
|
|
|
|
*/
|
|
|
|
|
public static function afterAnalysis(
|
|
|
|
|
Codebase $codebase,
|
|
|
|
|
array $issues,
|
|
|
|
|
array $build_info,
|
|
|
|
|
SourceControlInfo $source_control_info = null
|
|
|
|
|
) {
|
2019-03-31 20:18:22 +02:00
|
|
|
|
if (!function_exists('curl_init')) {
|
2019-05-30 16:30:41 +02:00
|
|
|
|
fwrite(STDERR, 'No curl found, cannot send data to ' . $codebase->config->shepherd_host . PHP_EOL);
|
2019-07-05 22:24:00 +02:00
|
|
|
|
|
2019-03-31 20:18:22 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 17:06:21 +01:00
|
|
|
|
if ($source_control_info instanceof \Psalm\SourceControl\Git\GitInfo && $build_info) {
|
2019-03-23 17:49:37 +01:00
|
|
|
|
$data = [
|
|
|
|
|
'build' => $build_info,
|
|
|
|
|
'git' => $source_control_info->toArray(),
|
|
|
|
|
'issues' => array_filter(
|
|
|
|
|
$issues,
|
|
|
|
|
function (array $i) : bool {
|
|
|
|
|
return $i['severity'] === 'error';
|
|
|
|
|
}
|
|
|
|
|
),
|
2019-07-05 22:24:00 +02:00
|
|
|
|
'coverage' => $codebase->analyzer->getTotalTypeCoverage($codebase),
|
2019-03-23 17:49:37 +01:00
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
$payload = json_encode($data);
|
|
|
|
|
|
2019-03-31 20:02:30 +02:00
|
|
|
|
$base_address = $codebase->config->shepherd_host;
|
2019-03-27 22:01:05 +01:00
|
|
|
|
|
|
|
|
|
if (parse_url($base_address, PHP_URL_SCHEME) === null) {
|
|
|
|
|
$base_address = 'https://' . $base_address;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-23 17:49:37 +01:00
|
|
|
|
// Prepare new cURL resource
|
2019-04-02 03:34:24 +02:00
|
|
|
|
$ch = curl_init($base_address . '/hooks/psalm');
|
2019-03-23 17:49:37 +01:00
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
|
|
|
|
curl_setopt($ch, CURLINFO_HEADER_OUT, true);
|
|
|
|
|
curl_setopt($ch, CURLOPT_POST, true);
|
|
|
|
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $payload);
|
|
|
|
|
|
|
|
|
|
// Set HTTP Header for POST request
|
|
|
|
|
curl_setopt(
|
|
|
|
|
$ch,
|
|
|
|
|
CURLOPT_HTTPHEADER,
|
|
|
|
|
[
|
|
|
|
|
'Content-Type: application/json',
|
2019-07-05 22:24:00 +02:00
|
|
|
|
'Content-Length: ' . strlen($payload),
|
2019-03-23 17:49:37 +01:00
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Submit the POST request
|
2019-03-28 17:06:21 +01:00
|
|
|
|
$return = curl_exec($ch);
|
|
|
|
|
|
|
|
|
|
if ($return !== '') {
|
2019-05-30 16:30:41 +02:00
|
|
|
|
fwrite(STDERR, 'Error with Psalm Shepherd:' . PHP_EOL);
|
2019-04-01 01:20:05 +02:00
|
|
|
|
|
|
|
|
|
if ($return === false) {
|
2019-07-16 21:34:02 +02:00
|
|
|
|
fwrite(STDERR, self::getCurlErrorMessage($ch) . PHP_EOL);
|
2019-04-01 01:20:05 +02:00
|
|
|
|
} else {
|
|
|
|
|
echo $return . PHP_EOL;
|
2019-04-01 02:47:33 +02:00
|
|
|
|
echo 'Git args: ' . var_export($source_control_info->toArray(), true) . PHP_EOL;
|
|
|
|
|
echo 'CI args: ' . var_export($build_info, true) . PHP_EOL;
|
2019-04-01 01:20:05 +02:00
|
|
|
|
}
|
2019-03-28 17:06:21 +01:00
|
|
|
|
}
|
2019-03-23 17:49:37 +01:00
|
|
|
|
|
|
|
|
|
// Close cURL session handle
|
|
|
|
|
curl_close($ch);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-16 21:34:02 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param resource $ch
|
|
|
|
|
*/
|
|
|
|
|
public static function getCurlErrorMessage($ch) : string
|
|
|
|
|
{
|
|
|
|
|
/** @var array */
|
|
|
|
|
$curl_info = curl_getinfo($ch);
|
|
|
|
|
|
|
|
|
|
if (($curl_info['ssl_verify_result'] ?? 0) !== 0) {
|
|
|
|
|
switch ($curl_info['ssl_verify_result']) {
|
|
|
|
|
case 2:
|
|
|
|
|
return 'unable to get issuer certificate';
|
|
|
|
|
case 3:
|
|
|
|
|
return 'unable to get certificate CRL';
|
|
|
|
|
case 4:
|
|
|
|
|
return 'unable to decrypt certificate’s signature';
|
|
|
|
|
case 5:
|
|
|
|
|
return 'unable to decrypt CRL’s signature';
|
|
|
|
|
case 6:
|
|
|
|
|
return 'unable to decode issuer public key';
|
|
|
|
|
case 7:
|
|
|
|
|
return 'certificate signature failure';
|
|
|
|
|
case 8:
|
|
|
|
|
return 'CRL signature failure';
|
|
|
|
|
case 9:
|
|
|
|
|
return 'certificate is not yet valid';
|
|
|
|
|
case 10:
|
|
|
|
|
return 'certificate has expired';
|
|
|
|
|
case 11:
|
|
|
|
|
return 'CRL is not yet valid';
|
|
|
|
|
case 12:
|
|
|
|
|
return 'CRL has expired';
|
|
|
|
|
case 13:
|
|
|
|
|
return 'format error in certificate’s notBefore field';
|
|
|
|
|
case 14:
|
|
|
|
|
return 'format error in certificate’s notAfter field';
|
|
|
|
|
case 15:
|
|
|
|
|
return 'format error in CRL’s lastUpdate field';
|
|
|
|
|
case 16:
|
|
|
|
|
return 'format error in CRL’s nextUpdate field';
|
|
|
|
|
case 17:
|
|
|
|
|
return 'out of memory';
|
|
|
|
|
case 18:
|
|
|
|
|
return 'self signed certificate';
|
|
|
|
|
case 19:
|
|
|
|
|
return 'self signed certificate in certificate chain';
|
|
|
|
|
case 20:
|
|
|
|
|
return 'unable to get local issuer certificate';
|
|
|
|
|
case 21:
|
|
|
|
|
return 'unable to verify the first certificate';
|
|
|
|
|
case 22:
|
|
|
|
|
return 'certificate chain too long';
|
|
|
|
|
case 23:
|
|
|
|
|
return 'certificate revoked';
|
|
|
|
|
case 24:
|
|
|
|
|
return 'invalid CA certificate';
|
|
|
|
|
case 25:
|
|
|
|
|
return 'path length constraint exceeded';
|
|
|
|
|
case 26:
|
|
|
|
|
return 'unsupported certificate purpose';
|
|
|
|
|
case 27:
|
|
|
|
|
return 'certificate not trusted';
|
|
|
|
|
case 28:
|
|
|
|
|
return 'certificate rejected';
|
|
|
|
|
case 29:
|
|
|
|
|
return 'subject issuer mismatch';
|
|
|
|
|
case 30:
|
|
|
|
|
return 'authority and subject key identifier mismatch';
|
|
|
|
|
case 31:
|
|
|
|
|
return 'authority and issuer serial number mismatch';
|
|
|
|
|
case 32:
|
|
|
|
|
return 'key usage does not include certificate signing';
|
|
|
|
|
case 50:
|
|
|
|
|
return 'application verification failure';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return var_export(curl_getinfo($ch), true);
|
|
|
|
|
}
|
2019-03-23 17:49:37 +01:00
|
|
|
|
}
|