cd1d43681d2d9a1167615c170365175cc3c5c9b4
[pharext/pharext] / src / pharext / Packager.php
1 <?php
2
3 namespace pharext;
4
5 use Phar;
6 use pharext\Cli\Args as CliArgs;
7 use pharext\Cli\Command as CliCommand;
8 use pharext\Exception;
9
10 /**
11 * The extension packaging command executed by bin/pharext
12 */
13 class Packager implements Command
14 {
15 use CliCommand;
16
17 /**
18 * Extension source directory
19 * @var pharext\SourceDir
20 */
21 private $source;
22
23 /**
24 * Cleanups
25 * @var array
26 */
27 private $cleanup = [];
28
29 /**
30 * Create the command
31 */
32 public function __construct() {
33 $this->args = new CliArgs([
34 ["h", "help", "Display this help",
35 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG|CliArgs::HALT],
36 ["v", "verbose", "More output",
37 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
38 ["q", "quiet", "Less output",
39 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
40 ["n", "name", "Extension name",
41 CliArgs::REQUIRED|CliArgs::SINGLE|CliArgs::REQARG],
42 ["r", "release", "Extension release version",
43 CliArgs::REQUIRED|CliArgs::SINGLE|CliArgs::REQARG],
44 ["s", "source", "Extension source directory",
45 CliArgs::REQUIRED|CliArgs::SINGLE|CliArgs::REQARG],
46 ["g", "git", "Use `git ls-tree` to determine file list",
47 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
48 ["p", "pecl", "Use PECL package.xml to determine file list, name and release",
49 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
50 ["d", "dest", "Destination directory",
51 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::REQARG,
52 "."],
53 ["z", "gzip", "Create additional PHAR compressed with gzip",
54 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
55 ["Z", "bzip", "Create additional PHAR compressed with bzip",
56 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG],
57 ["S", "sign", "Sign the PHAR with a private key",
58 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::REQARG],
59 ["E", "zend", "Mark as Zend Extension",
60 CliArgs::OPTIONAL|CliARgs::SINGLE|CliArgs::NOARG],
61 [null, "signature", "Show pharext signature",
62 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG|CliArgs::HALT],
63 [null, "license", "Show pharext license",
64 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG|CliArgs::HALT],
65 [null, "version", "Show pharext version",
66 CliArgs::OPTIONAL|CliArgs::SINGLE|CliArgs::NOARG|CliArgs::HALT],
67 ]);
68 }
69
70 /**
71 * Perform cleaniup
72 */
73 function __destruct() {
74 foreach ($this->cleanup as $cleanup) {
75 $cleanup->run();
76 }
77 }
78
79 /**
80 * @inheritdoc
81 * @see \pharext\Command::run()
82 */
83 public function run($argc, array $argv) {
84 $errs = [];
85 $prog = array_shift($argv);
86 foreach ($this->args->parse(--$argc, $argv) as $error) {
87 $errs[] = $error;
88 }
89
90 if ($this->args["help"]) {
91 $this->header();
92 $this->help($prog);
93 exit;
94 }
95 try {
96 foreach (["signature", "license", "version"] as $opt) {
97 if ($this->args[$opt]) {
98 printf("%s\n", $this->metadata($opt));
99 exit;
100 }
101 }
102 } catch (\Exception $e) {
103 $this->error("%s\n", $e->getMessage());
104 exit(self::EARGS);
105 }
106
107 try {
108 /* source needs to be evaluated before CliArgs validation,
109 * so e.g. name and version can be overriden and CliArgs
110 * does not complain about missing arguments
111 */
112 $this->loadSource();
113 } catch (\Exception $e) {
114 $errs[] = $e->getMessage();
115 }
116
117 foreach ($this->args->validate() as $error) {
118 $errs[] = $error;
119 }
120
121 if ($errs) {
122 if (!$this->args["quiet"]) {
123 $this->header();
124 }
125 foreach ($errs as $err) {
126 $this->error("%s\n", $err);
127 }
128 printf("\n");
129 if (!$this->args["quiet"]) {
130 $this->help($prog);
131 }
132 exit(self::EARGS);
133 }
134
135 $this->createPackage();
136 }
137
138 /**
139 * Download remote source
140 * @param string $source
141 * @return string local source
142 */
143 private function download($source) {
144 if ($this->args->git) {
145 $task = new Task\GitClone($source);
146 } else {
147 /* print newline only once */
148 $done = false;
149 $task = new Task\StreamFetch($source, function($bytes_pct) use(&$done) {
150 if (!$done) {
151 $this->info(" %3d%% [%s>%s] \r",
152 floor($bytes_pct*100),
153 str_repeat("=", round(50*$bytes_pct)),
154 str_repeat(" ", round(50*(1-$bytes_pct)))
155 );
156 if ($bytes_pct == 1) {
157 $done = true;
158 printf("\n");
159 }
160 }
161 });
162 }
163 $local = $task->run($this->verbosity());
164
165 $this->cleanup[] = new Task\Cleanup($local);
166 return $local;
167 }
168
169 /**
170 * Extract local archive
171 * @param stirng $source
172 * @return string extracted directory
173 */
174 private function extract($source) {
175 try {
176 $task = new Task\Extract($source);
177 $dest = $task->run($this->verbosity());
178 } catch (\Exception $e) {
179 if (false === strpos($e->getMessage(), "checksum mismatch")) {
180 throw $e;
181 }
182 $dest = (new Task\PaxFixup($source))->run($this->verbosity());
183 }
184
185 $this->cleanup[] = new Task\Cleanup($dest);
186 return $dest;
187 }
188
189 /**
190 * Localize a possibly remote source
191 * @param string $source
192 * @return string local source directory
193 */
194 private function localize($source) {
195 if (!stream_is_local($source)) {
196 $source = $this->download($source);
197 $this->cleanup[] = new Task\Cleanup($source);
198 }
199 $source = realpath($source);
200 if (!is_dir($source)) {
201 $source = $this->extract($source);
202 $this->cleanup[] = new Task\Cleanup($source);
203
204 if (!$this->args->git) {
205 $source = (new Task\PeclFixup($source))->run($this->verbosity());
206 }
207 }
208 return $source;
209 }
210
211 /**
212 * Load the source dir
213 * @throws \pharext\Exception
214 */
215 private function loadSource(){
216 if ($this->args["source"]) {
217 $source = $this->localize($this->args["source"]);
218
219 if ($this->args["pecl"]) {
220 $this->source = new SourceDir\Pecl($source);
221 } elseif ($this->args["git"]) {
222 $this->source = new SourceDir\Git($source);
223 } elseif (is_file("$source/pharext_package.php")) {
224 $this->source = include "$source/pharext_package.php";
225 } else {
226 $this->source = new SourceDir\Basic($source);
227 }
228
229 if (!$this->source instanceof SourceDir) {
230 throw new Exception("Unknown source dir $source");
231 }
232
233 foreach ($this->source->getPackageInfo() as $key => $val) {
234 $this->args->$key = $val;
235 }
236 }
237 }
238
239 /**
240 * Creates the extension phar
241 */
242 private function createPackage() {
243 try {
244 $meta = array_merge($this->metadata(), [
245 "date" => date("Y-m-d"),
246 "name" => $this->args->name,
247 "release" => $this->args->release,
248 "license" => @file_get_contents(current(glob($this->source->getBaseDir()."/LICENSE*"))),
249 "stub" => "pharext_installer.php",
250 "type" => $this->args->zend ? "zend_extension" : "extension",
251 ]);
252 $file = (new Task\PharBuild($this->source, $meta))->run($this->verbosity());
253 } catch (\Exception $e) {
254 $this->error("%s\n", $e->getMessage());
255 exit(self::EBUILD);
256 }
257
258 try {
259 if ($this->args->sign) {
260 $this->info("Using private key to sign phar ...\n");
261 $pass = (new Task\Askpass)->run($this->verbosity());
262 $sign = new Task\PharSign($file, $this->args->sign, $pass);
263 $pkey = $sign->run($this->verbosity());
264 }
265
266 } catch (\Exception $e) {
267 $this->error("%s\n", $e->getMessage());
268 exit(self::ESIGN);
269 }
270
271 if ($this->args->gzip) {
272 try {
273 $gzip = (new Task\PharCompress($file, Phar::GZ))->run();
274 $move = new Task\PharRename($gzip, $this->args->dest, $this->args->name ."-". $this->args->release);
275 $name = $move->run($this->verbosity());
276
277 $this->info("Created gzipped phar %s\n", $name);
278
279 if ($this->args->sign) {
280 $sign = new Task\PharSign($name, $this->args->sign, $pass);
281 $sign->run($this->verbosity())->exportPublicKey($name.".pubkey");
282 }
283
284 } catch (\Exception $e) {
285 $this->warn("%s\n", $e->getMessage());
286 }
287 }
288
289 if ($this->args->bzip) {
290 try {
291 $bzip = (new Task\PharCompress($file, Phar::BZ2))->run();
292 $move = new Task\PharRename($bzip, $this->args->dest, $this->args->name ."-". $this->args->release);
293 $name = $move->run($this->verbosity());
294
295 $this->info("Created bzipped phar %s\n", $name);
296
297 if ($this->args->sign) {
298 $sign = new Task\PharSign($name, $this->args->sign, $pass);
299 $sign->run($this->verbosity())->exportPublicKey($name.".pubkey");
300 }
301
302 } catch (\Exception $e) {
303 $this->warn("%s\n", $e->getMessage());
304 }
305 }
306
307 try {
308 $move = new Task\PharRename($file, $this->args->dest, $this->args->name ."-". $this->args->release);
309 $name = $move->run($this->verbosity());
310
311 $this->info("Created executable phar %s\n", $name);
312
313 if (isset($pkey)) {
314 $pkey->exportPublicKey($name.".pubkey");
315 }
316
317 } catch (\Exception $e) {
318 $this->error("%s\n", $e->getMessage());
319 exit(self::EBUILD);
320 }
321 }
322 }