c53878c614123c602cad9aad19140821a03831f6
[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\SourceDir;
12
13 require_once __DIR__."/../../../../vendor/m6w6/pharext/src/pharext/Version.php";
14
15 class Receive 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 $this->github->listHooks($repo->full_name, function($hooks) use($release, $repo) {
83 $repo->hooks = $hooks;
84 $asset = $this->createReleaseAsset($release, $repo);
85 $name = sprintf("%s-%s.ext.phar", $repo->name, $release->tag_name);
86 $url = uri_template($release->upload_url, compact("name"));
87 $this->github->createReleaseAsset($url, $asset, "application/phar", function($json) {
88 $response = $this->app->getResponse();
89 $response->setResponseCode(201);
90 $response->setHeader("Location", $json->url);
91 });
92 })->send();
93 }
94
95 private function createReleaseAsset($release, $repo) {
96 $source = (new Task\GitClone($repo->clone_url, $release->tag_name))->run();
97 $iterator = new SourceDir\Git($source);
98 $meta = [
99 "header" => sprintf("pharext v%s (c) Michael Wallner <mike@php.net>", \pharext\VERSION),
100 "version" => \pharext\VERSION,
101 "date" => date("Y-m-d"),
102 "name" => $repo->name,
103 "release" => $release->tag_name,
104 "license" => @file_get_contents(current(glob($iterator->getBaseDir()."/LICENSE*"))),
105 "stub" => "pharext_installer.php",
106 "type" => false ? "zend_extension" : "extension",
107 ];
108 $file = (new Task\PharBuild($iterator, $meta))->run();
109 return $file;
110 }
111
112 function create($create) {
113 if ($create->ref_type !== "tag") {
114 $response = $this->app->getResponse();
115
116 $response->setResponseCode(202);
117 $response->getBody()->append("Not a tag");
118 return;
119 }
120
121 $this->createReleaseFromTag($create->ref, $create->repository);
122 }
123
124 private function setTokenForUser($login) {
125 $relations = [
126 $this->accounts->getTokens()->getRelation("accounts"),
127 $this->accounts->getOwners()->getRelation("accounts")
128 ];
129 $tokens = $this->accounts->getTokens()->with($relations, [
130 "login=" => $login,
131 "tokens.authority=" => "github",
132 ]);
133
134 if (count($tokens)) {
135 $this->github->setToken($tokens->current()->token->get());
136 }
137 }
138
139 private function createReleaseFromTag($tag, $repo) {
140 $this->setTokenForUser($repo->owner->login);
141 $this->github->createRelease($repo->full_name, $tag, function($json) {
142 $response = $this->app->getResponse();
143 $response->setResponseCode(201);
144 $response->setHeader("Location", $json->url);
145 })->send();
146 }
147 }