Commit 4445a198031c612601fd0432a16bdcf9ee4d577a
Merge remote-tracking branch 'origin/master'
# Conflicts: # composer.json
Showing
24 changed files
with
1671 additions
and
535 deletions
Show diff stats
.gitignore
1 | +<?php | ||
2 | + | ||
3 | + /** | ||
4 | + * Class Initializer | ||
5 | + * | ||
6 | + * Class to initialize Artbox application | ||
7 | + */ | ||
8 | + class Initializer | ||
9 | + { | ||
10 | + private $params = []; | ||
11 | + private $root; | ||
12 | + private $envs = []; | ||
13 | + private $environment; | ||
14 | + private $db = []; | ||
15 | + private static $instance; | ||
16 | + | ||
17 | + const VERSION = '1.0'; | ||
18 | + const CALLBACKS = [ | ||
19 | + 'setCookieValidationKey', | ||
20 | + 'setWritable', | ||
21 | + 'setExecutable', | ||
22 | + 'createSymlink', | ||
23 | + ]; | ||
24 | + | ||
25 | + /** | ||
26 | + * Initialize application | ||
27 | + */ | ||
28 | + public static function initialize() | ||
29 | + { | ||
30 | + $instance = self::getInstance(); | ||
31 | + $instance->defineEnvironment(); | ||
32 | + $instance->startInitialization(); | ||
33 | + if ($instance->askInitDb()) { | ||
34 | + do { | ||
35 | + $instance->defineDb(); | ||
36 | + } while (!$instance->databaseCheck()); | ||
37 | + $instance->migrate(); | ||
38 | + $instance->createUser(); | ||
39 | + $instance->congratulate(); | ||
40 | + } | ||
41 | + } | ||
42 | + | ||
43 | + /** | ||
44 | + * Get instance of Initializer | ||
45 | + * | ||
46 | + * @return mixed | ||
47 | + */ | ||
48 | + public static function getInstance(): Initializer | ||
49 | + { | ||
50 | + if (empty( self::$instance )) { | ||
51 | + self::$instance = new self(); | ||
52 | + } | ||
53 | + return self::$instance; | ||
54 | + } | ||
55 | + | ||
56 | + /** | ||
57 | + * Initializer private constructor. You can get Singleton with getInstance() | ||
58 | + * | ||
59 | + * Availiable options: | ||
60 | + * * --env - One of available environments | ||
61 | + * * --overwrite - Do overwrite all files without confirmation | ||
62 | + * * --dbinit - Initialize database after app initialization | ||
63 | + * * --dbtype - Database type. Available: postgresql, mysql | ||
64 | + * * --host - Database host, default to 127.0.0.1 | ||
65 | + * * --port - Database port used by postgresql, default to 5432 | ||
66 | + * * --schema - Database schema for postresql, default to public | ||
67 | + * * --dbname - Database name, required | ||
68 | + * * --username - Database username, required | ||
69 | + * * --password - Database password, required | ||
70 | + * * --migrationPath - Migration path, default to vendor/artweb/artbox-core/migrations | ||
71 | + * * --migrate - Whether to migrate, default to apply, set no to skip migration | ||
72 | + * * --user_username - Username for user creation | ||
73 | + * * --user_email - Email for user creation | ||
74 | + * * --user_password - Password for user creation | ||
75 | + * * --user - Whether to create user, default to yes, set no to skip creation | ||
76 | + * * --defaultuser - Whether to use default user creation | ||
77 | + * * --o - Webpage to open after intallaction process | ||
78 | + * * --fun - Run HOLLYWOOD BABY!!! | ||
79 | + * | ||
80 | + * @see Initializer::getInstance() | ||
81 | + */ | ||
82 | + private function __construct() | ||
83 | + { | ||
84 | + echo $this->formatMessage( | ||
85 | + "\tArtbox Application Initialization Tool v" . self::VERSION, | ||
86 | + [ 'bold' ] | ||
87 | + ) . "\n\n"; | ||
88 | + } | ||
89 | + | ||
90 | + /** | ||
91 | + * Define environment for an application | ||
92 | + * | ||
93 | + * @return string | ||
94 | + */ | ||
95 | + private function defineEnvironment() | ||
96 | + { | ||
97 | + $envName = $this->getParamValue('env'); | ||
98 | + $envs = $this->getEnvironmentNames(); | ||
99 | + if (empty( $envName ) || $envName === '1') { | ||
100 | + do { | ||
101 | + $envName = $this->askEnvironment($envs); | ||
102 | + echo "\n Initialize the application under '{$envName}' environment? [yes|no] "; | ||
103 | + $answer = trim(fgets(STDIN)); | ||
104 | + } while (strncasecmp($answer, 'y', 1)); | ||
105 | + } else { | ||
106 | + if (!in_array($envName, $envs)) { | ||
107 | + $this->printError("Wrong environment name, available list: " . implode(', ', $envs)); | ||
108 | + exit( 1 ); | ||
109 | + } | ||
110 | + } | ||
111 | + $this->environment = $this->envs[ $envName ]; | ||
112 | + return $this->environment; | ||
113 | + } | ||
114 | + | ||
115 | + /** | ||
116 | + * Start actual application initialization with files overwriting | ||
117 | + */ | ||
118 | + private function startInitialization() | ||
119 | + { | ||
120 | + echo "\n Start initialization\n"; | ||
121 | + echo " =====================\n"; | ||
122 | + | ||
123 | + $this->rewriteFiles(); | ||
124 | + $this->callback(); | ||
125 | + echo "\n App initialization completed\n"; | ||
126 | + echo " ==============================\n"; | ||
127 | + } | ||
128 | + | ||
129 | + /** | ||
130 | + * Ask whether to init databse. If dbinit param is set, this step will be skipped. | ||
131 | + * | ||
132 | + * @return bool | ||
133 | + */ | ||
134 | + private function askInitDb(): bool | ||
135 | + { | ||
136 | + echo " Start database initialization\n"; | ||
137 | + $params = $this->getParams(); | ||
138 | + if (!isset( $params[ 'dbinit' ] )) { | ||
139 | + do { | ||
140 | + echo $this->formatMessage("\n Init the database? [yes|no] ", [ 'bold' ]); | ||
141 | + $answer = trim(fgets(STDIN)); | ||
142 | + } while (( strncasecmp($answer, 'y', 1) !== 0 ) && ( strncasecmp($answer, 'n', 1) !== 0 )); | ||
143 | + if (strncasecmp($answer, 'n', 1) == 0) { | ||
144 | + $this->noDatabaseMessage(); | ||
145 | + } | ||
146 | + } | ||
147 | + return true; | ||
148 | + } | ||
149 | + | ||
150 | + /** | ||
151 | + * Define which database driver to use. Available options: postgresql, mysql | ||
152 | + */ | ||
153 | + private function defineDb() | ||
154 | + { | ||
155 | + $params = $this->getParams(); | ||
156 | + if (isset( $params[ 'dbinit' ] )) { | ||
157 | + $this->validateConnection(); | ||
158 | + } else { | ||
159 | + $answer = ''; | ||
160 | + do { | ||
161 | + if (!empty( $answer )) { | ||
162 | + $this->printError("Incorrect database type. Try again or write 'quit' to exit"); | ||
163 | + } | ||
164 | + echo "\n Which database do you want to use? [postgresql|mysql] "; | ||
165 | + $answer = trim(fgets(STDIN)); | ||
166 | + | ||
167 | + } while (( strncasecmp($answer, 'q', 1) !== 0 ) && ( !in_array( | ||
168 | + $answer, | ||
169 | + [ | ||
170 | + 'postgresql', | ||
171 | + 'mysql', | ||
172 | + 'p', | ||
173 | + 'm', | ||
174 | + ] | ||
175 | + ) )); | ||
176 | + if (strncasecmp($answer, 'q', 1) === 0) { | ||
177 | + $this->noDatabaseMessage(); | ||
178 | + } | ||
179 | + if (strncasecmp($answer, 'p', 1) === 0) { | ||
180 | + $answer = 'postgresql'; | ||
181 | + } else { | ||
182 | + $answer = 'mysql'; | ||
183 | + } | ||
184 | + if ($answer == 'postgresql') { | ||
185 | + $this->initPostgres(); | ||
186 | + } else { | ||
187 | + $this->initMysql(); | ||
188 | + } | ||
189 | + } | ||
190 | + $this->initDb(); | ||
191 | + } | ||
192 | + | ||
193 | + /** | ||
194 | + * Get parsed params | ||
195 | + * | ||
196 | + * @see Initializer::parseParams() | ||
197 | + * | ||
198 | + * @return array | ||
199 | + */ | ||
200 | + private function getParams(): array | ||
201 | + { | ||
202 | + if (empty( $this->params )) { | ||
203 | + $this->parseParams(); | ||
204 | + } | ||
205 | + return $this->params; | ||
206 | + } | ||
207 | + | ||
208 | + /** | ||
209 | + * Get params from input string | ||
210 | + * For example | ||
211 | + * | ||
212 | + * **<code>php init test --param1=value1 --param2=value2</code>** | ||
213 | + * | ||
214 | + * will get | ||
215 | + * | ||
216 | + * <code>** | ||
217 | + * [ | ||
218 | + * test, | ||
219 | + * value1, | ||
220 | + * value2 | ||
221 | + * ] | ||
222 | + * </code> | ||
223 | + * | ||
224 | + * @return array | ||
225 | + */ | ||
226 | + private function parseParams(): array | ||
227 | + { | ||
228 | + $rawParams = []; | ||
229 | + if (isset( $_SERVER[ 'argv' ] )) { | ||
230 | + $rawParams = $_SERVER[ 'argv' ]; | ||
231 | + array_shift($rawParams); | ||
232 | + } | ||
233 | + | ||
234 | + $params = []; | ||
235 | + foreach ($rawParams as $param) { | ||
236 | + if (preg_match('/^--(\w+)(=(.*))?$/', $param, $matches)) { | ||
237 | + $name = $matches[ 1 ]; | ||
238 | + $params[ $name ] = isset( $matches[ 3 ] ) ? $matches[ 3 ] : true; | ||
239 | + } else { | ||
240 | + $params[] = $param; | ||
241 | + } | ||
242 | + } | ||
243 | + $this->params = $params; | ||
244 | + return $this->params; | ||
245 | + } | ||
246 | + | ||
247 | + /** | ||
248 | + * Get value of input param. Empty string if not exist. | ||
249 | + * | ||
250 | + * @param string $name | ||
251 | + * | ||
252 | + * @return string | ||
253 | + */ | ||
254 | + private function getParamValue(string $name): string | ||
255 | + { | ||
256 | + $params = $this->getParams(); | ||
257 | + if (isset( $params[ $name ] ) && !empty( $params[ $name ] )) { | ||
258 | + return $params[ $name ]; | ||
259 | + } else { | ||
260 | + return ''; | ||
261 | + } | ||
262 | + } | ||
263 | + | ||
264 | + /** | ||
265 | + * Get project root directory according to file system | ||
266 | + * | ||
267 | + * @return mixed | ||
268 | + */ | ||
269 | + private function getRoot(): string | ||
270 | + { | ||
271 | + if (empty( $this->root )) { | ||
272 | + $this->root = str_replace('\\', '/', __DIR__); | ||
273 | + } | ||
274 | + return $this->root; | ||
275 | + } | ||
276 | + | ||
277 | + /** | ||
278 | + * Get available environments from environments/index.php file | ||
279 | + * | ||
280 | + * Follow environments/index.php manifest to create your own environments | ||
281 | + * | ||
282 | + * @return array | ||
283 | + */ | ||
284 | + private function getEnvironments(): array | ||
285 | + { | ||
286 | + $path = "{$this->getRoot()}/environments/index.php"; | ||
287 | + if (empty( $this->envs )) { | ||
288 | + /** @noinspection PhpIncludeInspection */ | ||
289 | + $this->envs = require( $path ); | ||
290 | + } | ||
291 | + if (empty( $this->envs )) { | ||
292 | + die( "There are no available environments in $path" ); | ||
293 | + } | ||
294 | + return $this->envs; | ||
295 | + } | ||
296 | + | ||
297 | + /** | ||
298 | + * Get environment names | ||
299 | + * | ||
300 | + * @return array | ||
301 | + */ | ||
302 | + private function getEnvironmentNames(): array | ||
303 | + { | ||
304 | + return array_keys($this->getEnvironments()); | ||
305 | + } | ||
306 | + | ||
307 | + /** | ||
308 | + * Show user variants to choose environment from | ||
309 | + * | ||
310 | + * @param array $envs | ||
311 | + * | ||
312 | + * @return string | ||
313 | + */ | ||
314 | + private function askEnvironment(array $envs): string | ||
315 | + { | ||
316 | + echo "Which environment do you want the application to be initialized in?\n\n"; | ||
317 | + foreach ($envs as $i => $name) { | ||
318 | + echo " [$i] $name\n"; | ||
319 | + } | ||
320 | + $answer = $this->offerEnvironment(count($envs)); | ||
321 | + while (!( ( ctype_digit($answer) && in_array($answer, range(0, count($envs) - 1)) ) || $answer === 'q' )) { | ||
322 | + $answer = $this->offerEnvironment(count($envs)); | ||
323 | + } | ||
324 | + if ($answer === 'q') { | ||
325 | + echo "\n Quit initialization.\n"; | ||
326 | + exit( 0 ); | ||
327 | + } else { | ||
328 | + if (isset( $envs[ $answer ] )) { | ||
329 | + $envName = $envs[ $answer ]; | ||
330 | + } else { | ||
331 | + die( "Error while trying to get environment name. Try another one." ); | ||
332 | + } | ||
333 | + } | ||
334 | + return $envName; | ||
335 | + } | ||
336 | + | ||
337 | + /** | ||
338 | + * Offer user to choose environment number | ||
339 | + * | ||
340 | + * @param int $count | ||
341 | + * | ||
342 | + * @return string | ||
343 | + */ | ||
344 | + private function offerEnvironment(int $count): string | ||
345 | + { | ||
346 | + echo "\n Your choice [0-" . ( $count - 1 ) . ', or "q" to quit] '; | ||
347 | + return trim(fgets(STDIN)); | ||
348 | + } | ||
349 | + | ||
350 | + /** | ||
351 | + * Rewrite files by files in <code>environments/$environment['path']</code> | ||
352 | + */ | ||
353 | + private function rewriteFiles() | ||
354 | + { | ||
355 | + $environment = $this->environment; | ||
356 | + $root = $this->getRoot(); | ||
357 | + if (isset( $environment[ 'path' ] ) && !empty( $environment[ 'path' ] )) { | ||
358 | + $path = $environment[ 'path' ]; | ||
359 | + } else { | ||
360 | + $this->printError('Environment configuration failed. Please set path value.'); | ||
361 | + exit( 1 ); | ||
362 | + } | ||
363 | + $files = $this->getFileList("$root/environments/$path"); | ||
364 | + if (isset( $environment[ 'skipFiles' ] )) { | ||
365 | + $skipFiles = $environment[ 'skipFiles' ]; | ||
366 | + array_walk( | ||
367 | + $skipFiles, | ||
368 | + function (&$value) use ($root) { | ||
369 | + $value = "$root/$value"; | ||
370 | + } | ||
371 | + ); | ||
372 | + $files = array_diff( | ||
373 | + $files, | ||
374 | + array_intersect_key( | ||
375 | + $environment[ 'skipFiles' ], | ||
376 | + array_filter($skipFiles, 'file_exists') | ||
377 | + ) | ||
378 | + ); | ||
379 | + } | ||
380 | + $all = false; | ||
381 | + foreach ($files as $file) { | ||
382 | + if (!$this->copyFile("environments/{$path}/$file", $file, $all)) { | ||
383 | + break; | ||
384 | + } | ||
385 | + } | ||
386 | + } | ||
387 | + | ||
388 | + /** | ||
389 | + * Recursively get all files from $root directory. | ||
390 | + * | ||
391 | + * This will ignore .git, .svn directories. | ||
392 | + * | ||
393 | + * @param string $root | ||
394 | + * @param string $basePath | ||
395 | + * | ||
396 | + * @return array | ||
397 | + */ | ||
398 | + private function getFileList(string $root, string $basePath = ''): array | ||
399 | + { | ||
400 | + $files = []; | ||
401 | + $handle = opendir($root); | ||
402 | + while (( $path = readdir($handle) ) !== false) { | ||
403 | + if ($path === '.git' || $path === '.svn' || $path === '.' || $path === '..') { | ||
404 | + continue; | ||
405 | + } | ||
406 | + $fullPath = "$root/$path"; | ||
407 | + $relativePath = $basePath === '' ? $path : "$basePath/$path"; | ||
408 | + if (is_dir($fullPath)) { | ||
409 | + $files = array_merge($files, $this->getFileList($fullPath, $relativePath)); | ||
410 | + } else { | ||
411 | + $files[] = $relativePath; | ||
412 | + } | ||
413 | + } | ||
414 | + closedir($handle); | ||
415 | + return $files; | ||
416 | + } | ||
417 | + | ||
418 | + /** | ||
419 | + * Run callbacks for application initialization | ||
420 | + */ | ||
421 | + private function callback() | ||
422 | + { | ||
423 | + $environment = $this->environment; | ||
424 | + foreach (self::CALLBACKS as $callback) { | ||
425 | + if (!empty( $environment[ $callback ] )) { | ||
426 | + $this->$callback($environment[ $callback ]); | ||
427 | + } | ||
428 | + } | ||
429 | + } | ||
430 | + | ||
431 | + /** | ||
432 | + * Set directories in $environment['setWrotable'] to chmod 0777 | ||
433 | + * | ||
434 | + * @param $paths | ||
435 | + */ | ||
436 | + private function setWritable(array $paths) | ||
437 | + { | ||
438 | + $root = $this->getRoot(); | ||
439 | + foreach ($paths as $writable) { | ||
440 | + $fullPath = "$root/$writable"; | ||
441 | + if (is_dir($fullPath)) { | ||
442 | + if (@chmod($fullPath, 0777)) { | ||
443 | + echo $this->formatMessage( | ||
444 | + " ***** Set writable $writable (chmod 0777)", | ||
445 | + [ 'fg-yellow' ] | ||
446 | + ) . "\n"; | ||
447 | + } else { | ||
448 | + $this->printError("Operation chmod not permitted for directory $writable."); | ||
449 | + } | ||
450 | + } else { | ||
451 | + if (!@mkdir($fullPath, 0777, true)) { | ||
452 | + $this->printError("Directory $writable does not exist and cannot be created."); | ||
453 | + } | ||
454 | + } | ||
455 | + } | ||
456 | + } | ||
457 | + | ||
458 | + /** | ||
459 | + * Set files in $environment['setExecutable'] to chmod 0755 | ||
460 | + * | ||
461 | + * @param $paths | ||
462 | + */ | ||
463 | + private function setExecutable(array $paths) | ||
464 | + { | ||
465 | + $root = $this->getRoot(); | ||
466 | + foreach ($paths as $executable) { | ||
467 | + $fullPath = "$root/$executable"; | ||
468 | + if (file_exists($fullPath)) { | ||
469 | + if (@chmod($fullPath, 0755)) { | ||
470 | + echo $this->formatMessage(" ***** Set executable $executable (chmod 0755)") . "\n"; | ||
471 | + } else { | ||
472 | + $this->printError("Operation chmod not permitted for $executable."); | ||
473 | + } | ||
474 | + } else { | ||
475 | + $this->printError("$executable does not exist."); | ||
476 | + } | ||
477 | + } | ||
478 | + } | ||
479 | + | ||
480 | + /** | ||
481 | + * Set cookie validation keys to files in $environment['setCookieValidationKey']. | ||
482 | + * | ||
483 | + * @param $paths | ||
484 | + */ | ||
485 | + private function setCookieValidationKey(array $paths) | ||
486 | + { | ||
487 | + $root = $this->getRoot(); | ||
488 | + foreach ($paths as $file) { | ||
489 | + echo $this->formatMessage( | ||
490 | + " ***** Generate cookie validation key in $file", | ||
491 | + [ | ||
492 | + 'bold', | ||
493 | + 'fg-magenta', | ||
494 | + ] | ||
495 | + ) . "\n"; | ||
496 | + $file = $root . '/' . $file; | ||
497 | + $length = 32; | ||
498 | + $bytes = openssl_random_pseudo_bytes($length); | ||
499 | + $key = strtr(substr(base64_encode($bytes), 0, $length), '+/=', '_-.'); | ||
500 | + $this->setParam('cookieValidationKey', $key, $file); | ||
501 | + } | ||
502 | + } | ||
503 | + | ||
504 | + private function createSymlink(array $links) | ||
505 | + { | ||
506 | + $root = $this->getRoot(); | ||
507 | + foreach ($links as $link => $target) { | ||
508 | + //first removing folders to avoid errors if the folder already exists | ||
509 | + @rmdir($root . "/" . $link); | ||
510 | + //next removing existing symlink in order to update the target | ||
511 | + if (is_link($root . "/" . $link)) { | ||
512 | + @unlink($root . "/" . $link); | ||
513 | + } | ||
514 | + if (@symlink($root . "/" . $target, $root . "/" . $link)) { | ||
515 | + echo $this->formatMessage(" ***** Symlink $root/$target $root/$link", [ 'fg-blue' ]) . "\n"; | ||
516 | + } else { | ||
517 | + $this->printError("Cannot create symlink $root/$target $root/$link."); | ||
518 | + } | ||
519 | + } | ||
520 | + } | ||
521 | + | ||
522 | + /** | ||
523 | + * Copy file from environment directory to project directory | ||
524 | + * | ||
525 | + * @param string $source Environment source file path | ||
526 | + * @param string $target Project file path target | ||
527 | + * @param bool $all Rewrite all flag | ||
528 | + * | ||
529 | + * @return bool | ||
530 | + */ | ||
531 | + private function copyFile(string $source, string $target, bool &$all) | ||
532 | + { | ||
533 | + $root = $this->getRoot(); | ||
534 | + $params = $this->getParams(); | ||
535 | + if (!is_file($root . '/' . $source)) { | ||
536 | + echo $this->formatMessage(" ----- skip $target ($source not exist)", [ 'fg-cyan' ]) . "\n"; | ||
537 | + return true; | ||
538 | + } | ||
539 | + if (is_file($root . '/' . $target)) { | ||
540 | + if (file_get_contents($root . '/' . $source) === file_get_contents($root . '/' . $target)) { | ||
541 | + echo $this->formatMessage(" ----- unchanged $target", [ 'fg-cyan' ]) . "\n"; | ||
542 | + return true; | ||
543 | + } | ||
544 | + if ($all) { | ||
545 | + echo $this->formatMessage(" ----- overwrite $target", [ 'fg-blue' ]) . "\n"; | ||
546 | + } else { | ||
547 | + echo $this->formatMessage(" -- exist $target", [ 'fg-magenta' ]) . "\n"; | ||
548 | + echo " overwrite? [Yes|No|All|Quit] "; | ||
549 | + | ||
550 | + $answer = !empty( $params[ 'overwrite' ] ) ? $params[ 'overwrite' ] : trim(fgets(STDIN)); | ||
551 | + echo "\n"; | ||
552 | + if (!strncasecmp($answer, 'q', 1)) { | ||
553 | + return false; | ||
554 | + } else { | ||
555 | + if (!strncasecmp($answer, 'y', 1)) { | ||
556 | + echo $this->formatMessage(" ----- overwrite $target", [ 'fg-blue' ]) . "\n"; | ||
557 | + } else { | ||
558 | + if (!strncasecmp($answer, 'a', 1)) { | ||
559 | + echo $this->formatMessage(" ----- overwrite $target", [ 'fg-blue' ]) . "\n"; | ||
560 | + $all = true; | ||
561 | + } else { | ||
562 | + echo $this->formatMessage( | ||
563 | + " ----- skip $target ($source not exist)", | ||
564 | + [ 'fg-cyan' ] | ||
565 | + ) . "\n"; | ||
566 | + return true; | ||
567 | + } | ||
568 | + } | ||
569 | + } | ||
570 | + } | ||
571 | + file_put_contents($root . '/' . $target, file_get_contents($root . '/' . $source)); | ||
572 | + return true; | ||
573 | + } | ||
574 | + echo "\n" . $this->formatMessage(" ----- generate $target", [ 'fg-green' ]) . "\n"; | ||
575 | + @mkdir(dirname($root . '/' . $target), 0777, true); | ||
576 | + file_put_contents($root . '/' . $target, file_get_contents($root . '/' . $source)); | ||
577 | + return true; | ||
578 | + } | ||
579 | + | ||
580 | + /** | ||
581 | + * Set param in file to particular value. | ||
582 | + * | ||
583 | + * For example: setParam('param', 'value', 'config.php') will replace content in config.php | ||
584 | + * | ||
585 | + * from | ||
586 | + * | ||
587 | + * 'param' => '' | ||
588 | + * | ||
589 | + * to | ||
590 | + * | ||
591 | + * 'param' => 'value' | ||
592 | + * | ||
593 | + * @param string $param | ||
594 | + * @param string $value | ||
595 | + * @param string $file | ||
596 | + * @param bool $noQuote | ||
597 | + * @param bool $force | ||
598 | + * @param bool $alternate | ||
599 | + */ | ||
600 | + private function setParam( | ||
601 | + string $param, | ||
602 | + string $value, | ||
603 | + string $file, | ||
604 | + bool $noQuote = false, | ||
605 | + bool $force = false, | ||
606 | + bool $alternate = false | ||
607 | + ) { | ||
608 | + if ($alternate) { | ||
609 | + $regexp = '/(("|\')db("|\')\s*=>\s*)(.*|.*)/'; | ||
610 | + } else { | ||
611 | + if ($force) { | ||
612 | + $regexp = '/(("|\')' . $param . '("|\')\s*=>\s*)(".*"|\'.*\')/'; | ||
613 | + } else { | ||
614 | + $regexp = '/(("|\')' . $param . '("|\')\s*=>\s*)(""|\'\')/'; | ||
615 | + } | ||
616 | + } | ||
617 | + if ($noQuote) { | ||
618 | + $content = preg_replace( | ||
619 | + $regexp, | ||
620 | + "\\1$value", | ||
621 | + file_get_contents($file) | ||
622 | + ); | ||
623 | + } else { | ||
624 | + $content = preg_replace( | ||
625 | + $regexp, | ||
626 | + "\\1'$value'", | ||
627 | + file_get_contents($file) | ||
628 | + ); | ||
629 | + } | ||
630 | + file_put_contents($file, $content); | ||
631 | + } | ||
632 | + | ||
633 | + /** | ||
634 | + * Init postgres connection | ||
635 | + * | ||
636 | + * @return array | ||
637 | + */ | ||
638 | + private function initPostgres(): array | ||
639 | + { | ||
640 | + $db = [ | ||
641 | + 'dbtype' => 'postgresql', | ||
642 | + ]; | ||
643 | + echo "\n Enter your database host: "; | ||
644 | + $host = trim(fgets(STDIN)); | ||
645 | + echo "\n Enter your database port: "; | ||
646 | + $port = trim(fgets(STDIN)); | ||
647 | + echo "\n Enter your database schema: "; | ||
648 | + $db[ 'schema' ] = trim(fgets(STDIN)); | ||
649 | + echo "\n Enter your database name: "; | ||
650 | + $name = trim(fgets(STDIN)); | ||
651 | + echo "\n Enter your database username: "; | ||
652 | + $db[ 'username' ] = trim(fgets(STDIN)); | ||
653 | + echo "\n Enter your database password: "; | ||
654 | + $db[ 'password' ] = trim(fgets(STDIN)); | ||
655 | + $db[ 'dsn' ] = "pgsql:host={$host};port={$port};dbname={$name}"; | ||
656 | + $this->db = $db; | ||
657 | + return $db; | ||
658 | + } | ||
659 | + | ||
660 | + /** | ||
661 | + * Init mysql connection | ||
662 | + * | ||
663 | + * @return array | ||
664 | + */ | ||
665 | + private function initMysql(): array | ||
666 | + { | ||
667 | + $db = [ | ||
668 | + 'dbtype' => 'mysql', | ||
669 | + ]; | ||
670 | + echo "\n Enter your database host: "; | ||
671 | + $host = trim(fgets(STDIN)); | ||
672 | + echo "\n Enter your database name: "; | ||
673 | + $name = trim(fgets(STDIN)); | ||
674 | + echo "\n Enter your database username: "; | ||
675 | + $db[ 'username' ] = trim(fgets(STDIN)); | ||
676 | + echo "\n Enter your database password: "; | ||
677 | + $db[ 'password' ] = trim(fgets(STDIN)); | ||
678 | + $db[ 'dsn' ] = "mysql:host={$host};dbname={$name}"; | ||
679 | + $this->db = $db; | ||
680 | + return $db; | ||
681 | + } | ||
682 | + | ||
683 | + /** | ||
684 | + * Validate non-interactive db data and fill it for further actions. | ||
685 | + * | ||
686 | + * @return array | ||
687 | + */ | ||
688 | + private function validateConnection(): array | ||
689 | + { | ||
690 | + $db = []; | ||
691 | + $dbType = strtolower($this->getParamValue('dbtype')); | ||
692 | + if ($dbType !== 'postgresql' && $dbType !== 'mysql') { | ||
693 | + die( "Supported DB types are postgresql and mysql" ); | ||
694 | + } | ||
695 | + if ($dbType === 'postgresql') { | ||
696 | + $host = $this->getParamValue('host') ? : '127.0.0.1'; | ||
697 | + $port = $this->getParamValue('port') ? : '5432'; | ||
698 | + $db[ 'schema' ] = $this->getParamValue('schema') ? : 'public'; | ||
699 | + $name = $this->getParamValue('dbname'); | ||
700 | + if (empty( $name )) { | ||
701 | + die( "Database name must be set" ); | ||
702 | + } | ||
703 | + $db[ 'dsn' ] = "pgsql:host={$host};port={$port};dbname={$name}"; | ||
704 | + if (empty( $username = $this->getParamValue('username') )) { | ||
705 | + die( "Database username must be set" ); | ||
706 | + } | ||
707 | + $db[ 'username' ] = $username; | ||
708 | + if (empty( $password = $this->getParamValue('password') )) { | ||
709 | + die( "Database password must be set" ); | ||
710 | + } | ||
711 | + $db[ 'password' ] = $password; | ||
712 | + } else { | ||
713 | + $host = $this->getParamValue('host') ? : '127.0.0.1'; | ||
714 | + $name = $this->getParamValue('dbname'); | ||
715 | + if (empty( $name )) { | ||
716 | + die( "Database name must be set" ); | ||
717 | + } | ||
718 | + $db[ 'dsn' ] = "mysql:host={$host};dbname={$name}"; | ||
719 | + if (empty( $username = $this->getParamValue('username') )) { | ||
720 | + die( "Database username must be set" ); | ||
721 | + } | ||
722 | + $db[ 'username' ] = $username; | ||
723 | + if (empty( $password = $this->getParamValue('password') )) { | ||
724 | + die( "Database password must be set" ); | ||
725 | + } | ||
726 | + $db[ 'password' ] = $password; | ||
727 | + } | ||
728 | + $db[ 'dbtype' ] = $dbType; | ||
729 | + $this->db = $db; | ||
730 | + return $db; | ||
731 | + } | ||
732 | + | ||
733 | + /** | ||
734 | + * Copy db connection file accroding to choosen driver | ||
735 | + */ | ||
736 | + private function initDb() | ||
737 | + { | ||
738 | + if (!empty( $configPath = $this->environment[ 'setDbConnection' ] )) { | ||
739 | + if (preg_match('/(.*)\/.*/', $configPath, $matches)) { | ||
740 | + $path = $matches[ 1 ]; | ||
741 | + } else { | ||
742 | + $this->printError("Unknown error while trying to init database"); | ||
743 | + exit( 1 ); | ||
744 | + } | ||
745 | + $filename = "db{$this->db['dbtype']}.php"; | ||
746 | + $all = !empty( $this->getParamValue('overwrite') ) ? false : true; | ||
747 | + $fullpath = "{$path}/{$filename}"; | ||
748 | + if ($this->copyFile("environments/{$filename}", $fullpath, $all)) { | ||
749 | + $this->rewriteDb($fullpath, $configPath); | ||
750 | + } | ||
751 | + } | ||
752 | + } | ||
753 | + | ||
754 | + /** | ||
755 | + * Rewrite params in db config and local config files | ||
756 | + * | ||
757 | + * @param string $path | ||
758 | + * @param string $config | ||
759 | + */ | ||
760 | + private function rewriteDb(string $path, string $config) | ||
761 | + { | ||
762 | + $db = $this->db; | ||
763 | + $this->setParam('dsn', $db[ 'dsn' ], $path); | ||
764 | + $this->setParam('username', $db[ 'username' ], $path); | ||
765 | + $this->setParam('password', $db[ 'password' ], $path); | ||
766 | + if ($this->db[ 'dbtype' ] == 'postgresql') { | ||
767 | + $this->setParam('defaultSchema', $db[ 'schema' ], $path); | ||
768 | + } | ||
769 | + $filename = "db{$this->db['dbtype']}.php"; | ||
770 | + $this->setParam('db', "require('{$filename}')", $config, true, true); | ||
771 | + $driver = ucfirst($db[ 'dbtype' ]); | ||
772 | + echo $this->formatMessage( | ||
773 | + "Database access file $path was created and filled with configurations for database driver {$driver}", | ||
774 | + [ 'fg-green' ] | ||
775 | + ) . "\n"; | ||
776 | + echo $this->formatMessage( | ||
777 | + "Database access file $path was linked to local config file $config", | ||
778 | + [ 'fg-green' ] | ||
779 | + ) . "\n"; | ||
780 | + echo $this->formatMessage("Database Access: ", [ 'bg-yellow' ]) . $this->formatMessage( | ||
781 | + $this->db[ 'dsn' ], | ||
782 | + [ | ||
783 | + 'bg-yellow', | ||
784 | + 'fg-blue', | ||
785 | + ] | ||
786 | + ) . $this->formatMessage(', username: ', [ 'bg-yellow' ]) . $this->formatMessage( | ||
787 | + $this->db[ 'username' ], | ||
788 | + [ | ||
789 | + 'bg-yellow', | ||
790 | + 'fg-blue', | ||
791 | + ] | ||
792 | + ) . "\n"; | ||
793 | + echo " Database initialization completed\n"; | ||
794 | + echo " =================================\n"; | ||
795 | + } | ||
796 | + | ||
797 | + /** | ||
798 | + * Prints error message. | ||
799 | + * | ||
800 | + * @param string $message message | ||
801 | + */ | ||
802 | + private function printError(string $message) | ||
803 | + { | ||
804 | + echo "\n " . $this->formatMessage( | ||
805 | + "Error. $message", | ||
806 | + [ | ||
807 | + 'fg-red', | ||
808 | + 'bold', | ||
809 | + ] | ||
810 | + ) . " \n"; | ||
811 | + } | ||
812 | + | ||
813 | + /** | ||
814 | + * Returns true if the stream supports colorization. ANSI colors are disabled if not supported by the stream. | ||
815 | + * - windows without ansicon | ||
816 | + * - not tty consoles | ||
817 | + * | ||
818 | + * @return boolean true if the stream supports ANSI colors, otherwise false. | ||
819 | + */ | ||
820 | + private function ansiColorsSupported(): bool | ||
821 | + { | ||
822 | + return DIRECTORY_SEPARATOR === '\\' ? getenv('ANSICON') !== false || getenv( | ||
823 | + 'ConEmuANSI' | ||
824 | + ) === 'ON' : function_exists('posix_isatty') && @posix_isatty(STDOUT); | ||
825 | + } | ||
826 | + | ||
827 | + /** | ||
828 | + * Get ANSI code of style. | ||
829 | + * | ||
830 | + * @param string $name style name | ||
831 | + * | ||
832 | + * @return integer ANSI code of style. | ||
833 | + */ | ||
834 | + private function getStyleCode(string $name): int | ||
835 | + { | ||
836 | + $styles = [ | ||
837 | + 'bold' => 1, | ||
838 | + 'fg-black' => 30, | ||
839 | + 'fg-red' => 31, | ||
840 | + 'fg-green' => 32, | ||
841 | + 'fg-yellow' => 33, | ||
842 | + 'fg-blue' => 34, | ||
843 | + 'fg-magenta' => 35, | ||
844 | + 'fg-cyan' => 36, | ||
845 | + 'fg-white' => 37, | ||
846 | + 'bg-black' => 40, | ||
847 | + 'bg-red' => 41, | ||
848 | + 'bg-green' => 42, | ||
849 | + 'bg-yellow' => 43, | ||
850 | + 'bg-blue' => 44, | ||
851 | + 'bg-magenta' => 45, | ||
852 | + 'bg-cyan' => 46, | ||
853 | + 'bg-white' => 47, | ||
854 | + ]; | ||
855 | + return $styles[ $name ]; | ||
856 | + } | ||
857 | + | ||
858 | + /** | ||
859 | + * Formats message using styles if STDOUT supports it. | ||
860 | + * | ||
861 | + * @param string $message message | ||
862 | + * @param string[] $styles styles | ||
863 | + * | ||
864 | + * @return string formatted message. | ||
865 | + */ | ||
866 | + public function formatMessage(string $message, array $styles = []): string | ||
867 | + { | ||
868 | + if (empty( $styles ) || !$this->ansiColorsSupported()) { | ||
869 | + return $message; | ||
870 | + } | ||
871 | + return sprintf( | ||
872 | + "\x1b[%sm", | ||
873 | + implode( | ||
874 | + ';', | ||
875 | + array_map( | ||
876 | + [ | ||
877 | + $this, | ||
878 | + 'getStyleCode', | ||
879 | + ], | ||
880 | + $styles | ||
881 | + ) | ||
882 | + ) | ||
883 | + ) . $message . "\x1b[0m"; | ||
884 | + } | ||
885 | + | ||
886 | + /** | ||
887 | + * Inform that an application initialized without database. Exit code 0, means success. | ||
888 | + */ | ||
889 | + private function noDatabaseMessage() | ||
890 | + { | ||
891 | + echo $this->formatMessage( | ||
892 | + "\n Attention: ", | ||
893 | + [ | ||
894 | + 'bold', | ||
895 | + 'fg-red', | ||
896 | + 'bg-yellow', | ||
897 | + ] | ||
898 | + ); | ||
899 | + echo $this->formatMessage( | ||
900 | + "Application initialized without database. Set it manually in {$this->getRoot()}/common/config/main-local.php and run migration", | ||
901 | + [ | ||
902 | + 'bold', | ||
903 | + 'bg-yellow', | ||
904 | + ] | ||
905 | + ) . "\n"; | ||
906 | + exit( 0 ); | ||
907 | + } | ||
908 | + | ||
909 | + /** | ||
910 | + * Perform database migration if input param migrate doesn't set to 'no' | ||
911 | + */ | ||
912 | + private function migrate() | ||
913 | + { | ||
914 | + $migrate = $this->getParamValue('migrate'); | ||
915 | + if ($migrate == 'no') { | ||
916 | + echo $this->formatMessage( | ||
917 | + "Migration skipped by user. In order to application work correct you will need to run it manually", | ||
918 | + [ | ||
919 | + 'bg-yellow', | ||
920 | + 'bold', | ||
921 | + ] | ||
922 | + ) . "\n"; | ||
923 | + exit( 0 ); | ||
924 | + } elseif ($migrate != 'yes') { | ||
925 | + do { | ||
926 | + echo " Do you want to perform database migration? [yes|no]"; | ||
927 | + $answer = trim(fgets(STDIN)); | ||
928 | + } while (strncasecmp($answer, 'y', 1) !== 0 && strncasecmp($answer, 'n', 1) !== 0); | ||
929 | + if (strncasecmp($answer, 'n', 1) === 0) { | ||
930 | + echo $this->formatMessage( | ||
931 | + "Migration skipped by user. In order to application work correct you will need to run it manually", | ||
932 | + [ | ||
933 | + 'bg-yellow', | ||
934 | + 'bold', | ||
935 | + ] | ||
936 | + ) . "\n"; | ||
937 | + exit( 0 ); | ||
938 | + } | ||
939 | + } | ||
940 | + echo $this->formatMessage("Migration begins...", [ 'fg-yellow' ]) . "\n"; | ||
941 | + $migrationPath = $this->getParamValue('migrationPath'); | ||
942 | + if (empty( $migrationPath )) { | ||
943 | + $migrationPath = 'vendor/artweb/artbox-core/migrations'; | ||
944 | + } | ||
945 | + $result = exec("php yii migrate --migrationPath=$migrationPath --interactive=0", $output, $return); | ||
946 | + if ($return !== 0) { | ||
947 | + $this->printError("Migration cannot be applied. Run it manually to check the reason"); | ||
948 | + exit( 1 ); | ||
949 | + } | ||
950 | + $this->writeLine($result); | ||
951 | + foreach ($output as $value) { | ||
952 | + $this->writeLine($value); | ||
953 | + } | ||
954 | + echo $this->formatMessage("Migration ended successfully", [ 'fg-yellow' ]) . "\n"; | ||
955 | + } | ||
956 | + | ||
957 | + /** | ||
958 | + * Write line of code followed by new line symbol | ||
959 | + * | ||
960 | + * @param string $string | ||
961 | + */ | ||
962 | + private function writeLine(string $string) | ||
963 | + { | ||
964 | + echo $string . "\n"; | ||
965 | + } | ||
966 | + | ||
967 | + /** | ||
968 | + * Perform user creation if input param user doesn't set to 'no' | ||
969 | + */ | ||
970 | + private function createUser() | ||
971 | + { | ||
972 | + $params = $this->getParams(); | ||
973 | + if (!isset( $params[ 'defaultuser' ] )) { | ||
974 | + if ($this->getParamValue('user') == 'no') { | ||
975 | + echo $this->formatMessage( | ||
976 | + "User creation skipped by user. Run command 'php yii user/create' manually to create user.", | ||
977 | + [ | ||
978 | + 'bg-yellow', | ||
979 | + 'bold', | ||
980 | + ] | ||
981 | + ) . "\n"; | ||
982 | + exit( 0 ); | ||
983 | + } | ||
984 | + do { | ||
985 | + echo " Do you want to create user? [yes|no]"; | ||
986 | + $answer = trim(fgets(STDIN)); | ||
987 | + } while (strncasecmp($answer, 'y', 1) !== 0 && strncasecmp($answer, 'n', 1) !== 0); | ||
988 | + if (strncasecmp($answer, 'n', 1) === 0) { | ||
989 | + echo $this->formatMessage( | ||
990 | + "User creation skipped by user. Run command 'php yii user/create' manually to create user.", | ||
991 | + [ | ||
992 | + 'bg-yellow', | ||
993 | + 'bold', | ||
994 | + ] | ||
995 | + ) . "\n"; | ||
996 | + exit( 0 ); | ||
997 | + } | ||
998 | + echo "\n Enter username: "; | ||
999 | + $username = trim(fgets(STDIN)); | ||
1000 | + echo "\n Enter email: "; | ||
1001 | + $email = trim(fgets(STDIN)); | ||
1002 | + echo "\n Enter password: "; | ||
1003 | + $password = trim(fgets(STDIN)); | ||
1004 | + echo "\n"; | ||
1005 | + $result = exec("php yii create/user $username $email $password", $output, $return); | ||
1006 | + } else { | ||
1007 | + $result = exec("php yii create/user", $output, $return); | ||
1008 | + } | ||
1009 | + $this->handleUserCreation($result, $output, $return); | ||
1010 | + } | ||
1011 | + | ||
1012 | + /** | ||
1013 | + * Handle user creation result | ||
1014 | + * | ||
1015 | + * @param string $result | ||
1016 | + * @param array $output | ||
1017 | + * @param int $return | ||
1018 | + */ | ||
1019 | + private function handleUserCreation($result, $output, $return) | ||
1020 | + { | ||
1021 | + if ($return !== 0) { | ||
1022 | + $this->printError("User cannot be created. Run 'php yii create/user' manually to check the reason"); | ||
1023 | + exit( 1 ); | ||
1024 | + } | ||
1025 | + $this->writeLine($result); | ||
1026 | + echo $this->formatMessage("User created successfully", [ 'fg-yellow' ]) . "\n"; | ||
1027 | + } | ||
1028 | + | ||
1029 | + /** | ||
1030 | + * Congratulate with successfull installation and optionally open browser | ||
1031 | + */ | ||
1032 | + private function congratulate() | ||
1033 | + { | ||
1034 | + echo "\n" . $this->formatMessage( | ||
1035 | + "Congratulations. Artbox Basic has been successfully installed.", | ||
1036 | + [ | ||
1037 | + 'fg-yellow', | ||
1038 | + 'bold', | ||
1039 | + ] | ||
1040 | + ) . "\n"; | ||
1041 | + $url = $this->getParamValue('o'); | ||
1042 | + if (!empty( $url )) { | ||
1043 | + if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') { | ||
1044 | + shell_exec("explorer '{$url}'"); | ||
1045 | + } else { | ||
1046 | + shell_exec("sensible-browser {$url} > /dev/null 2>/dev/null &"); | ||
1047 | + $params = $this->getParams(); | ||
1048 | + if (isset( $params[ 'fun' ] )) { | ||
1049 | + echo shell_exec("hollywood"); | ||
1050 | + } | ||
1051 | + } | ||
1052 | + } | ||
1053 | + } | ||
1054 | + | ||
1055 | + /** | ||
1056 | + * Check if database connection could be established | ||
1057 | + * | ||
1058 | + * @return bool | ||
1059 | + * @throws \Exception | ||
1060 | + */ | ||
1061 | + private function databaseCheck(): bool | ||
1062 | + { | ||
1063 | + $params = $this->getParams(); | ||
1064 | + if (isset( $params[ 'checkdb' ] )) { | ||
1065 | + $configPath = $this->environment[ 'setDbConnection' ]; | ||
1066 | + if (preg_match('/(.*)\/.*/', $configPath, $matches)) { | ||
1067 | + $path = $matches[ 1 ]; | ||
1068 | + } else { | ||
1069 | + $this->printError("Unknown error while trying to check connection"); | ||
1070 | + exit( 1 ); | ||
1071 | + } | ||
1072 | + $filename = "db{$this->db['dbtype']}.php"; | ||
1073 | + $fullpath = "{$path}/{$filename}"; | ||
1074 | + exec("php yii check/connection {$fullpath}", $output, $return); | ||
1075 | + if ($return > 0) { | ||
1076 | + if (isset( $params[ 'dbinit' ] )) { | ||
1077 | + throw new Exception("Couldn't connect to the database"); | ||
1078 | + } | ||
1079 | + echo $this->printError("Database unavailable. Please choose another database!"); | ||
1080 | + return false; | ||
1081 | + } | ||
1082 | + } | ||
1083 | + echo $this->formatMessage(" Database connection successful!", [ 'fg-green' ]) . "\n"; | ||
1084 | + return true; | ||
1085 | + } | ||
1086 | + } | ||
1087 | + | ||
0 | \ No newline at end of file | 1088 | \ No newline at end of file |
1 | +{ | ||
2 | + "name": "artweb/artbox-basic", | ||
3 | + "description": "Artbox Basic Project Template", | ||
4 | + "keywords": [ | ||
5 | + "yii2", | ||
6 | + "cms", | ||
7 | + "artweb", | ||
8 | + "artbox", | ||
9 | + "basic", | ||
10 | + "project template" | ||
11 | + ], | ||
12 | + "homepage": "http://artweb.ua", | ||
13 | + "type": "project", | ||
14 | + "license": "BSD-3-Clause", | ||
15 | + "support": { | ||
16 | + "wiki": "http://www.yiiframework.com/wiki/" | ||
17 | + }, | ||
18 | + "minimum-stability": "dev", | ||
19 | + "require": { | ||
20 | + "php": ">=7.0", | ||
21 | + "artweb/artbox-core": "*", | ||
22 | + "yiisoft/yii2": "~2.0", | ||
23 | + "yiisoft/yii2-bootstrap": "~2.0", | ||
24 | + "yiisoft/yii2-swiftmailer": "~2.0", | ||
25 | + "noam148/yii2-image-manager": "~1.0", | ||
26 | + "yiister/yii2-gentelella": "~1.0", | ||
27 | + "yii2tech/filedb": "~1.0", | ||
28 | + "hiqdev/yii2-asset-pnotify": "~2.0", | ||
29 | + "hiqdev/yii2-asset-icheck": "~1.0", | ||
30 | + "google/apiclient": "^2.0" | ||
31 | + }, | ||
32 | + "require-dev": { | ||
33 | + "yiisoft/yii2-debug": "~2.0.0", | ||
34 | + "yiisoft/yii2-gii": "~2.0.0", | ||
35 | + "yiisoft/yii2-faker": "~2.0.0", | ||
36 | + "codeception/base": "^2.2.3", | ||
37 | + "codeception/verify": "~0.3.1" | ||
38 | + }, | ||
39 | + "config": { | ||
40 | + "process-timeout": 1800 | ||
41 | + }, | ||
42 | + "extra": { | ||
43 | + "asset-installer-paths": { | ||
44 | + "npm-asset-library": "vendor/npm", | ||
45 | + "bower-asset-library": "vendor/bower" | ||
46 | + } | ||
47 | + }, | ||
48 | + "scripts": { | ||
49 | + "post-install-cmd": "php init --env=Development --overwrite=n" | ||
50 | + }, | ||
51 | + "autoload": { | ||
52 | + "psr-4": { | ||
53 | + "artbox\\core\\": "artweb/artbox-core/", | ||
54 | + "artbox\\gentelella\\": "artweb/artbox-gentelella/" | ||
55 | + } | ||
56 | + } | ||
57 | +} |
backend/config/main.php
@@ -38,11 +38,12 @@ | @@ -38,11 +38,12 @@ | ||
38 | 'assetManager' => [ | 38 | 'assetManager' => [ |
39 | 'bundles' => [ | 39 | 'bundles' => [ |
40 | 'yiister\gentelella\assets\ThemeAsset' => [ | 40 | 'yiister\gentelella\assets\ThemeAsset' => [ |
41 | - 'sourcePath' => '@backend/assets/', | ||
42 | - 'js' => [ | 41 | + 'basePath' => '@webroot', |
42 | + 'baseUrl' => '@web', | ||
43 | + 'js' => [ | ||
43 | 'js/custom.js', | 44 | 'js/custom.js', |
44 | ], | 45 | ], |
45 | - 'css' => [ | 46 | + 'css' => [ |
46 | 'css/custom.css', | 47 | 'css/custom.css', |
47 | ], | 48 | ], |
48 | ], | 49 | ], |
1 | +<?php | ||
2 | + namespace backend\controllers; | ||
3 | + | ||
4 | + use common\models\Settings; | ||
5 | + use yii\base\InvalidConfigException; | ||
6 | + use yii\filters\AccessControl; | ||
7 | + use yii\web\Controller; | ||
8 | + use Yii; | ||
9 | + | ||
10 | + /** | ||
11 | + * Class SettingsController | ||
12 | + * | ||
13 | + * @package artbox\core\controllers | ||
14 | + */ | ||
15 | + class SettingsController extends Controller | ||
16 | + { | ||
17 | + /** | ||
18 | + * @inheritdoc | ||
19 | + */ | ||
20 | + public function behaviors() | ||
21 | + { | ||
22 | + return [ | ||
23 | + 'access' => [ | ||
24 | + 'class' => AccessControl::className(), | ||
25 | + 'rules' => [ | ||
26 | + [ | ||
27 | + 'actions' => [ | ||
28 | + 'login', | ||
29 | + 'error', | ||
30 | + ], | ||
31 | + 'allow' => true, | ||
32 | + ], | ||
33 | + [ | ||
34 | + 'actions' => [ | ||
35 | + 'logout', | ||
36 | + 'index', | ||
37 | + ], | ||
38 | + 'allow' => true, | ||
39 | + 'roles' => [ '@' ], | ||
40 | + ], | ||
41 | + ], | ||
42 | + ], | ||
43 | + ]; | ||
44 | + } | ||
45 | + /** | ||
46 | + * @inheritdoc | ||
47 | + */ | ||
48 | + public function getViewPath() | ||
49 | + { | ||
50 | + return \Yii::getAlias('@artbox/core/views/settings'); | ||
51 | + } | ||
52 | + | ||
53 | + /** | ||
54 | + * Display site settings page | ||
55 | + * | ||
56 | + * @return string|\yii\web\Response | ||
57 | + */ | ||
58 | + public function actionIndex() | ||
59 | + { | ||
60 | + $model = $this->findSettings(); | ||
61 | + | ||
62 | + if ($model->load(Yii::$app->request->post()) && $model->save()) { | ||
63 | + Yii::$app->session->setFlash('success', 'Settings saved'); | ||
64 | + | ||
65 | + return $this->goHome(); | ||
66 | + } | ||
67 | + | ||
68 | + return $this->render( | ||
69 | + 'settings', | ||
70 | + [ | ||
71 | + 'model' => $model, | ||
72 | + ] | ||
73 | + ); | ||
74 | + } | ||
75 | + | ||
76 | + /** | ||
77 | + * Find site settings | ||
78 | + * | ||
79 | + * @return \yii2tech\filedb\ActiveRecord | ||
80 | + * @throws \yii\base\InvalidConfigException | ||
81 | + */ | ||
82 | + public function findSettings() | ||
83 | + { | ||
84 | + if ($model = Settings::find() | ||
85 | + ->one() | ||
86 | + ) { | ||
87 | + return $model; | ||
88 | + } else { | ||
89 | + throw new InvalidConfigException('Settings file not found'); | ||
90 | + } | ||
91 | + } | ||
92 | + } | ||
93 | + | ||
0 | \ No newline at end of file | 94 | \ No newline at end of file |
backend/controllers/SiteController.php
1 | <?php | 1 | <?php |
2 | -namespace backend\controllers; | ||
3 | - | ||
4 | -use Yii; | ||
5 | -use yii\web\Controller; | ||
6 | -use yii\filters\VerbFilter; | ||
7 | -use yii\filters\AccessControl; | ||
8 | -use common\models\LoginForm; | ||
9 | - | ||
10 | -/** | ||
11 | - * Site controller | ||
12 | - */ | ||
13 | -class SiteController extends Controller | ||
14 | -{ | 2 | + namespace backend\controllers; |
3 | + | ||
4 | + use Yii; | ||
5 | + use yii\web\Controller; | ||
6 | + use yii\filters\VerbFilter; | ||
7 | + use yii\filters\AccessControl; | ||
8 | + use common\models\LoginForm; | ||
9 | + | ||
15 | /** | 10 | /** |
16 | - * @inheritdoc | 11 | + * Site controller |
17 | */ | 12 | */ |
18 | - public function behaviors() | 13 | + class SiteController extends Controller |
19 | { | 14 | { |
20 | - return [ | ||
21 | - 'access' => [ | ||
22 | - 'class' => AccessControl::className(), | ||
23 | - 'rules' => [ | ||
24 | - [ | ||
25 | - 'actions' => ['login', 'error'], | ||
26 | - 'allow' => true, | ||
27 | - ], | ||
28 | - [ | ||
29 | - 'actions' => [ | ||
30 | - 'logout', | ||
31 | - 'index', | ||
32 | - 'analytic', | 15 | + /** |
16 | + * @inheritdoc | ||
17 | + */ | ||
18 | + public function behaviors() | ||
19 | + { | ||
20 | + return [ | ||
21 | + 'access' => [ | ||
22 | + 'class' => AccessControl::className(), | ||
23 | + 'rules' => [ | ||
24 | + [ | ||
25 | + 'actions' => [ | ||
26 | + 'login', | ||
27 | + 'error', | ||
28 | + ], | ||
29 | + 'allow' => true, | ||
30 | + ], | ||
31 | + [ | ||
32 | + 'actions' => [ | ||
33 | + 'logout', | ||
34 | + 'index', | ||
35 | + 'analytic', | ||
36 | + ], | ||
37 | + 'allow' => true, | ||
38 | + 'roles' => [ '@' ], | ||
33 | ], | 39 | ], |
34 | - 'allow' => true, | ||
35 | - 'roles' => ['@'], | ||
36 | ], | 40 | ], |
37 | ], | 41 | ], |
38 | - ], | ||
39 | - 'verbs' => [ | ||
40 | - 'class' => VerbFilter::className(), | ||
41 | - 'actions' => [ | ||
42 | - 'logout' => ['post'], | 42 | + 'verbs' => [ |
43 | + 'class' => VerbFilter::className(), | ||
44 | + 'actions' => [ | ||
45 | + 'logout' => [ 'post' ], | ||
46 | + ], | ||
43 | ], | 47 | ], |
44 | - ], | ||
45 | - ]; | ||
46 | - } | ||
47 | - | ||
48 | - /** | ||
49 | - * @inheritdoc | ||
50 | - */ | ||
51 | - public function actions() | ||
52 | - { | ||
53 | - return [ | ||
54 | - 'error' => [ | ||
55 | - 'class' => 'yii\web\ErrorAction', | ||
56 | - ], | ||
57 | - ]; | ||
58 | - } | ||
59 | - | ||
60 | - /** | ||
61 | - * Displays homepage. | ||
62 | - * | ||
63 | - * @return string | ||
64 | - */ | ||
65 | - public function actionIndex() | ||
66 | - { | ||
67 | - return $this->render('index'); | ||
68 | - } | ||
69 | - | ||
70 | - /** | ||
71 | - * Login action. | ||
72 | - * | ||
73 | - * @return string | ||
74 | - */ | ||
75 | - public function actionLogin() | ||
76 | - { | ||
77 | - if (!Yii::$app->user->isGuest) { | 48 | + ]; |
49 | + } | ||
50 | + | ||
51 | + /** | ||
52 | + * @inheritdoc | ||
53 | + */ | ||
54 | + public function actions() | ||
55 | + { | ||
56 | + return [ | ||
57 | + 'error' => [ | ||
58 | + 'class' => 'yii\web\ErrorAction', | ||
59 | + ], | ||
60 | + ]; | ||
61 | + } | ||
62 | + | ||
63 | + /** | ||
64 | + * Displays homepage. | ||
65 | + * | ||
66 | + * @return string | ||
67 | + */ | ||
68 | + public function actionIndex() | ||
69 | + { | ||
70 | + return $this->render('index'); | ||
71 | + } | ||
72 | + | ||
73 | + /** | ||
74 | + * Login action. | ||
75 | + * | ||
76 | + * @return string | ||
77 | + */ | ||
78 | + public function actionLogin() | ||
79 | + { | ||
80 | + if (!Yii::$app->user->isGuest) { | ||
81 | + return $this->goHome(); | ||
82 | + } | ||
83 | + | ||
84 | + $model = new LoginForm(); | ||
85 | + if ($model->load(Yii::$app->request->post()) && $model->login()) { | ||
86 | + return $this->goBack(); | ||
87 | + } else { | ||
88 | + return $this->renderPartial( | ||
89 | + 'login', | ||
90 | + [ | ||
91 | + 'model' => $model, | ||
92 | + ] | ||
93 | + ); | ||
94 | + } | ||
95 | + } | ||
96 | + | ||
97 | + /** | ||
98 | + * Logout action. | ||
99 | + * | ||
100 | + * @return string | ||
101 | + */ | ||
102 | + public function actionLogout() | ||
103 | + { | ||
104 | + Yii::$app->user->logout(); | ||
105 | + | ||
78 | return $this->goHome(); | 106 | return $this->goHome(); |
79 | } | 107 | } |
80 | - | ||
81 | - $model = new LoginForm(); | ||
82 | - if ($model->load(Yii::$app->request->post()) && $model->login()) { | ||
83 | - return $this->goBack(); | ||
84 | - } else { | ||
85 | - return $this->renderPartial('login', [ | ||
86 | - 'model' => $model, | ||
87 | - ]); | 108 | + |
109 | + public function actionAnalytic() | ||
110 | + { | ||
111 | + return $this->renderPartial('analytic'); | ||
88 | } | 112 | } |
89 | } | 113 | } |
90 | - | ||
91 | - /** | ||
92 | - * Logout action. | ||
93 | - * | ||
94 | - * @return string | ||
95 | - */ | ||
96 | - public function actionLogout() | ||
97 | - { | ||
98 | - Yii::$app->user->logout(); | ||
99 | - | ||
100 | - return $this->goHome(); | ||
101 | - } | ||
102 | - | ||
103 | - public function actionAnalytic() | ||
104 | - { | ||
105 | - return $this->renderPartial('analytic'); | ||
106 | - } | ||
107 | -} |
backend/views/layouts/main.php
@@ -7,25 +7,24 @@ | @@ -7,25 +7,24 @@ | ||
7 | 7 | ||
8 | use artbox\core\assets\ArtboxCoreAsset; | 8 | use artbox\core\assets\ArtboxCoreAsset; |
9 | use artbox\core\models\User; | 9 | use artbox\core\models\User; |
10 | + use artbox\core\models\UserData; | ||
10 | use artbox\core\widgets\FeedbackWidget; | 11 | use artbox\core\widgets\FeedbackWidget; |
11 | use artbox\core\widgets\FlashWidget; | 12 | use artbox\core\widgets\FlashWidget; |
12 | - use hiqdev\assets\icheck\iCheckAsset; | ||
13 | - use hiqdev\assets\pnotify\PNotifyAsset; | ||
14 | use yii\bootstrap\Html; | 13 | use yii\bootstrap\Html; |
15 | use yii\web\UrlManager; | 14 | use yii\web\UrlManager; |
16 | use yii\web\View; | 15 | use yii\web\View; |
17 | use yii\widgets\Breadcrumbs; | 16 | use yii\widgets\Breadcrumbs; |
18 | use yiister\gentelella\widgets\Menu; | 17 | use yiister\gentelella\widgets\Menu; |
19 | 18 | ||
20 | - yiister\gentelella\assets\Asset::register($this); | ||
21 | - PNotifyAsset::register($this); | ||
22 | ArtboxCoreAsset::register($this); | 19 | ArtboxCoreAsset::register($this); |
23 | - iCheckAsset::register($this); | ||
24 | 20 | ||
25 | /** | 21 | /** |
26 | * @var User $user | 22 | * @var User $user |
27 | */ | 23 | */ |
28 | $user = \Yii::$app->user->identity; | 24 | $user = \Yii::$app->user->identity; |
25 | + /** | ||
26 | + * @var UserData $userData | ||
27 | + */ | ||
29 | $userData = $user->ensureExistance(); | 28 | $userData = $user->ensureExistance(); |
30 | ?> | 29 | ?> |
31 | <?php $this->beginPage(); ?> | 30 | <?php $this->beginPage(); ?> |
1 | +<?php | ||
2 | + /** | ||
3 | + * @var View $this | ||
4 | + * @var Settings $model | ||
5 | + */ | ||
6 | + | ||
7 | + use common\models\Settings; | ||
8 | + use artbox\gentelella\widgets\XPanel; | ||
9 | + use yii\bootstrap\ActiveForm; | ||
10 | + use yii\bootstrap\Html; | ||
11 | + use yii\web\View; | ||
12 | + | ||
13 | + $this->title = 'Settings'; | ||
14 | + | ||
15 | + $this->params[ 'breadcrumbs' ][] = $this->title; | ||
16 | +?> | ||
17 | + | ||
18 | +<?php $panel = XPanel::begin( | ||
19 | + [ | ||
20 | + 'title' => $this->title, | ||
21 | + 'toolbar' => false, | ||
22 | + ] | ||
23 | +); ?> | ||
24 | + | ||
25 | +<div class="settings-form"> | ||
26 | + | ||
27 | + <?php | ||
28 | + $form = ActiveForm::begin(); | ||
29 | + | ||
30 | + echo $form->field($model, 'id') | ||
31 | + ->textInput(); | ||
32 | + | ||
33 | + echo $form->field($model, 'name') | ||
34 | + ->textInput(); | ||
35 | + | ||
36 | + echo $form->field($model, 'description') | ||
37 | + ->textInput(); | ||
38 | + | ||
39 | + echo $form->field($model, 'analytics') | ||
40 | + ->textarea( | ||
41 | + [ | ||
42 | + 'rows' => 11, | ||
43 | + ] | ||
44 | + ); | ||
45 | + | ||
46 | + echo Html::submitButton( | ||
47 | + 'Save', | ||
48 | + [ | ||
49 | + 'class' => 'btn btn-primary', | ||
50 | + ] | ||
51 | + ); | ||
52 | + ActiveForm::end(); | ||
53 | + ?> | ||
54 | + | ||
55 | +</div> | ||
56 | + | ||
57 | +<?php $panel::end(); ?> |
backend/views/site/error.php
1 | <?php | 1 | <?php |
2 | - | ||
3 | -/* @var $this yii\web\View */ | ||
4 | -/* @var $name string */ | ||
5 | -/* @var $message string */ | ||
6 | -/* @var $exception Exception */ | ||
7 | - | ||
8 | -use yii\helpers\Html; | ||
9 | - | ||
10 | -$this->title = $name; | 2 | + use yii\helpers\Html; |
3 | + | ||
4 | + /** | ||
5 | + * @var $this yii\web\View | ||
6 | + * @var $name string | ||
7 | + * @var $message string | ||
8 | + * @var $exception Exception | ||
9 | + */ | ||
10 | + | ||
11 | + $this->title = $name; | ||
11 | ?> | 12 | ?> |
12 | <div class="site-error"> | 13 | <div class="site-error"> |
13 | - | 14 | + |
14 | <h1><?= Html::encode($this->title) ?></h1> | 15 | <h1><?= Html::encode($this->title) ?></h1> |
15 | - | 16 | + |
16 | <div class="alert alert-danger"> | 17 | <div class="alert alert-danger"> |
17 | <?= nl2br(Html::encode($message)) ?> | 18 | <?= nl2br(Html::encode($message)) ?> |
18 | </div> | 19 | </div> |
19 | - | 20 | + |
20 | <p> | 21 | <p> |
21 | The above error occurred while the Web server was processing your request. | 22 | The above error occurred while the Web server was processing your request. |
22 | </p> | 23 | </p> |
backend/views/site/login.php
@@ -3,8 +3,10 @@ | @@ -3,8 +3,10 @@ | ||
3 | /** | 3 | /** |
4 | * @var string $content | 4 | * @var string $content |
5 | * @var \yii\web\View $this | 5 | * @var \yii\web\View $this |
6 | + * @var LoginForm $model | ||
6 | */ | 7 | */ |
7 | 8 | ||
9 | + use common\models\LoginForm; | ||
8 | use hiqdev\assets\icheck\iCheckAsset; | 10 | use hiqdev\assets\icheck\iCheckAsset; |
9 | use yii\helpers\Html; | 11 | use yii\helpers\Html; |
10 | use yii\web\View; | 12 | use yii\web\View; |
@@ -26,72 +28,68 @@ JS; | @@ -26,72 +28,68 @@ JS; | ||
26 | <?php $this->beginPage(); ?> | 28 | <?php $this->beginPage(); ?> |
27 | <!DOCTYPE html> | 29 | <!DOCTYPE html> |
28 | <html lang="<?= Yii::$app->language ?>"> | 30 | <html lang="<?= Yii::$app->language ?>"> |
29 | -<head> | ||
30 | - <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/> | ||
31 | - <meta charset="<?= Yii::$app->charset ?>"/> | ||
32 | - <meta http-equiv="X-UA-Compatible" content="IE=edge"/> | ||
33 | - <meta name="viewport" content="width=device-width, initial-scale=1"/> | ||
34 | - <?= Html::csrfMetaTags() ?> | ||
35 | - <title><?= Html::encode($this->title) ?></title> | ||
36 | - <?php $this->head() ?> | ||
37 | - <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries --> | ||
38 | - <!--[if lt IE 9]> | ||
39 | - <script src="https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js"></script> | ||
40 | - <script src="https://oss.maxcdn.com/respond/1.4.2/respond.min.js"></script> | ||
41 | - <![endif]--> | ||
42 | -</head> | ||
43 | -<body> | ||
44 | -<?php $this->beginBody(); ?> | ||
45 | -<div class="container" style="background-color: #F7F7F7; height: 100%"> | ||
46 | - <div class="row" style="margin-top: 10%"> | ||
47 | - <div class="col-md-4 col-md-offset-4"> | ||
48 | - | ||
49 | - | ||
50 | - <div class="site-login x_panel"> | ||
51 | - <div class="x_title"> | ||
52 | - <h1><?= Html::encode($this->title) ?></h1> | ||
53 | - | ||
54 | - <p>Please fill out the following fields to login:</p> | ||
55 | - </div> | ||
56 | - <div class="row"> | ||
57 | - <div> | ||
58 | - <?php $form = ActiveForm::begin( | ||
59 | - [ | ||
60 | - 'id' => 'login-form', | ||
61 | - 'validateOnBlur' => false, | ||
62 | - ] | ||
63 | - ); ?> | ||
64 | - | ||
65 | - <?= $form->field($model, 'username') | ||
66 | - ->textInput([ 'autofocus' => true ]) ?> | ||
67 | - | ||
68 | - <?= $form->field($model, 'password') | ||
69 | - ->passwordInput() ?> | ||
70 | - | ||
71 | - <?= $form->field($model, 'rememberMe') | ||
72 | - ->checkbox() ?> | ||
73 | - | ||
74 | - <div class="form-group"> | ||
75 | - <?= Html::submitButton('Login', | ||
76 | - [ | ||
77 | - 'class' => 'btn btn-default', | ||
78 | - 'name' => 'login-button', | ||
79 | - ] | ||
80 | - ) ?> | 31 | + <head> |
32 | + <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/> | ||
33 | + <meta charset="<?= Yii::$app->charset ?>"/> | ||
34 | + <meta http-equiv="X-UA-Compatible" content="IE=edge"/> | ||
35 | + <meta name="viewport" content="width=device-width, initial-scale=1"/> | ||
36 | + <?= Html::csrfMetaTags() ?> | ||
37 | + <title><?= Html::encode($this->title) ?></title> | ||
38 | + <?php $this->head() ?> | ||
39 | + </head> | ||
40 | + <body> | ||
41 | + <?php $this->beginBody(); ?> | ||
42 | + <div class="container" style="background-color: #F7F7F7; height: 100%"> | ||
43 | + <div class="row" style="margin-top: 10%"> | ||
44 | + <div class="col-md-4 col-md-offset-4"> | ||
45 | + | ||
46 | + | ||
47 | + <div class="site-login x_panel"> | ||
48 | + <div class="x_title"> | ||
49 | + <h1><?= Html::encode($this->title) ?></h1> | ||
50 | + | ||
51 | + <p>Please fill out the following fields to login:</p> | ||
52 | + </div> | ||
53 | + <div class="row"> | ||
54 | + <div> | ||
55 | + <?php $form = ActiveForm::begin( | ||
56 | + [ | ||
57 | + 'id' => 'login-form', | ||
58 | + 'validateOnBlur' => false, | ||
59 | + ] | ||
60 | + ); ?> | ||
61 | + | ||
62 | + <?= $form->field($model, 'username') | ||
63 | + ->textInput([ 'autofocus' => true ]) ?> | ||
64 | + | ||
65 | + <?= $form->field($model, 'password') | ||
66 | + ->passwordInput() ?> | ||
67 | + | ||
68 | + <?= $form->field($model, 'rememberMe') | ||
69 | + ->checkbox() ?> | ||
70 | + | ||
71 | + <div class="form-group"> | ||
72 | + <?= Html::submitButton( | ||
73 | + 'Login', | ||
74 | + [ | ||
75 | + 'class' => 'btn btn-default', | ||
76 | + 'name' => 'login-button', | ||
77 | + ] | ||
78 | + ) ?> | ||
79 | + </div> | ||
80 | + | ||
81 | + <?php ActiveForm::end(); ?> | ||
82 | + </div> | ||
83 | + </div> | ||
84 | + </div> | ||
85 | + | ||
86 | + | ||
87 | + </div> | ||
81 | </div> | 88 | </div> |
82 | - | ||
83 | - <?php ActiveForm::end(); ?> | ||
84 | - </div> | 89 | + |
85 | </div> | 90 | </div> |
86 | - </div> | ||
87 | - | ||
88 | - | ||
89 | - </div> | ||
90 | - </div> | ||
91 | - | ||
92 | -</div> | ||
93 | -<?php $this->endBody(); ?> | ||
94 | -</body> | 91 | + <?php $this->endBody(); ?> |
92 | + </body> | ||
95 | </html> | 93 | </html> |
96 | <?php $this->endPage(); ?> | 94 | <?php $this->endPage(); ?> |
97 | 95 |
backend/assets/css/custom.css renamed to backend/web/css/custom.css
backend/assets/js/custom.js renamed to backend/web/js/custom.js
common/config/.gitignore
1 | +<?php | ||
2 | + namespace common\models; | ||
3 | + | ||
4 | + use yii2tech\filedb\ActiveRecord; | ||
5 | + use Yii; | ||
6 | + | ||
7 | + /** | ||
8 | + * Class Settings | ||
9 | + * | ||
10 | + * @package artbox\core\models | ||
11 | + * @property string $name | ||
12 | + * @property string $id | ||
13 | + * @property string $description | ||
14 | + * @property string $analytics | ||
15 | + */ | ||
16 | + class Settings extends ActiveRecord | ||
17 | + { | ||
18 | + /** | ||
19 | + * @inheritdoc | ||
20 | + */ | ||
21 | + public function rules() | ||
22 | + { | ||
23 | + return [ | ||
24 | + [ | ||
25 | + [ | ||
26 | + 'name', | ||
27 | + 'description', | ||
28 | + 'id', | ||
29 | + 'analytics', | ||
30 | + ], | ||
31 | + 'string', | ||
32 | + ], | ||
33 | + ]; | ||
34 | + } | ||
35 | + | ||
36 | + /** | ||
37 | + * @inheritdoc | ||
38 | + */ | ||
39 | + public static function fileName() | ||
40 | + { | ||
41 | + return 'settings'; | ||
42 | + } | ||
43 | + | ||
44 | + /** | ||
45 | + * @inheritdoc | ||
46 | + */ | ||
47 | + public function attributeLabels() | ||
48 | + { | ||
49 | + return [ | ||
50 | + 'id' => Yii::t('core', 'ID'), | ||
51 | + 'name' => Yii::t('core', 'Name'), | ||
52 | + 'description' => Yii::t('core', 'Description'), | ||
53 | + 'analytics' => Yii::t('core', 'Google Analytics Code'), | ||
54 | + ]; | ||
55 | + } | ||
56 | + | ||
57 | + /** | ||
58 | + * Get Settings model instance | ||
59 | + * | ||
60 | + * @return Settings | ||
61 | + */ | ||
62 | + public static function getInstance() | ||
63 | + { | ||
64 | + return self::findOne([ 'id' => 1 ]); | ||
65 | + } | ||
66 | + } | ||
67 | + | ||
0 | \ No newline at end of file | 68 | \ No newline at end of file |
composer.json
1 | { | 1 | { |
2 | - "name": "yiisoft/yii2-app-advanced", | ||
3 | - "description": "Yii 2 Advanced Project Template", | 2 | + "name": "artweb/artbox-basic", |
3 | + "description": "Artbox Basic Project Template", | ||
4 | "keywords": [ | 4 | "keywords": [ |
5 | "yii2", | 5 | "yii2", |
6 | - "framework", | ||
7 | - "advanced", | 6 | + "cms", |
7 | + "artweb", | ||
8 | + "artbox", | ||
9 | + "basic", | ||
8 | "project template" | 10 | "project template" |
9 | ], | 11 | ], |
10 | - "homepage": "http://www.yiiframework.com/", | 12 | + "homepage": "http://artweb.ua", |
11 | "type": "project", | 13 | "type": "project", |
12 | "license": "BSD-3-Clause", | 14 | "license": "BSD-3-Clause", |
13 | "support": { | 15 | "support": { |
14 | - "issues": "https://github.com/yiisoft/yii2/issues?state=open", | ||
15 | - "forum": "http://www.yiiframework.com/forum/", | ||
16 | - "wiki": "http://www.yiiframework.com/wiki/", | ||
17 | - "irc": "irc://irc.freenode.net/yii", | ||
18 | - "source": "https://github.com/yiisoft/yii2" | 16 | + "wiki": "http://www.yiiframework.com/wiki/" |
19 | }, | 17 | }, |
20 | - "minimum-stability": "stable", | 18 | + "minimum-stability": "dev", |
21 | "require": { | 19 | "require": { |
22 | - "php": ">=5.4.0", | ||
23 | - "yiisoft/yii2": "~2.0.6", | ||
24 | - "yiisoft/yii2-bootstrap": "~2.0.0", | ||
25 | - "yiisoft/yii2-swiftmailer": "~2.0.0", | ||
26 | - "insolita/yii2-migration-generator": "~2.2", | 20 | + "php": ">=7.0", |
21 | + "artweb/artbox-core": "*", | ||
22 | + "yiisoft/yii2": "~2.0", | ||
23 | + "yiisoft/yii2-bootstrap": "~2.0", | ||
24 | + "yiisoft/yii2-swiftmailer": "~2.0", | ||
25 | + "noam148/yii2-image-manager": "~1.0", | ||
27 | "yiister/yii2-gentelella": "~1.0", | 26 | "yiister/yii2-gentelella": "~1.0", |
28 | - "hiqdev/yii2-asset-icheck": "1.0.2.5", | ||
29 | - "noam148/yii2-image-manager": "*", | ||
30 | - "yii2tech/filedb": "^1.0", | ||
31 | - "hiqdev/yii2-asset-pnotify": "^2.1", | ||
32 | - "yiisoft/yii2-imagine": "*", | ||
33 | - "2amigos/yii2-tinymce-widget": "*", | ||
34 | - "google/apiclient": "^2.0", | ||
35 | - "2amigos/yii2-highcharts-widget": "~1.0" | 27 | + "yii2tech/filedb": "~1.0", |
28 | + "hiqdev/yii2-asset-pnotify": "~2.0", | ||
29 | + "hiqdev/yii2-asset-icheck": "~1.0", | ||
30 | + "google/apiclient": "^2.0" | ||
36 | }, | 31 | }, |
37 | "require-dev": { | 32 | "require-dev": { |
38 | "yiisoft/yii2-debug": "~2.0.0", | 33 | "yiisoft/yii2-debug": "~2.0.0", |
@@ -51,12 +46,16 @@ | @@ -51,12 +46,16 @@ | ||
51 | } | 46 | } |
52 | }, | 47 | }, |
53 | "scripts": { | 48 | "scripts": { |
54 | - "post-install-cmd": "php init --env=Development --overwrite=n" | 49 | + "post-install-cmd": "php init --env=Development --overwrite=All --dbinit --dbtype=postgresql --dbname=artbox_basic --username=postgres --password=postgres --migrationPath=artweb/artbox-core/migrations --migrate=yes --defaultuser --checkdb" |
55 | }, | 50 | }, |
56 | - "autoload": { | ||
57 | - "psr-4": { | ||
58 | - "artbox\\core\\": "artweb/artbox-core/", | ||
59 | - "artbox\\gentelella\\": "artweb/artbox-gentelella/" | 51 | + "repositories": [ |
52 | + { | ||
53 | + "type": "vsc", | ||
54 | + "url": "git@gitlab.artweb.com.ua:yarik.nechyporuk/artbox-core.git" | ||
55 | + }, | ||
56 | + { | ||
57 | + "type": "vsc", | ||
58 | + "url": "git@gitlab.artweb.com.ua:yarik.nechyporuk/artbox-gentelella.git" | ||
60 | } | 59 | } |
61 | - } | 60 | + ] |
62 | } | 61 | } |
1 | +<?php | ||
2 | + namespace console\controllers; | ||
3 | + | ||
4 | + use yii\console\Controller; | ||
5 | + use yii\db\Connection; | ||
6 | + use yii\db\Exception; | ||
7 | + use yii\helpers\ArrayHelper; | ||
8 | + | ||
9 | + /** | ||
10 | + * Class CheckController | ||
11 | + * | ||
12 | + * @package console\controllers | ||
13 | + */ | ||
14 | + class CheckController extends Controller | ||
15 | + { | ||
16 | + public function actionConnection($path) | ||
17 | + { | ||
18 | + $connection = \Yii::createObject(require( $path )); | ||
19 | + try { | ||
20 | + $connection->open(); | ||
21 | + } catch (Exception $exception) { | ||
22 | + exit( 1 ); | ||
23 | + } | ||
24 | + exit( 0 ); | ||
25 | + } | ||
26 | + } | ||
0 | \ No newline at end of file | 27 | \ No newline at end of file |
console/controllers/CreateController.php
@@ -12,18 +12,30 @@ | @@ -12,18 +12,30 @@ | ||
12 | */ | 12 | */ |
13 | class CreateController extends Controller | 13 | class CreateController extends Controller |
14 | { | 14 | { |
15 | - public function actionUser() | 15 | + public function actionUser($username = null, $email = null, $password = null) |
16 | { | 16 | { |
17 | - $user = new User(); | ||
18 | - $user->username = 'admin'; | ||
19 | - $user->email = 'admin@example.com'; | ||
20 | - $user->setPassword('admin321'); | 17 | + $username = $username ? : 'admin'; |
18 | + $user = User::find() | ||
19 | + ->where([ 'username' => $username ]) | ||
20 | + ->one(); | ||
21 | + if (empty( $user )) { | ||
22 | + $user = new User(); | ||
23 | + $user->username = $username; | ||
24 | + } | ||
25 | + $user->email = $email ? : 'admin@example.com'; | ||
26 | + $user->setPassword($password ? : 'admin321'); | ||
21 | $user->generateAuthKey(); | 27 | $user->generateAuthKey(); |
22 | 28 | ||
23 | if ($user->save()) { | 29 | if ($user->save()) { |
24 | - $this->stdout('User created' . "\n", Console::FG_GREEN); | 30 | + if ($user->isNewRecord) { |
31 | + $this->stdout('User created' . "\n", Console::FG_GREEN); | ||
32 | + } else { | ||
33 | + $this->stdout('User updated' . "\n", Console::FG_GREEN); | ||
34 | + } | ||
25 | } else { | 35 | } else { |
26 | $this->stdout('Error!' . "\n", Console::FG_RED); | 36 | $this->stdout('Error!' . "\n", Console::FG_RED); |
37 | + return self::EXIT_CODE_ERROR; | ||
27 | } | 38 | } |
39 | + return self::EXIT_CODE_NORMAL; | ||
28 | } | 40 | } |
29 | } | 41 | } |
30 | \ No newline at end of file | 42 | \ No newline at end of file |
1 | +<?php | ||
2 | + return [ | ||
3 | + 'class' => 'yii\db\Connection', | ||
4 | + 'dsn' => '', | ||
5 | + 'username' => '', | ||
6 | + 'password' => '', | ||
7 | + 'charset' => 'utf8', | ||
8 | + 'schemaMap' => [ | ||
9 | + 'pgsql' => [ | ||
10 | + 'class' => 'yii\db\pgsql\Schema', | ||
11 | + 'defaultSchema' => '', | ||
12 | + ], | ||
13 | + ], | ||
14 | + ]; | ||
15 | + //DSN: pgsql:host=127.0.0.1;port=5432;dbname=name | ||
0 | \ No newline at end of file | 16 | \ No newline at end of file |
environments/dev/common/config/main-local.php
1 | <?php | 1 | <?php |
2 | return [ | 2 | return [ |
3 | 'components' => [ | 3 | 'components' => [ |
4 | - 'db' => [ | ||
5 | - 'class' => 'yii\db\Connection', | ||
6 | - 'dsn' => 'mysql:host=localhost;dbname=yii2advanced', | ||
7 | - 'username' => 'root', | ||
8 | - 'password' => '', | ||
9 | - 'charset' => 'utf8', | ||
10 | - ], | 4 | + 'db' => '', |
11 | 'mailer' => [ | 5 | 'mailer' => [ |
12 | 'class' => 'yii\swiftmailer\Mailer', | 6 | 'class' => 'yii\swiftmailer\Mailer', |
13 | 'viewPath' => '@common/mail', | 7 | 'viewPath' => '@common/mail', |
environments/index.php
@@ -30,14 +30,14 @@ | @@ -30,14 +30,14 @@ | ||
30 | */ | 30 | */ |
31 | return [ | 31 | return [ |
32 | 'Development' => [ | 32 | 'Development' => [ |
33 | - 'path' => 'dev', | ||
34 | - 'setWritable' => [ | 33 | + 'path' => 'dev', |
34 | + 'setWritable' => [ | ||
35 | 'backend/runtime', | 35 | 'backend/runtime', |
36 | 'backend/web/assets', | 36 | 'backend/web/assets', |
37 | 'frontend/runtime', | 37 | 'frontend/runtime', |
38 | 'frontend/web/assets', | 38 | 'frontend/web/assets', |
39 | ], | 39 | ], |
40 | - 'setExecutable' => [ | 40 | + 'setExecutable' => [ |
41 | 'yii', | 41 | 'yii', |
42 | 'yii_test', | 42 | 'yii_test', |
43 | ], | 43 | ], |
@@ -45,21 +45,23 @@ return [ | @@ -45,21 +45,23 @@ return [ | ||
45 | 'backend/config/main-local.php', | 45 | 'backend/config/main-local.php', |
46 | 'frontend/config/main-local.php', | 46 | 'frontend/config/main-local.php', |
47 | ], | 47 | ], |
48 | + 'setDbConnection' => 'common/config/main-local.php', | ||
48 | ], | 49 | ], |
49 | 'Production' => [ | 50 | 'Production' => [ |
50 | - 'path' => 'prod', | ||
51 | - 'setWritable' => [ | 51 | + 'path' => 'prod', |
52 | + 'setWritable' => [ | ||
52 | 'backend/runtime', | 53 | 'backend/runtime', |
53 | 'backend/web/assets', | 54 | 'backend/web/assets', |
54 | 'frontend/runtime', | 55 | 'frontend/runtime', |
55 | 'frontend/web/assets', | 56 | 'frontend/web/assets', |
56 | ], | 57 | ], |
57 | - 'setExecutable' => [ | 58 | + 'setExecutable' => [ |
58 | 'yii', | 59 | 'yii', |
59 | ], | 60 | ], |
60 | 'setCookieValidationKey' => [ | 61 | 'setCookieValidationKey' => [ |
61 | 'backend/config/main-local.php', | 62 | 'backend/config/main-local.php', |
62 | 'frontend/config/main-local.php', | 63 | 'frontend/config/main-local.php', |
63 | ], | 64 | ], |
65 | + 'setDbConnection' => 'common/config/main-local.php', | ||
64 | ], | 66 | ], |
65 | ]; | 67 | ]; |
environments/prod/common/config/main-local.php
1 | <?php | 1 | <?php |
2 | return [ | 2 | return [ |
3 | 'components' => [ | 3 | 'components' => [ |
4 | - 'db' => [ | ||
5 | - 'class' => 'yii\db\Connection', | ||
6 | - 'dsn' => 'mysql:host=localhost;dbname=yii2advanced', | ||
7 | - 'username' => 'root', | ||
8 | - 'password' => '', | ||
9 | - 'charset' => 'utf8', | ||
10 | - ], | 4 | + 'db' => '', |
11 | 'mailer' => [ | 5 | 'mailer' => [ |
12 | 'class' => 'yii\swiftmailer\Mailer', | 6 | 'class' => 'yii\swiftmailer\Mailer', |
13 | 'viewPath' => '@common/mail', | 7 | 'viewPath' => '@common/mail', |
frontend/views/layouts/main.php
@@ -4,7 +4,7 @@ | @@ -4,7 +4,7 @@ | ||
4 | /* @var $content string */ | 4 | /* @var $content string */ |
5 | 5 | ||
6 | use artbox\core\components\SeoComponent; | 6 | use artbox\core\components\SeoComponent; |
7 | - use artbox\core\models\Settings; | 7 | + use common\models\Settings; |
8 | use artbox\core\models\User; | 8 | use artbox\core\models\User; |
9 | use yii\helpers\Html; | 9 | use yii\helpers\Html; |
10 | use yii\bootstrap\Nav; | 10 | use yii\bootstrap\Nav; |
@@ -31,7 +31,7 @@ | @@ -31,7 +31,7 @@ | ||
31 | <meta charset="<?= Yii::$app->charset ?>"> | 31 | <meta charset="<?= Yii::$app->charset ?>"> |
32 | <meta name="viewport" content="width=device-width, initial-scale=1"> | 32 | <meta name="viewport" content="width=device-width, initial-scale=1"> |
33 | <?= Html::csrfMetaTags() ?> | 33 | <?= Html::csrfMetaTags() ?> |
34 | - <title><?= Html::encode($seo->title) ?></title> | 34 | + <title><?= Html::encode($seo->title) ?></title> |
35 | <?php $this->head() ?> | 35 | <?php $this->head() ?> |
36 | </head> | 36 | </head> |
37 | <body> | 37 | <body> |
1 | #!/usr/bin/env php | 1 | #!/usr/bin/env php |
2 | <?php | 2 | <?php |
3 | -/** | ||
4 | - * Yii Application Initialization Tool | ||
5 | - * | ||
6 | - * In order to run in non-interactive mode: | ||
7 | - * | ||
8 | - * init --env=Development --overwrite=n | ||
9 | - * | ||
10 | - * @author Alexander Makarov <sam@rmcreative.ru> | ||
11 | - * | ||
12 | - * @link http://www.yiiframework.com/ | ||
13 | - * @copyright Copyright (c) 2008 Yii Software LLC | ||
14 | - * @license http://www.yiiframework.com/license/ | ||
15 | - */ | ||
16 | - | ||
17 | -if (!extension_loaded('openssl')) { | ||
18 | - die('The OpenSSL PHP extension is required by Yii2.'); | ||
19 | -} | ||
20 | - | ||
21 | -$params = getParams(); | ||
22 | -$root = str_replace('\\', '/', __DIR__); | ||
23 | -$envs = require("$root/environments/index.php"); | ||
24 | -$envNames = array_keys($envs); | ||
25 | - | ||
26 | -echo "Yii Application Initialization Tool v1.0\n\n"; | ||
27 | - | ||
28 | -$envName = null; | ||
29 | -if (empty($params['env']) || $params['env'] === '1') { | ||
30 | - echo "Which environment do you want the application to be initialized in?\n\n"; | ||
31 | - foreach ($envNames as $i => $name) { | ||
32 | - echo " [$i] $name\n"; | ||
33 | - } | ||
34 | - echo "\n Your choice [0-" . (count($envs) - 1) . ', or "q" to quit] '; | ||
35 | - $answer = trim(fgets(STDIN)); | ||
36 | - | ||
37 | - if (!ctype_digit($answer) || !in_array($answer, range(0, count($envs) - 1))) { | ||
38 | - echo "\n Quit initialization.\n"; | ||
39 | - exit(0); | ||
40 | - } | ||
41 | - | ||
42 | - if (isset($envNames[$answer])) { | ||
43 | - $envName = $envNames[$answer]; | ||
44 | - } | ||
45 | -} else { | ||
46 | - $envName = $params['env']; | ||
47 | -} | ||
48 | - | ||
49 | -if (!in_array($envName, $envNames)) { | ||
50 | - $envsList = implode(', ', $envNames); | ||
51 | - echo "\n $envName is not a valid environment. Try one of the following: $envsList. \n"; | ||
52 | - exit(2); | ||
53 | -} | ||
54 | - | ||
55 | -$env = $envs[$envName]; | ||
56 | - | ||
57 | -if (empty($params['env'])) { | ||
58 | - echo "\n Initialize the application under '{$envNames[$answer]}' environment? [yes|no] "; | ||
59 | - $answer = trim(fgets(STDIN)); | ||
60 | - if (strncasecmp($answer, 'y', 1)) { | ||
61 | - echo "\n Quit initialization.\n"; | ||
62 | - exit(0); | ||
63 | - } | ||
64 | -} | ||
65 | - | ||
66 | -echo "\n Start initialization ...\n\n"; | ||
67 | -$files = getFileList("$root/environments/{$env['path']}"); | ||
68 | -if (isset($env['skipFiles'])) { | ||
69 | - $skipFiles = $env['skipFiles']; | ||
70 | - array_walk($skipFiles, function(&$value) use($env, $root) { $value = "$root/$value"; }); | ||
71 | - $files = array_diff($files, array_intersect_key($env['skipFiles'], array_filter($skipFiles, 'file_exists'))); | ||
72 | -} | ||
73 | -$all = false; | ||
74 | -foreach ($files as $file) { | ||
75 | - if (!copyFile($root, "environments/{$env['path']}/$file", $file, $all, $params)) { | ||
76 | - break; | ||
77 | - } | ||
78 | -} | ||
79 | - | ||
80 | -$callbacks = ['setCookieValidationKey', 'setWritable', 'setExecutable', 'createSymlink']; | ||
81 | -foreach ($callbacks as $callback) { | ||
82 | - if (!empty($env[$callback])) { | ||
83 | - $callback($root, $env[$callback]); | ||
84 | - } | ||
85 | -} | ||
86 | - | ||
87 | -echo "\n ... initialization completed.\n\n"; | ||
88 | - | ||
89 | -function getFileList($root, $basePath = '') | ||
90 | -{ | ||
91 | - $files = []; | ||
92 | - $handle = opendir($root); | ||
93 | - while (($path = readdir($handle)) !== false) { | ||
94 | - if ($path === '.git' || $path === '.svn' || $path === '.' || $path === '..') { | ||
95 | - continue; | ||
96 | - } | ||
97 | - $fullPath = "$root/$path"; | ||
98 | - $relativePath = $basePath === '' ? $path : "$basePath/$path"; | ||
99 | - if (is_dir($fullPath)) { | ||
100 | - $files = array_merge($files, getFileList($fullPath, $relativePath)); | ||
101 | - } else { | ||
102 | - $files[] = $relativePath; | ||
103 | - } | ||
104 | - } | ||
105 | - closedir($handle); | ||
106 | - return $files; | ||
107 | -} | ||
108 | - | ||
109 | -function copyFile($root, $source, $target, &$all, $params) | ||
110 | -{ | ||
111 | - if (!is_file($root . '/' . $source)) { | ||
112 | - echo " skip $target ($source not exist)\n"; | ||
113 | - return true; | ||
114 | - } | ||
115 | - if (is_file($root . '/' . $target)) { | ||
116 | - if (file_get_contents($root . '/' . $source) === file_get_contents($root . '/' . $target)) { | ||
117 | - echo " unchanged $target\n"; | ||
118 | - return true; | ||
119 | - } | ||
120 | - if ($all) { | ||
121 | - echo " overwrite $target\n"; | ||
122 | - } else { | ||
123 | - echo " exist $target\n"; | ||
124 | - echo " ...overwrite? [Yes|No|All|Quit] "; | ||
125 | - | ||
126 | - | ||
127 | - $answer = !empty($params['overwrite']) ? $params['overwrite'] : trim(fgets(STDIN)); | ||
128 | - if (!strncasecmp($answer, 'q', 1)) { | ||
129 | - return false; | ||
130 | - } else { | ||
131 | - if (!strncasecmp($answer, 'y', 1)) { | ||
132 | - echo " overwrite $target\n"; | ||
133 | - } else { | ||
134 | - if (!strncasecmp($answer, 'a', 1)) { | ||
135 | - echo " overwrite $target\n"; | ||
136 | - $all = true; | ||
137 | - } else { | ||
138 | - echo " skip $target\n"; | ||
139 | - return true; | ||
140 | - } | ||
141 | - } | ||
142 | - } | ||
143 | - } | ||
144 | - file_put_contents($root . '/' . $target, file_get_contents($root . '/' . $source)); | ||
145 | - return true; | ||
146 | - } | ||
147 | - echo " generate $target\n"; | ||
148 | - @mkdir(dirname($root . '/' . $target), 0777, true); | ||
149 | - file_put_contents($root . '/' . $target, file_get_contents($root . '/' . $source)); | ||
150 | - return true; | ||
151 | -} | ||
152 | - | ||
153 | -function getParams() | ||
154 | -{ | ||
155 | - $rawParams = []; | ||
156 | - if (isset($_SERVER['argv'])) { | ||
157 | - $rawParams = $_SERVER['argv']; | ||
158 | - array_shift($rawParams); | ||
159 | - } | ||
160 | - | ||
161 | - $params = []; | ||
162 | - foreach ($rawParams as $param) { | ||
163 | - if (preg_match('/^--(\w+)(=(.*))?$/', $param, $matches)) { | ||
164 | - $name = $matches[1]; | ||
165 | - $params[$name] = isset($matches[3]) ? $matches[3] : true; | ||
166 | - } else { | ||
167 | - $params[] = $param; | ||
168 | - } | ||
169 | - } | ||
170 | - return $params; | ||
171 | -} | ||
172 | - | ||
173 | -function setWritable($root, $paths) | ||
174 | -{ | ||
175 | - foreach ($paths as $writable) { | ||
176 | - if (is_dir("$root/$writable")) { | ||
177 | - if (@chmod("$root/$writable", 0777)) { | ||
178 | - echo " chmod 0777 $writable\n"; | ||
179 | - } else { | ||
180 | - printError("Operation chmod not permitted for directory $writable."); | ||
181 | - } | ||
182 | - } else { | ||
183 | - printError("Directory $writable does not exist."); | ||
184 | - } | ||
185 | - } | ||
186 | -} | ||
187 | - | ||
188 | -function setExecutable($root, $paths) | ||
189 | -{ | ||
190 | - foreach ($paths as $executable) { | ||
191 | - if (file_exists("$root/$executable")) { | ||
192 | - if (@chmod("$root/$executable", 0755)) { | ||
193 | - echo " chmod 0755 $executable\n"; | ||
194 | - } else { | ||
195 | - printError("Operation chmod not permitted for $executable."); | ||
196 | - } | ||
197 | - } else { | ||
198 | - printError("$executable does not exist."); | ||
199 | - } | ||
200 | - } | ||
201 | -} | ||
202 | - | ||
203 | -function setCookieValidationKey($root, $paths) | ||
204 | -{ | ||
205 | - foreach ($paths as $file) { | ||
206 | - echo " generate cookie validation key in $file\n"; | ||
207 | - $file = $root . '/' . $file; | ||
208 | - $length = 32; | ||
209 | - $bytes = openssl_random_pseudo_bytes($length); | ||
210 | - $key = strtr(substr(base64_encode($bytes), 0, $length), '+/=', '_-.'); | ||
211 | - $content = preg_replace('/(("|\')cookieValidationKey("|\')\s*=>\s*)(""|\'\')/', "\\1'$key'", file_get_contents($file)); | ||
212 | - file_put_contents($file, $content); | ||
213 | - } | ||
214 | -} | ||
215 | - | ||
216 | -function createSymlink($root, $links) | ||
217 | -{ | ||
218 | - foreach ($links as $link => $target) { | ||
219 | - //first removing folders to avoid errors if the folder already exists | ||
220 | - @rmdir($root . "/" . $link); | ||
221 | - //next removing existing symlink in order to update the target | ||
222 | - if (is_link($root . "/" . $link)) { | ||
223 | - @unlink($root . "/" . $link); | ||
224 | - } | ||
225 | - if (@symlink($root . "/" . $target, $root . "/" . $link)) { | ||
226 | - echo " symlink $root/$target $root/$link\n"; | ||
227 | - } else { | ||
228 | - printError("Cannot create symlink $root/$target $root/$link."); | ||
229 | - } | ||
230 | - } | ||
231 | -} | ||
232 | - | ||
233 | -/** | ||
234 | - * Prints error message. | ||
235 | - * @param string $message message | ||
236 | - */ | ||
237 | -function printError($message) | ||
238 | -{ | ||
239 | - echo "\n " . formatMessage("Error. $message", ['fg-red']) . " \n"; | ||
240 | -} | ||
241 | - | ||
242 | -/** | ||
243 | - * Returns true if the stream supports colorization. ANSI colors are disabled if not supported by the stream. | ||
244 | - * | ||
245 | - * - windows without ansicon | ||
246 | - * - not tty consoles | ||
247 | - * | ||
248 | - * @return boolean true if the stream supports ANSI colors, otherwise false. | ||
249 | - */ | ||
250 | -function ansiColorsSupported() | ||
251 | -{ | ||
252 | - return DIRECTORY_SEPARATOR === '\\' | ||
253 | - ? getenv('ANSICON') !== false || getenv('ConEmuANSI') === 'ON' | ||
254 | - : function_exists('posix_isatty') && @posix_isatty(STDOUT); | ||
255 | -} | ||
256 | - | ||
257 | -/** | ||
258 | - * Get ANSI code of style. | ||
259 | - * @param string $name style name | ||
260 | - * @return integer ANSI code of style. | ||
261 | - */ | ||
262 | -function getStyleCode($name) | ||
263 | -{ | ||
264 | - $styles = [ | ||
265 | - 'bold' => 1, | ||
266 | - 'fg-black' => 30, | ||
267 | - 'fg-red' => 31, | ||
268 | - 'fg-green' => 32, | ||
269 | - 'fg-yellow' => 33, | ||
270 | - 'fg-blue' => 34, | ||
271 | - 'fg-magenta' => 35, | ||
272 | - 'fg-cyan' => 36, | ||
273 | - 'fg-white' => 37, | ||
274 | - 'bg-black' => 40, | ||
275 | - 'bg-red' => 41, | ||
276 | - 'bg-green' => 42, | ||
277 | - 'bg-yellow' => 43, | ||
278 | - 'bg-blue' => 44, | ||
279 | - 'bg-magenta' => 45, | ||
280 | - 'bg-cyan' => 46, | ||
281 | - 'bg-white' => 47, | ||
282 | - ]; | ||
283 | - return $styles[$name]; | ||
284 | -} | ||
285 | - | ||
286 | -/** | ||
287 | - * Formats message using styles if STDOUT supports it. | ||
288 | - * @param string $message message | ||
289 | - * @param string[] $styles styles | ||
290 | - * @return string formatted message. | ||
291 | - */ | ||
292 | -function formatMessage($message, $styles) | ||
293 | -{ | ||
294 | - if (empty($styles) || !ansiColorsSupported()) { | ||
295 | - return $message; | ||
296 | - } | ||
297 | - | ||
298 | - return sprintf("\x1b[%sm", implode(';', array_map('getStyleCode', $styles))) . $message . "\x1b[0m"; | ||
299 | -} | 3 | + /** |
4 | + * Yii Application Initialization Tool | ||
5 | + * In order to run in non-interactive mode: | ||
6 | + * init --env=Development --overwrite=n | ||
7 | + * | ||
8 | + * @author Alexander Makarov <sam@rmcreative.ru> | ||
9 | + * @link http://www.yiiframework.com/ | ||
10 | + * @copyright Copyright (c) 2008 Yii Software LLC | ||
11 | + * @license http://www.yiiframework.com/license/ | ||
12 | + */ | ||
13 | + require( 'Initializer.php' ); | ||
14 | + if (!extension_loaded('openssl')) { | ||
15 | + die( 'The OpenSSL PHP extension is required by Yii2.' ); | ||
16 | + } | ||
17 | + Initializer::initialize(); |