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