fix for bug #64343
[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 $task = new Task\StreamFetch($source, function($bytes_pct) {
148 $this->info(" %3d%% [%s>%s] \r%s",
149 floor($bytes_pct*100),
150 str_repeat("=", round(50*$bytes_pct)),
151 str_repeat(" ", round(50*(1-$bytes_pct))),
152 $bytes_pct == 1 ? "\n":""
153 );
154 });
155 }
156 $local = $task->run($this->verbosity());
157
158 $this->cleanup[] = new Task\Cleanup($local);
159 return $local;
160 }
161
162 /**
163 * Extract local archive
164 * @param stirng $source
165 * @return string extracted directory
166 */
167 private function extract($source) {
168 try {
169 $task = new Task\Extract($source);
170 $dest = $task->run($this->verbosity());
171 } catch (\Exception $e) {
172 if (false === strpos($e->getMessage(), "checksum mismatch")) {
173 throw $e;
174 }
175 $dest = (new Task\PaxFixup($source))->run($this->verbosity());
176 }
177
178 $this->cleanup[] = new Task\Cleanup($dest);
179 return $dest;
180 }
181
182 /**
183 * Localize a possibly remote source
184 * @param string $source
185 * @return string local source directory
186 */
187 private function localize($source) {
188 if (!stream_is_local($source)) {
189 $source = $this->download($source);
190 $this->cleanup[] = new Task\Cleanup($source);
191 }
192 $source = realpath($source);
193 if (!is_dir($source)) {
194 $source = $this->extract($source);
195 $this->cleanup[] = new Task\Cleanup($source);
196
197 if (!$this->args->git) {
198 $source = (new Task\PeclFixup($source))->run($this->verbosity());
199 }
200 }
201 return $source;
202 }
203
204 /**
205 * Load the source dir
206 * @throws \pharext\Exception
207 */
208 private function loadSource(){
209 if ($this->args["source"]) {
210 $source = $this->localize($this->args["source"]);
211
212 if ($this->args["pecl"]) {
213 $this->source = new SourceDir\Pecl($source);
214 } elseif ($this->args["git"]) {
215 $this->source = new SourceDir\Git($source);
216 } elseif (is_file("$source/parext_package.php")) {
217 $this->source = include "$source/pharext_package.php";
218 } else {
219 $this->source = new SourceDir\Basic($source);
220 }
221
222 if (!$this->source instanceof SourceDir) {
223 throw new Exception("Unknown source dir $source");
224 }
225
226 foreach ($this->source->getPackageInfo() as $key => $val) {
227 $this->args->$key = $val;
228 }
229 }
230 }
231
232 /**
233 * Creates the extension phar
234 */
235 private function createPackage() {
236 try {
237 $meta = array_merge($this->metadata(), [
238 "date" => date("Y-m-d"),
239 "name" => $this->args->name,
240 "release" => $this->args->release,
241 "license" => @file_get_contents(current(glob($this->source->getBaseDir()."/LICENSE*"))),
242 "stub" => "pharext_installer.php",
243 "type" => $this->args->zend ? "zend_extension" : "extension",
244 ]);
245 $file = (new Task\PharBuild($this->source, $meta))->run($this->verbosity());
246 } catch (\Exception $e) {
247 $this->error("%s\n", $e->getMessage());
248 exit(self::EBUILD);
249 }
250
251 try {
252 if ($this->args->sign) {
253 $this->info("Using private key to sign phar ...\n");
254 $pass = (new Task\Askpass)->run($this->verbosity());
255 $sign = new Task\PharSign($file, $this->args->sign, $pass);
256 $pkey = $sign->run($this->verbosity());
257 }
258
259 } catch (\Exception $e) {
260 $this->error("%s\n", $e->getMessage());
261 exit(self::ESIGN);
262 }
263
264 if ($this->args->gzip) {
265 try {
266 $gzip = (new Task\PharCompress($file, Phar::GZ))->run();
267 $move = new Task\PharRename($gzip, $this->args->dest, $this->args->name ."-". $this->args->release);
268 $name = $move->run($this->verbosity());
269
270 $this->info("Created gzipped phar %s\n", $name);
271
272 if ($this->args->sign) {
273 $sign = new Task\PharSign($name, $this->args->sign, $pass);
274 $sign->run($this->verbosity())->exportPublicKey($name.".pubkey");
275 }
276
277 } catch (\Exception $e) {
278 $this->warn("%s\n", $e->getMessage());
279 }
280 }
281
282 if ($this->args->bzip) {
283 try {
284 $bzip = (new Task\PharCompress($file, Phar::BZ2))->run();
285 $move = new Task\PharRename($bzip, $this->args->dest, $this->args->name ."-". $this->args->release);
286 $name = $move->run($this->verbosity());
287
288 $this->info("Created bzipped phar %s\n", $name);
289
290 if ($this->args->sign) {
291 $sign = new Task\PharSign($name, $this->args->sign, $pass);
292 $sign->run($this->verbosity())->exportPublicKey($name.".pubkey");
293 }
294
295 } catch (\Exception $e) {
296 $this->warn("%s\n", $e->getMessage());
297 }
298 }
299
300 try {
301 $move = new Task\PharRename($file, $this->args->dest, $this->args->name ."-". $this->args->release);
302 $name = $move->run($this->verbosity());
303
304 $this->info("Created executable phar %s\n", $name);
305
306 if (isset($pkey)) {
307 $pkey->exportPublicKey($name.".pubkey");
308 }
309
310 } catch (\Exception $e) {
311 $this->error("%s\n", $e->getMessage());
312 exit(self::EBUILD);
313 }
314 }
315 }