8161dde9b83ded19f65c8a1b629dcbd5930d470c
[pharext/pharext.org] / app / Controller / Github / Hook.php
1 <?php
2
3 namespace app\Controller\Github;
4
5 use app\Controller;
6 use app\Github\API;
7 use app\Model\Accounts;
8 use app\Web;
9 use http\Params;
10 use pharext\Task;
11 use pharext\SourceDir\Git;
12
13 require_once __DIR__."/../../../vendor/m6w6/pharext/src/pharext/Version.php";
14
15 class Hook implements Controller
16 {
17 private $app;
18 private $github;
19 private $accounts;
20
21 function __construct(Web $app, API $github, Accounts $accounts) {
22 $this->app = $app;
23 $this->github = $github;
24 $this->accounts = $accounts;
25 }
26
27 function __invoke(array $args = []) {
28 $request = $this->app->getRequest();
29 $response = $this->app->getResponse();
30
31 if (!($sig = $request->getHeader("X-Hub-Signature")) || !($evt = $request->getHeader("X-Github-Event"))) {
32 $response->setResponseCode(400);
33 $response->setContentType("message/http");
34 $response->getBody()->append($request);
35 } else {
36 $key = $this->github->getConfig()->client->secret;
37 foreach ((new Params($sig))->params as $algo => $mac) {
38 if ($mac["value"] !== hash_hmac($algo, $request->getBody(), $key)) {
39 $response->setResponseCode(403);
40 $response->getBody()->append("Invalid signature");
41 return;
42 }
43 }
44 }
45
46 switch ($evt) {
47 default:
48 $response->setResponseCode(202);
49 $response->getBody()->append("Not a configured event");
50 break;
51 case "ping";
52 $response->setResponseCode(204);
53 $response->setResponseStatus("PONG");
54 break;
55 case "create":
56 case "release":
57 if (($json = json_decode($request->getBody()))) {
58 $this->$evt($json);
59 } else {
60 $response->setResponseCode(415);
61 $response->setContentType($request->getHeader("Content-Type"));
62 $response->getBody()->append($request->getBody());
63 }
64 break;
65 }
66 }
67
68 function release($release) {
69 if ($release->action !== "published") {
70 $response = $this->app->getResponse();
71
72 $response->setResponseCode(202);
73 $response->getBody()->append("Not published");
74 return;
75 }
76
77 $this->uploadAssetForRelease($release->release, $release->repository);
78 }
79
80 private function uploadAssetForRelease($release, $repo) {
81 $this->setTokenForUser($repo->owner->login);
82 $asset = $this->createReleaseAsset($release, $repo);
83 $this->github->createReleaseAsset($release->upload_url, $asset, "application/phar", function($json) {
84 $response = $this->app->getResponse();
85 $response->setResponseCode(201);
86 $response->setHeader("Location", $json->url);
87 })->send();
88 }
89
90 private function createReleaseAsset($release, $repo) {
91 define("STDERR", fopen("/var/log/apache2/php_errors.log", "a"));
92 $source = (new Task\GitClone($repo->clone_url, $release->tag_name))->run();
93 $iterator = new Git($source);
94 $meta = [
95 "header" => sprintf("pharext v%s (c) Michael Wallner <mike@php.net>", \pharext\VERSION),
96 "version" => \pharext\VERSION,
97 "date" => date("Y-m-d"),
98 "name" => $repo->name,
99 "release" => $release->tag_name,
100 "license" => @file_get_contents(current(glob($iterator->getBaseDir()."/LICENSE*"))),
101 "stub" => "pharext_installer.php",
102 "type" => false ? "zend_extension" : "extension",
103 ];
104 $file = (new Task\PharBuild($iterator, $meta))->run();
105 return $file;
106 }
107
108 function create($create) {
109 if ($create->ref_type !== "tag") {
110 $response = $this->app->getResponse();
111
112 $response->setResponseCode(202);
113 $response->getBody()->append("Not a tag");
114 return;
115 }
116
117 $this->createReleaseFromTag($create->ref, $create->repository);
118 }
119
120 private function setTokenForUser($login) {
121 $relations = [
122 $this->accounts->getTokens()->getRelation("accounts"),
123 $this->accounts->getOwners()->getRelation("accounts")
124 ];
125 $tokens = $this->accounts->getTokens()->with($relations, [
126 "login=" => $login,
127 "tokens.authority=" => "github",
128 ]);
129
130 if (count($tokens)) {
131 $this->github->setToken($tokens->current()->token->get());
132 }
133 }
134
135 private function createReleaseFromTag($tag, $repo) {
136 $this->setTokenForUser($repo->owner->login);
137 $this->github->createRelease($repo->full_name, $tag, function($json) {
138 $response = $this->app->getResponse();
139 $response->setResponseCode(201);
140 $response->setHeader("Location", $json->url);
141 })->send();
142 }
143 }