hoopoe/chirp.inc.php

341 lines
8.7 KiB
PHP
Raw Normal View History

2022-08-29 16:06:59 +03:00
<?php
namespace hoopoe;
use Exception;
function gitea_login($gitea_url, $user, $pass)
{
$token_field = "test_".time();
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, "https://git.bit5.ru/api/v1/users/$user/tokens");
curl_setopt($ch, CURLOPT_HTTPHEADER, array('Content-Type: application/json'));
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode(array("name" => $token_field)));
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);
curl_setopt($ch, CURLOPT_USERPWD, $user . ":" . $pass);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
$json = curl_exec($ch);
curl_close($ch);
$res = json_decode($json, true);
if(!isset($res['sha1']))
throw new Exception("Could not login to gitea");
return $res['sha1'];
}
function gitea_api_call($auth_token, $gitea_url, $api_path)
{
$api_url = "$gitea_url/api/v1/$api_path";
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $api_url);
curl_setopt($ch, CURLOPT_HEADER, true);
curl_setopt($ch, CURLOPT_HTTPHEADER,
array(
'Accept: application/json',
"Authorization: token $auth_token"
)
);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
$response = curl_exec($ch);
$header_size = curl_getinfo($ch, CURLINFO_HEADER_SIZE);
$header = substr($response, 0, $header_size);
$json = substr($response, $header_size);
$info = curl_getinfo($ch);
curl_close($ch);
if($info['http_code'] != 200)
return $info['http_code'];
return json_decode($json, true);
}
function gitea_get_tags($auth_token, $gitea_url, $repo)
{
$res = gitea_api_call($auth_token, $gitea_url, "repos/$repo/tags");
if($res === 404)
return array();
else if(is_numeric($res))
throw new Exception("Bad response: $res");
return $res;
}
function gitea_get_source($auth_token, $gitea_url, $repo, $path)
{
$res = gitea_api_call($auth_token, $gitea_url, "repos/$repo/contents/$path");
if(is_numeric($res))
throw new Exception("Bad response: $res");
if(!isset($res['content']))
throw new Exception("Could not get file at path '$path'");
return base64_decode($res['content']);
}
function gitea_get_forks($auth_token, $gitea_url, $repo)
{
$res = gitea_api_call($auth_token, $gitea_url, "repos/$repo/forks");
if(is_numeric($res))
throw new Exception("Bad response: $res");
return $res;
}
function gitea_get_commits($auth_token, $gitea_url, $repo)
{
$res = gitea_api_call($auth_token, $gitea_url, "repos/$repo/commits");
if(is_numeric($res))
throw new Exception("Bad response: $res");
return $res;
}
function gitea_get_commit($auth_token, $gitea_url, $repo, $sha)
{
$res = gitea_api_call($auth_token, $gitea_url, "repos/$repo/git/commits/$sha");
if($res === 404)
return null;
else if(is_numeric($res))
throw new Exception("Bad response: $res");
else
return $res;
}
function github_get_tags($github_repo)
{
$api_url = "https://api.github.com/repos/$github_repo/tags";
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $api_url);
curl_setopt($ch, CURLOPT_HEADER, true);
curl_setopt($ch, CURLOPT_USERAGENT, "test");
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
$response = curl_exec($ch);
$header_size = curl_getinfo($ch, CURLINFO_HEADER_SIZE);
$header = substr($response, 0, $header_size);
$json = substr($response, $header_size);
$info = curl_getinfo($ch);
curl_close($ch);
if($info['http_code'] != 200)
return $info['http_code'];
return json_decode($json, true);
}
function _is_composer_git_repo(array $repos, $repo)
{
foreach($repos as $item)
{
if($item['type'] == 'git' && $item['url'] == $repo)
return true;
}
return false;
}
function _is_composer_github_package(array $repos, $repo, &$github_repo)
{
foreach($repos as $item)
{
if($item['type'] == 'package' &&
$item['package']['name'] == $repo &&
preg_match('~https?://github.com/([^/]+)/([^/]+)~', $item['package']['dist']['url'], $matches))
{
$github_repo = $matches[1].'/'.$matches[2];
return true;
}
}
return false;
}
interface IReporter
{
public function write($msg);
public function flush();
}
class CliReporter implements IReporter
{
public function write($msg)
{
echo $msg;
}
public function flush()
{
}
}
class SlackReporter implements IReporter
{
private $buffer = '';
public function write($msg)
{
$this->buffer .= $msg;
}
public function flush()
{
slack_post($this->buffer);
}
}
class Reporters implements IReporter
{
private $all = array();
public function add(IReporter $r)
{
$this->all[] = $r;
}
public function write($msg)
{
foreach($this->all as $r)
$r->write($msg);
}
public function flush()
{
foreach($this->all as $r)
$r->flush();
}
}
function gitea_check_repo(
$auth_token,
$gitea_url,
$base_repo,
array $opts
)
{
$r = new Reporters();
$r->add(new SlackReporter());
$r->add(new CliReporter());
$r->write("************************* Chirping about repo '$base_repo' *************************\n");
$commits = gitea_get_commits($auth_token, $gitea_url, $base_repo);
if($opts['check_forks'])
{
$forks = gitea_get_forks($auth_token, $gitea_url, $base_repo);
$behind_forks = array();
$r->write("== Checking forks ==\n");
foreach($forks as $fork)
{
$repo_fork = $fork['full_name'];
foreach($commits as $commit)
{
$status = gitea_get_commit($auth_token, $gitea_url, $repo_fork, $commit['sha']);
if($status === null)
{
$behind_forks[] = $repo_fork;
break;
}
//early exit if commit exists
else if(isset($status['sha']))
{
break;
}
}
}
$r->write("Behind forks: " . implode(", ", $behind_forks) . "\n");
}
if($opts['check_composer'])
{
$r->write("== Checking Composer manifest ==\n");
$composer_json = gitea_get_source($auth_token, $gitea_url, $base_repo, "composer/composer.json");
$composer_arr = json_decode($composer_json, true);
foreach($composer_arr['require'] as $repo => $version)
{
if(strpos($repo, "bit/") !== 0)
continue;
if(_is_composer_git_repo($composer_arr['repositories'], "$gitea_url/$repo"))
{
$tags = gitea_get_tags($auth_token, $gitea_url, $repo);
if(isset($tags[0]))
{
$last_tag_info = $tags[0];
if(ltrim($last_tag_info["name"], 'v') != ltrim($version, 'v'))
$r->write("Package '$repo': current $version, latest {$last_tag_info["name"]}\n");
}
else
$r->write("Could not fetch tags for $repo\n");
}
else if(_is_composer_github_package($composer_arr['repositories'], $repo, $github_repo))
{
$tags = github_get_tags($github_repo);
if(isset($tags[0]))
{
$last_tag_info = $tags[0];
if(ltrim($last_tag_info["name"], 'v') != ltrim($version, 'v'))
$r->write("Package '$repo': current $version, latest {$last_tag_info["name"]}\n");
}
}
}
}
if($opts['check_upm'])
{
$r->write("== Checking UPM manifest ==\n");
$upm_json = gitea_get_source($auth_token, $gitea_url, $base_repo, "unity/Packages/manifest.json");
$upm_arr = json_decode($upm_json, true);
foreach($upm_arr['dependencies'] as $name => $version)
{
if(strpos($name, "com.bitgames.") !== 0)
continue;
$name = str_replace("com.bitgames.ecs", "com.bitgames.leoecs", $name);
$name = str_replace("com.bitgames.", "", $name);
$name = str_replace(".", "-", $name);
$repo = 'bit-upm/' . $name;
$tags = gitea_get_tags($auth_token, $gitea_url, $repo);
if(isset($tags[0]))
{
$last_tag_info = $tags[0];
if(ltrim($last_tag_info["name"], 'v') != ltrim($version, 'v'))
$r->write("Package '$repo': current $version, latest {$last_tag_info["name"]}\n");
}
else
$r->write("Could not fetch tags for $repo\n");
}
}
$r->flush();
}
function slack_post($message, $fields = array())
{
$ch = curl_init("https://slack.com/api/chat.postMessage");
$fields["token"] = 'xoxb-141599046048-4017237103457-fsUiANhvIcuScMRkrTB7obEU';
$fields["channel"] = "#tech";
$fields["text"] = $message;
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($fields));
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
$result = curl_exec($ch);
curl_close($ch);
return json_decode($result, true);
}