1: | <?php declare(strict_types=1); |
2: | |
3: | namespace Salient\Console\Format; |
4: | |
5: | use Salient\Console\HasConsoleRegex; |
6: | use Salient\Contract\Console\Format\FormatInterface; |
7: | use Salient\Contract\Console\Format\FormatterInterface; |
8: | use Salient\Contract\Console\Format\MessageFormatInterface; |
9: | use Salient\Contract\Console\ConsoleInterface as Console; |
10: | use Salient\Core\Concern\ImmutableTrait; |
11: | use Salient\Utility\Exception\ShouldNotHappenException; |
12: | use Salient\Utility\Regex; |
13: | use Salient\Utility\Str; |
14: | use Closure; |
15: | use LogicException; |
16: | |
17: | |
18: | |
19: | |
20: | class Formatter implements FormatterInterface, HasConsoleRegex |
21: | { |
22: | use ImmutableTrait; |
23: | |
24: | public const DEFAULT_LEVEL_PREFIX_MAP = [ |
25: | Console::LEVEL_EMERGENCY => '! ', |
26: | Console::LEVEL_ALERT => '! ', |
27: | Console::LEVEL_CRITICAL => '! ', |
28: | Console::LEVEL_ERROR => '! ', |
29: | Console::LEVEL_WARNING => '^ ', |
30: | Console::LEVEL_NOTICE => '> ', |
31: | Console::LEVEL_INFO => '- ', |
32: | Console::LEVEL_DEBUG => ': ', |
33: | ]; |
34: | |
35: | public const DEFAULT_TYPE_PREFIX_MAP = [ |
36: | Console::TYPE_PROGRESS => '⠿ ', |
37: | Console::TYPE_GROUP_START => '» ', |
38: | Console::TYPE_GROUP_END => '« ', |
39: | Console::TYPE_SUMMARY => '» ', |
40: | Console::TYPE_SUCCESS => '✔ ', |
41: | Console::TYPE_FAILURE => '✘ ', |
42: | ]; |
43: | |
44: | |
45: | |
46: | |
47: | |
48: | |
49: | protected const SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏']; |
50: | |
51: | private const TAG_MAP = [ |
52: | '___' => self::TAG_HEADING, |
53: | '***' => self::TAG_HEADING, |
54: | '##' => self::TAG_HEADING, |
55: | '__' => self::TAG_BOLD, |
56: | '**' => self::TAG_BOLD, |
57: | '_' => self::TAG_ITALIC, |
58: | '*' => self::TAG_ITALIC, |
59: | '<' => self::TAG_UNDERLINE, |
60: | '~~' => self::TAG_LOW_PRIORITY, |
61: | ]; |
62: | |
63: | private static self $NullFormatter; |
64: | private static TagFormats $NullTagFormats; |
65: | private static MessageFormats $NullMessageFormats; |
66: | private static TagFormats $LoopbackTagFormats; |
67: | private TagFormats $TagFormats; |
68: | private MessageFormats $MessageFormats; |
69: | |
70: | private Closure $WidthCallback; |
71: | |
72: | private array $LevelPrefixMap; |
73: | |
74: | private array $TypePrefixMap; |
75: | |
76: | private array $SpinnerState; |
77: | |
78: | |
79: | |
80: | |
81: | |
82: | |
83: | |
84: | |
85: | public function __construct( |
86: | ?TagFormats $tagFormats = null, |
87: | ?MessageFormats $messageFormats = null, |
88: | ?Closure $widthCallback = null, |
89: | array $levelPrefixMap = Formatter::DEFAULT_LEVEL_PREFIX_MAP, |
90: | array $typePrefixMap = Formatter::DEFAULT_TYPE_PREFIX_MAP |
91: | ) { |
92: | $this->TagFormats = $tagFormats ?? self::getNullTagFormats(); |
93: | $this->MessageFormats = $messageFormats ?? self::getNullMessageFormats(); |
94: | $this->WidthCallback = $widthCallback ?? fn() => null; |
95: | $this->LevelPrefixMap = $levelPrefixMap; |
96: | $this->TypePrefixMap = $typePrefixMap; |
97: | $spinnerState = [0, null]; |
98: | $this->SpinnerState = &$spinnerState; |
99: | } |
100: | |
101: | |
102: | |
103: | |
104: | public function removesEscapes(): bool |
105: | { |
106: | return $this->TagFormats->removesEscapes(); |
107: | } |
108: | |
109: | |
110: | |
111: | |
112: | public function wrapsAfterFormatting(): bool |
113: | { |
114: | return $this->TagFormats->wrapsAfterFormatting(); |
115: | } |
116: | |
117: | |
118: | |
119: | |
120: | public function withRemoveEscapes(bool $remove = true) |
121: | { |
122: | return $this->with('TagFormats', $this->TagFormats->withRemoveEscapes($remove)); |
123: | } |
124: | |
125: | |
126: | |
127: | |
128: | public function withWrapAfterFormatting(bool $value = true) |
129: | { |
130: | return $this->with('TagFormats', $this->TagFormats->withWrapAfterFormatting($value)); |
131: | } |
132: | |
133: | |
134: | |
135: | |
136: | public function getTagFormat(int $tag): FormatInterface |
137: | { |
138: | return $this->TagFormats->getFormat($tag); |
139: | } |
140: | |
141: | |
142: | |
143: | |
144: | public function getMessageFormat( |
145: | int $level, |
146: | int $type = Console::TYPE_STANDARD |
147: | ): MessageFormatInterface { |
148: | return $this->MessageFormats->getFormat($level, $type); |
149: | } |
150: | |
151: | |
152: | |
153: | |
154: | public function getMessagePrefix( |
155: | int $level, |
156: | int $type = Console::TYPE_STANDARD |
157: | ): string { |
158: | if ( |
159: | $type === Console::TYPE_UNDECORATED |
160: | || $type === Console::TYPE_UNFORMATTED |
161: | ) { |
162: | return ''; |
163: | } |
164: | |
165: | if ($type === Console::TYPE_PROGRESS) { |
166: | $now = (float) (hrtime(true) / 1000); |
167: | if ($this->SpinnerState[1] === null) { |
168: | $this->SpinnerState[1] = $now; |
169: | } elseif ($now - $this->SpinnerState[1] >= 80000) { |
170: | $this->SpinnerState[0]++; |
171: | $this->SpinnerState[0] %= count(static::SPINNER_FRAMES); |
172: | $this->SpinnerState[1] = $now; |
173: | } |
174: | $prefix = static::SPINNER_FRAMES[$this->SpinnerState[0]] . ' '; |
175: | } |
176: | |
177: | return $prefix |
178: | ?? $this->TypePrefixMap[$type] |
179: | ?? $this->LevelPrefixMap[$level] |
180: | ?? ''; |
181: | } |
182: | |
183: | |
184: | |
185: | |
186: | public function format( |
187: | string $string, |
188: | bool $unwrap = false, |
189: | $wrapTo = null, |
190: | bool $unformat = false, |
191: | string $break = "\n" |
192: | ): string { |
193: | if ($string === '' || $string === "\r") { |
194: | return $string; |
195: | } |
196: | |
197: | |
198: | |
199: | $replace = []; |
200: | $append = ''; |
201: | $removeEscapes = $this->removesEscapes(); |
202: | $wrapAfterFormatting = $this->wrapsAfterFormatting(); |
203: | $wrapFormats = $wrapAfterFormatting |
204: | ? $this->TagFormats |
205: | : self::getNullTagFormats(); |
206: | $formats = $unformat |
207: | ? self::getLoopbackTagFormats() |
208: | : $this->TagFormats; |
209: | |
210: | |
211: | if ($string[-1] === "\r") { |
212: | $append .= "\r"; |
213: | $string = substr($string, 0, -1); |
214: | } |
215: | |
216: | |
217: | |
218: | if (!Regex::matchAll( |
219: | self::FORMAT_REGEX, |
220: | Str::setEol($string), |
221: | $matches, |
222: | \PREG_SET_ORDER | \PREG_UNMATCHED_AS_NULL, |
223: | )) { |
224: | throw new ShouldNotHappenException(sprintf( |
225: | 'Unable to parse: %s', |
226: | $string, |
227: | )); |
228: | } |
229: | |
230: | $string = ''; |
231: | foreach ($matches as $match) { |
232: | $indent = (string) $match['indent']; |
233: | |
234: | if ($match['breaks'] !== null) { |
235: | $breaks = $match['breaks']; |
236: | if ($unwrap && strpos($breaks, "\n") !== false) { |
237: | |
238: | $breaks = substr(Str::unwrap(".$breaks.", "\n", false, true, true), 1, -1); |
239: | } |
240: | $string .= $indent . $breaks; |
241: | continue; |
242: | } |
243: | |
244: | |
245: | if ($match['extra'] !== null) { |
246: | $string .= $indent . $match['extra']; |
247: | continue; |
248: | } |
249: | |
250: | $baseOffset = strlen($string . $indent); |
251: | |
252: | if ($match['text'] !== null) { |
253: | $text = $match['text']; |
254: | if ($unwrap && strpos($text, "\n") !== false) { |
255: | |
256: | $text = substr(Str::unwrap(".$text.", "\n", false, true, true), 1, -1); |
257: | } |
258: | |
259: | $adjust = 0; |
260: | $text = Regex::replaceCallback( |
261: | self::TAG_REGEX, |
262: | function ($matches) use ( |
263: | &$replace, |
264: | $wrapFormats, |
265: | $formats, |
266: | $baseOffset, |
267: | &$adjust |
268: | ): string { |
269: | $text = $this->applyTags( |
270: | $matches, |
271: | true, |
272: | $wrapFormats->removesEscapes(), |
273: | $wrapFormats, |
274: | ); |
275: | $placeholder = Regex::replace('/[^ ]/u', 'x', $text); |
276: | $formatted = $wrapFormats === $formats |
277: | ? $text |
278: | : $this->applyTags( |
279: | $matches, |
280: | true, |
281: | $formats->removesEscapes(), |
282: | $formats, |
283: | ); |
284: | $replace[] = [ |
285: | $baseOffset + $matches[0][1] + $adjust, |
286: | strlen($placeholder), |
287: | $formatted, |
288: | ]; |
289: | $adjust += strlen($placeholder) - strlen($matches[0][0]); |
290: | return $placeholder; |
291: | }, |
292: | $text, |
293: | -1, |
294: | $count, |
295: | \PREG_OFFSET_CAPTURE, |
296: | ); |
297: | |
298: | $string .= $indent . $text; |
299: | continue; |
300: | } |
301: | |
302: | if ($match['block'] !== null) { |
303: | |
304: | if ($unwrap && $string !== '' && $string[-1] !== "\n") { |
305: | $string[-1] = "\n"; |
306: | } |
307: | |
308: | |
309: | $formatted = $formats->apply( |
310: | $match['block'], |
311: | new TagAttributes( |
312: | self::TAG_CODE_BLOCK, |
313: | $match['fence'], |
314: | 0, |
315: | false, |
316: | $indent, |
317: | Str::coalesce(trim($match['infostring']), null), |
318: | ) |
319: | ); |
320: | $placeholder = '?'; |
321: | $replace[] = [ |
322: | $baseOffset, |
323: | 1, |
324: | $formatted, |
325: | ]; |
326: | |
327: | $string .= $indent . $placeholder; |
328: | continue; |
329: | } |
330: | |
331: | if ($match['span'] !== null) { |
332: | |
333: | $span = $match['span']; |
334: | |
335: | |
336: | |
337: | |
338: | $span = Regex::replace( |
339: | '/^ ((?> *[^ ]+).*) $/u', |
340: | '$1', |
341: | strtr($span, "\n", ' '), |
342: | ); |
343: | $attributes = new TagAttributes( |
344: | self::TAG_CODE_SPAN, |
345: | $match['backtickstring'], |
346: | ); |
347: | $text = $wrapFormats->apply($span, $attributes); |
348: | $placeholder = Regex::replace('/[^ ]/u', 'x', $text); |
349: | $formatted = $wrapFormats === $formats |
350: | ? $text |
351: | : $formats->apply($span, $attributes); |
352: | $replace[] = [ |
353: | $baseOffset, |
354: | strlen($placeholder), |
355: | $formatted, |
356: | ]; |
357: | |
358: | $string .= $indent . $placeholder; |
359: | continue; |
360: | } |
361: | } |
362: | |
363: | |
364: | |
365: | $replacements = count($replace); |
366: | $adjustable = []; |
367: | foreach ($replace as $i => [$offset]) { |
368: | $adjustable[$i] = $offset; |
369: | } |
370: | $adjust = 0; |
371: | $string = Regex::replaceCallback( |
372: | self::ESCAPE_REGEX, |
373: | function ($matches) use ( |
374: | $unformat, |
375: | $removeEscapes, |
376: | $wrapAfterFormatting, |
377: | &$replace, |
378: | &$adjustable, |
379: | &$adjust |
380: | ): string { |
381: | |
382: | |
383: | if ($wrapAfterFormatting && !$removeEscapes) { |
384: | if ($matches[1][0] !== ' ') { |
385: | return $matches[0][0]; |
386: | } |
387: | $placeholder = '\x'; |
388: | $replace[] = [$matches[0][1] + $adjust, 2, $matches[0][0]]; |
389: | return $placeholder; |
390: | } |
391: | |
392: | $delta = strlen($matches[1][0]) - strlen($matches[0][0]); |
393: | foreach ($adjustable as $i => $offset) { |
394: | if ($offset < $matches[0][1]) { |
395: | continue; |
396: | } |
397: | $replace[$i][0] += $delta; |
398: | } |
399: | |
400: | $placeholder = null; |
401: | if ($matches[1][0] === ' ') { |
402: | $placeholder = 'x'; |
403: | } |
404: | |
405: | if ($unformat || !$removeEscapes || $placeholder !== null) { |
406: | |
407: | $replace[] = [ |
408: | $matches[0][1] + $adjust, |
409: | strlen($matches[1][0]), |
410: | $unformat || !$removeEscapes ? $matches[0][0] : $matches[1][0], |
411: | ]; |
412: | } |
413: | |
414: | $adjust += $delta; |
415: | |
416: | return $placeholder ?? $matches[1][0]; |
417: | }, |
418: | $string, |
419: | -1, |
420: | $count, |
421: | \PREG_OFFSET_CAPTURE |
422: | ); |
423: | |
424: | if (is_array($wrapTo)) { |
425: | for ($i = 0; $i < 2; $i++) { |
426: | if ($wrapTo[$i] <= 0) { |
427: | $width ??= ($this->WidthCallback)(); |
428: | if ($width === null) { |
429: | $wrapTo = null; |
430: | break; |
431: | } |
432: | $wrapTo[$i] = max(0, $wrapTo[$i] + $width); |
433: | } |
434: | } |
435: | } elseif (is_int($wrapTo) && $wrapTo <= 0) { |
436: | $width = ($this->WidthCallback)(); |
437: | $wrapTo = $width === null |
438: | ? null |
439: | : max(0, $wrapTo + $width); |
440: | } |
441: | if ($wrapTo !== null) { |
442: | if ($break === "\n") { |
443: | $string = Str::wrap($string, $wrapTo); |
444: | } else { |
445: | |
446: | $wrapped = Str::wrap($string, $wrapTo); |
447: | $length = strlen($wrapped); |
448: | for ($i = 0; $i < $length; $i++) { |
449: | if ($wrapped[$i] === "\n" && $string[$i] !== "\n") { |
450: | $replace[] = [$i, 1, $break]; |
451: | } |
452: | } |
453: | } |
454: | } |
455: | |
456: | |
457: | |
458: | if (count($replace) !== $replacements) { |
459: | usort($replace, fn($a, $b) => $b[0] <=> $a[0]); |
460: | } else { |
461: | $replace = array_reverse($replace); |
462: | } |
463: | |
464: | foreach ($replace as [$offset, $length, $replacement]) { |
465: | $string = substr_replace($string, $replacement, $offset, $length); |
466: | } |
467: | |
468: | return $string . $append; |
469: | } |
470: | |
471: | |
472: | |
473: | |
474: | private function applyTags( |
475: | array $matches, |
476: | bool $matchesHasOffset, |
477: | bool $unescape, |
478: | TagFormats $formats, |
479: | int $depth = 0 |
480: | ): string { |
481: | |
482: | $text = $matchesHasOffset ? $matches['text'][0] : $matches['text']; |
483: | |
484: | $tag = $matchesHasOffset ? $matches['tag'][0] : $matches['tag']; |
485: | |
486: | $text = Regex::replaceCallback( |
487: | self::TAG_REGEX, |
488: | fn($matches) => |
489: | $this->applyTags($matches, false, $unescape, $formats, $depth + 1), |
490: | $text, |
491: | -1, |
492: | $count, |
493: | ); |
494: | |
495: | if ($unescape) { |
496: | $text = Regex::replace(self::ESCAPE_REGEX, '$1', $text); |
497: | } |
498: | |
499: | $tagId = self::TAG_MAP[$tag] ?? null; |
500: | if ($tagId === null) { |
501: | throw new LogicException(sprintf('Invalid tag: %s', $tag)); |
502: | } |
503: | |
504: | return $formats->apply( |
505: | $text, |
506: | new TagAttributes($tagId, $tag, $depth, (bool) $count), |
507: | ); |
508: | } |
509: | |
510: | |
511: | |
512: | |
513: | public function formatDiff(string $diff): string |
514: | { |
515: | $formats = [ |
516: | '---' => $this->TagFormats->getFormat(self::TAG_DIFF_HEADER), |
517: | '+++' => $this->TagFormats->getFormat(self::TAG_DIFF_HEADER), |
518: | '@' => $this->TagFormats->getFormat(self::TAG_DIFF_RANGE), |
519: | '+' => $this->TagFormats->getFormat(self::TAG_DIFF_ADDITION), |
520: | '-' => $this->TagFormats->getFormat(self::TAG_DIFF_REMOVAL), |
521: | ]; |
522: | |
523: | $attributes = [ |
524: | '---' => new TagAttributes(self::TAG_DIFF_HEADER, '---'), |
525: | '+++' => new TagAttributes(self::TAG_DIFF_HEADER, '+++'), |
526: | '@' => new TagAttributes(self::TAG_DIFF_RANGE, '@'), |
527: | '+' => new TagAttributes(self::TAG_DIFF_ADDITION, '+'), |
528: | '-' => new TagAttributes(self::TAG_DIFF_REMOVAL, '-'), |
529: | ]; |
530: | |
531: | return Regex::replaceCallback( |
532: | '/^(-{3}|\+{3}|[-+@]).*/m', |
533: | fn($matches) => |
534: | $formats[$matches[1]]->apply($matches[0], $attributes[$matches[1]]), |
535: | $diff, |
536: | ); |
537: | } |
538: | |
539: | |
540: | |
541: | |
542: | public function formatMessage( |
543: | string $msg1, |
544: | ?string $msg2 = null, |
545: | int $level = Console::LEVEL_INFO, |
546: | int $type = Console::TYPE_STANDARD |
547: | ): string { |
548: | $attributes = new MessageAttributes($level, $type); |
549: | |
550: | if ($type === Console::TYPE_UNFORMATTED) { |
551: | $formats = self::getNullMessageFormats(); |
552: | $prefix = ''; |
553: | } else { |
554: | $formats = $this->MessageFormats; |
555: | $prefix = $this->getMessagePrefix($level, $type); |
556: | } |
557: | |
558: | return $formats |
559: | ->getFormat($level, $type) |
560: | ->apply($msg1, $msg2, $prefix, $attributes); |
561: | } |
562: | |
563: | |
564: | |
565: | |
566: | public static function getNullFormatter(): self |
567: | { |
568: | return self::$NullFormatter ??= new self(); |
569: | } |
570: | |
571: | private static function getNullTagFormats(): TagFormats |
572: | { |
573: | return self::$NullTagFormats ??= new TagFormats(); |
574: | } |
575: | |
576: | private static function getNullMessageFormats(): MessageFormats |
577: | { |
578: | return self::$NullMessageFormats ??= new MessageFormats(); |
579: | } |
580: | |
581: | private static function getLoopbackTagFormats(): TagFormats |
582: | { |
583: | return self::$LoopbackTagFormats ??= LoopbackFormat::getFormatter()->TagFormats; |
584: | } |
585: | } |
586: | |