1 | <?php |
||
13 | class Builder { |
||
14 | /** |
||
15 | * @var string |
||
16 | */ |
||
17 | protected $root; |
||
18 | /** |
||
19 | * @var string |
||
20 | */ |
||
21 | protected $target; |
||
22 | /** |
||
23 | * @param string $root |
||
24 | * @param string $target |
||
25 | */ |
||
26 | 14 | public function __construct ($root, $target) { |
|
27 | 14 | $this->root = $root; |
|
28 | 14 | $this->target = $target; |
|
29 | /** @noinspection MkdirRaceConditionInspection */ |
||
30 | 14 | @mkdir($target); |
|
31 | 14 | } |
|
32 | /** |
||
33 | * @param string[] $modules |
||
34 | * @param string[] $themes |
||
35 | * @param null|string $suffix |
||
36 | * |
||
37 | * @return string |
||
38 | * |
||
39 | * @throws \UnexpectedValueException |
||
40 | * @throws \BadMethodCallException |
||
41 | */ |
||
42 | 7 | public function core ($modules = [], $themes = [], $suffix = null) { |
|
43 | 7 | $suffix = $suffix ? "_$suffix" : ''; |
|
44 | 7 | $version = file_get_json("$this->root/modules/System/meta.json")['version']; |
|
45 | 7 | $target_file = "$this->target/CleverStyle_Framework_$version$suffix.phar.php"; |
|
46 | 7 | if (file_exists($target_file)) { |
|
47 | unlink($target_file); |
||
48 | } |
||
49 | 7 | $phar = new Phar($target_file); |
|
50 | 7 | unset($target_file); |
|
51 | 7 | $phar->startBuffering(); |
|
52 | 7 | $length = strlen("$this->root/"); |
|
53 | /** @noinspection ForeachSourceInspection */ |
||
54 | 7 | foreach (get_files_list("$this->root/install", false, 'f', '', true) as $file) { |
|
55 | 7 | $phar->addFile("$this->root/install/$file", $file); |
|
56 | } |
||
57 | 7 | unset($file); |
|
58 | 7 | $phar->addFile("$this->root/includes/img/logo.svg", 'logo.svg'); |
|
59 | /** |
||
60 | * Core files to be included into installation package |
||
61 | */ |
||
62 | 7 | $core_files = $this->get_core_files(); |
|
63 | /** |
||
64 | * Add modules that should be built-in into package |
||
65 | */ |
||
66 | 7 | $components_files = []; |
|
67 | 7 | $modules = $this->filter_and_add_components("$this->root/modules", $modules, $components_files); |
|
68 | 7 | $phar->addFromString('modules.json', _json_encode($modules)); |
|
69 | /** |
||
70 | * Add themes that should be built-in into package |
||
71 | */ |
||
72 | 7 | $themes = $this->filter_and_add_components("$this->root/themes", $themes, $components_files); |
|
73 | 7 | $phar->addFromString('themes.json', _json_encode($themes)); |
|
74 | /** |
||
75 | * Joining system and components files |
||
76 | */ |
||
77 | 7 | $core_files = array_merge($core_files, $components_files); |
|
78 | /** |
||
79 | * Addition of files into package |
||
80 | */ |
||
81 | 7 | foreach ($core_files as $index => &$file) { |
|
82 | 7 | $phar->addFile($file, "fs/$index"); |
|
83 | 7 | $file = substr($file, $length); |
|
84 | } |
||
85 | 7 | unset($index, $file); |
|
86 | /** |
||
87 | * Addition of separate files into package |
||
88 | */ |
||
89 | 7 | $phar->addFromString( |
|
90 | 7 | 'languages.json', |
|
91 | _json_encode( |
||
92 | array_merge( |
||
93 | 7 | _substr(get_files_list("$this->root/core/languages", '/^.*?\.php$/i', 'f'), 0, -4) ?: [], |
|
94 | 7 | _substr(get_files_list("$this->root/core/languages", '/^.*?\.json$/i', 'f'), 0, -5) ?: [] |
|
95 | ) |
||
96 | ) |
||
97 | ); |
||
98 | 7 | $phar->addFromString( |
|
99 | 7 | 'db_engines.json', |
|
100 | _json_encode( |
||
101 | 7 | _substr(get_files_list("$this->root/core/engines/DB", '/^[^_].*?\.php$/i', 'f'), 0, -4) |
|
102 | ) |
||
103 | ); |
||
104 | /** |
||
105 | * Fixation of system files list (without components files), it is needed for future system updating |
||
106 | */ |
||
107 | 7 | $phar->addFromString( |
|
108 | 7 | 'fs.json', |
|
109 | _json_encode( |
||
110 | array_flip( |
||
111 | array_diff( |
||
112 | $core_files, |
||
113 | _substr($components_files, $length) |
||
114 | ) |
||
115 | ) |
||
116 | ) |
||
117 | ); |
||
118 | 7 | unset($components_files, $length); |
|
119 | /** |
||
120 | * Addition of files, that are needed only for installation |
||
121 | */ |
||
122 | 7 | $phar->addFromString('fs/'.count($core_files), $this->get_htaccess()); |
|
123 | 7 | $core_files[] = '.htaccess'; |
|
124 | 7 | $phar->addFile("$this->root/config/main.php", 'fs/'.count($core_files)); |
|
125 | 7 | $core_files[] = 'config/main.php'; |
|
126 | 7 | $phar->addFile("$this->root/favicon.ico", 'fs/'.count($core_files)); |
|
127 | 7 | $core_files[] = 'favicon.ico'; |
|
128 | 7 | $phar->addFile("$this->root/.gitignore", 'fs/'.count($core_files)); |
|
129 | 7 | $core_files[] = '.gitignore'; |
|
130 | /** |
||
131 | * Flip array to have direct access to files by name during extracting and installation, and fixing of files list for installation |
||
132 | */ |
||
133 | 7 | $phar->addFromString( |
|
134 | 7 | 'fs_installer.json', |
|
135 | _json_encode( |
||
136 | array_flip($core_files) |
||
137 | ) |
||
138 | ); |
||
139 | 7 | unset($core_files); |
|
140 | /** |
||
141 | * Addition of supplementary files, that are needed directly for installation process: installer with GUI interface, readme, license, some additional |
||
142 | * information about available languages, themes, current version of system |
||
143 | */ |
||
144 | 7 | $phar->addFile("$this->root/license.txt", 'license.txt'); |
|
145 | 7 | $phar->addFile("$this->root/modules/System/meta.json", 'meta.json'); |
|
146 | 7 | $phar->setStub( |
|
147 | /** @lang PHP */ |
||
148 | <<<STUB |
||
149 | <?php |
||
150 | if (version_compare(PHP_VERSION, '5.6', '<')) { |
||
151 | echo 'CleverStyle Framework require PHP 5.6 or higher'; |
||
152 | return; |
||
153 | } |
||
154 | |||
155 | if (php_sapi_name() == 'cli') { |
||
156 | Phar::mapPhar('cleverstyle_framework.phar'); |
||
157 | include 'phar://cleverstyle_framework.phar/cli.php'; |
||
158 | } else { |
||
159 | Phar::webPhar(null, 'web.php'); |
||
160 | } |
||
161 | 7 | __HALT_COMPILER(); |
|
162 | STUB |
||
163 | ); |
||
164 | 7 | $phar->stopBuffering(); |
|
165 | 7 | return "Done! CleverStyle Framework $version"; |
|
166 | } |
||
167 | /** |
||
168 | * Get array of files |
||
169 | * |
||
170 | * @return string[] |
||
171 | */ |
||
172 | 7 | protected function get_core_files () { |
|
173 | $files_to_include = [ |
||
174 | 7 | "$this->root/modules/System", |
|
175 | 7 | "$this->root/blocks/.gitkept", |
|
176 | 7 | "$this->root/core", |
|
177 | 7 | "$this->root/custom", |
|
178 | 7 | "$this->root/includes", |
|
179 | 7 | "$this->root/storage/.htaccess", |
|
180 | 7 | "$this->root/storage/pcache/.htaccess", |
|
181 | 7 | "$this->root/storage/public/.htaccess", |
|
182 | 7 | "$this->root/storage/temp/.htaccess", |
|
183 | 7 | "$this->root/themes/CleverStyle", |
|
184 | 7 | "$this->root/bower.json", |
|
185 | 7 | "$this->root/cli", |
|
186 | 7 | "$this->root/composer.json", |
|
187 | 7 | "$this->root/composer.lock", |
|
188 | 7 | "$this->root/index.php", |
|
189 | 7 | "$this->root/license.txt", |
|
190 | 7 | "$this->root/package.json" |
|
191 | ]; |
||
192 | 7 | $files = []; |
|
193 | 7 | foreach ($files_to_include as $s) { |
|
194 | 7 | if (is_file($s)) { |
|
195 | 7 | $files[] = $s; |
|
196 | 7 | } elseif (is_dir($s)) { |
|
197 | /** @noinspection SlowArrayOperationsInLoopInspection */ |
||
198 | 7 | $files = array_merge( |
|
199 | $files, |
||
200 | 7 | get_files_list($s, false, 'f', true, true, false, false, true) |
|
201 | ); |
||
202 | } |
||
203 | } |
||
204 | 7 | return $files; |
|
205 | } |
||
206 | /** |
||
207 | * @param string $dir |
||
208 | * @param string[] $components |
||
209 | * @param string[] $components_files |
||
210 | * |
||
211 | * @return string[] |
||
212 | */ |
||
213 | 7 | protected function filter_and_add_components ($dir, $components, &$components_files) { |
|
223 | /** |
||
224 | * @param string $component_root |
||
225 | * @param string[] $files Array, where new files will be appended |
||
226 | * |
||
227 | * @return bool |
||
228 | */ |
||
229 | 1 | protected function get_component_files ($component_root, &$files) { |
|
230 | /** |
||
231 | * Do not allow building System module and CleverStyle theme |
||
232 | */ |
||
233 | 1 | if (in_array(basename($component_root), ['System', 'CleverStyle'])) { |
|
234 | 1 | return false; |
|
235 | } |
||
236 | /** |
||
237 | * Components without meta.json also not allowed |
||
238 | */ |
||
239 | 1 | if (!file_exists("$component_root/meta.json")) { |
|
240 | 1 | return false; |
|
241 | } |
||
242 | 1 | @unlink("$component_root/fs.json"); |
|
243 | 1 | $local_files = get_files_list($component_root, false, 'f', true, true, false, false, true); |
|
244 | 1 | $files = array_merge($files, $local_files); |
|
245 | 1 | file_put_json( |
|
246 | 1 | "$component_root/fs.json", |
|
247 | array_values( |
||
248 | _substr( |
||
249 | $local_files, |
||
250 | 1 | strlen("$component_root/") |
|
251 | ) |
||
252 | ) |
||
253 | ); |
||
254 | 1 | $files[] = "$component_root/fs.json"; |
|
255 | 1 | return true; |
|
256 | } |
||
257 | /** |
||
258 | * @return string |
||
259 | */ |
||
260 | 7 | protected function get_htaccess () { |
|
293 | /** |
||
294 | * @param string $module |
||
295 | * @param null|string $suffix |
||
296 | * |
||
297 | * @return string |
||
298 | */ |
||
299 | 4 | public function module ($module, $suffix = null) { |
|
305 | /** |
||
306 | * @param string $theme |
||
307 | * @param null|string $suffix |
||
308 | * |
||
309 | * @return string |
||
310 | */ |
||
311 | 3 | public function theme ($theme, $suffix = null) { |
|
380 | } |
||
381 |
Our type inference engine in quite powerful, but sometimes the code does not provide enough clues to go by. In these cases we request you to add a
@return
annotation as described here.