1ac998477d3493898fa98716caf903653358f271
[pharext/pharext.org] / app / Controller / Github / Hook / Receive.php
1 <?php
2
3 namespace app\Controller\Github\Hook;
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\Metadata;
12 use pharext\SourceDir;
13
14 class Receive implements Controller
15 {
16 private $app;
17 private $github;
18 private $accounts;
19
20 function __construct(Web $app, API $github, Accounts $accounts) {
21 $this->app = $app;
22 $this->github = $github;
23 $this->accounts = $accounts;
24 }
25
26 function __invoke(array $args = []) {
27 $request = $this->app->getRequest();
28 $response = $this->app->getResponse();
29
30 if (!($sig = $request->getHeader("X-Hub-Signature")) || !($evt = $request->getHeader("X-Github-Event"))) {
31 $response->setResponseCode(400);
32 $response->setContentType("message/http");
33 $response->getBody()->append($request);
34 } else {
35 $key = $this->github->getConfig()->client->secret;
36 foreach ((new Params($sig))->params as $algo => $mac) {
37 if ($mac["value"] !== hash_hmac($algo, $request->getBody(), $key)) {
38 $response->setResponseCode(403);
39 $response->getBody()->append("Invalid signature");
40 return;
41 }
42 }
43 }
44
45 switch ($evt) {
46 default:
47 $response->setResponseCode(202);
48 $response->getBody()->append("Not a configured event");
49 break;
50 case "ping";
51 $response->setResponseCode(204);
52 $response->setResponseStatus("PONG");
53 break;
54 case "create":
55 case "release":
56 if (($json = json_decode($request->getBody()))) {
57 $this->$evt($json);
58 } else {
59 $response->setResponseCode(415);
60 $response->setContentType($request->getHeader("Content-Type"));
61 $response->getBody()->append($request->getBody());
62 }
63 break;
64 }
65 }
66
67 function release($release) {
68 if ($release->action !== "published") {
69 $response = $this->app->getResponse();
70
71 $response->setResponseCode(202);
72 $response->getBody()->append("Not published");
73 return;
74 }
75 if (!empty($release->release->assets)) {
76 foreach ($release->release->assets as $asset) {
77 if ($asset->content_type === "application/phar") {
78 /* we've already uploaded the asset when we created the release */
79 return;
80 }
81 }
82 }
83
84 $this->uploadAssetForRelease($release->release, $release->repository)->send();
85 }
86
87 private function uploadAssetForRelease($release, $repo) {
88 $this->setTokenForUser($repo->owner->login);
89 return $this->github->listHooks($repo->full_name, function($hooks) use($release, $repo) {
90 $repo->hooks = $hooks;
91 $asset = $this->createReleaseAsset($release, $repo);
92 $name = sprintf("%s-%s.ext.phar", $repo->name, $release->tag_name);
93 $url = uri_template($release->upload_url, compact("name"));
94 $this->github->createReleaseAsset($url, $asset, "application/phar", function($json) use($release, $repo) {
95 if ($release->draft) {
96 $this->github->publishRelease($repo->full_name, $release->id, $release->tag_name, function($json) {
97 $response = $this->app->getResponse();
98 $response->setResponseCode(201);
99 $response->setHeader("Location", $json->url);
100 });
101 } else {
102 $response = $this->app->getResponse();
103 $response->setResponseCode(201);
104 $response->setHeader("Location", $json->url);
105 }
106 });
107 });
108 }
109
110 private function createReleaseAsset($release, $repo) {
111 $hook = $this->github->checkRepoHook($repo);
112 $dir = (new Task\GitClone($repo->clone_url, $release->tag_name))->run();
113 if (!empty($hook->config->pecl)) {
114 $src = new SoureDir\Pecl($dir);
115 } else {
116 $src = new SourceDir\Git($dir);
117 }
118 $meta = Metadata::all() + [
119 "name" => $repo->name,
120 "release" => $release->tag_name,
121 "license" => $src->getLicense(),
122 "stub" => "pharext_installer.php",
123 "type" => !empty($hook->config->zend) ? "zend_extension" : "extension",
124 ];
125 $file = (new Task\PharBuild($src, $meta))->run();
126 return $file;
127 }
128
129 function create($create) {
130 if ($create->ref_type !== "tag") {
131 $response = $this->app->getResponse();
132
133 $response->setResponseCode(202);
134 $response->getBody()->append("Not a tag");
135 return;
136 }
137
138 $this->createReleaseFromTag($create->ref, $create->repository)->send();
139 }
140
141 private function setTokenForUser($login) {
142 $relations = [
143 $this->accounts->getTokens()->getRelation("accounts"),
144 $this->accounts->getOwners()->getRelation("accounts")
145 ];
146 $tokens = $this->accounts->getTokens()->with($relations, [
147 "login=" => $login,
148 "tokens.authority=" => "github",
149 ]);
150
151 if (count($tokens)) {
152 $this->github->setToken($tokens->current()->token->get());
153 }
154 }
155
156 private function createReleaseFromTag($tag, $repo) {
157 $this->setTokenForUser($repo->owner->login);
158 return $this->github->createRelease($repo->full_name, $tag, function($json) use($repo) {
159 $this->uploadAssetForRelease($json, $repo);
160 });
161 }
162 }