create-berna-stencil 1.0.47 → 1.0.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (176) hide show
  1. package/bin/create.js +1 -0
  2. package/docs/Assistant CLI.md +36 -0
  3. package/docs/Components.md +63 -0
  4. package/docs/Creating pages.md +46 -0
  5. package/docs/Head and SEO.md +44 -0
  6. package/docs/Javascript.md +53 -0
  7. package/docs/Styling with SCSS.md +138 -0
  8. package/package.json +3 -15
  9. package/src/backend/_core/vendor/composer/autoload_static.php +10 -10
  10. package/src/backend/_core/vendor/composer/installed.json +6 -6
  11. package/src/backend/_core/vendor/composer/installed.php +2 -2
  12. package/src/backend/_core/vendor/graham-campbell/result-type/LICENSE +21 -21
  13. package/src/backend/_core/vendor/graham-campbell/result-type/composer.json +33 -33
  14. package/src/backend/_core/vendor/graham-campbell/result-type/src/Error.php +121 -121
  15. package/src/backend/_core/vendor/graham-campbell/result-type/src/Result.php +69 -69
  16. package/src/backend/_core/vendor/graham-campbell/result-type/src/Success.php +120 -120
  17. package/src/backend/_core/vendor/phpoption/phpoption/LICENSE +200 -200
  18. package/src/backend/_core/vendor/phpoption/phpoption/composer.json +50 -50
  19. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/LazyOption.php +175 -175
  20. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/None.php +136 -136
  21. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/Option.php +434 -434
  22. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/Some.php +169 -169
  23. package/src/backend/_core/vendor/symfony/polyfill-ctype/Ctype.php +232 -232
  24. package/src/backend/_core/vendor/symfony/polyfill-ctype/LICENSE +19 -19
  25. package/src/backend/_core/vendor/symfony/polyfill-ctype/README.md +12 -12
  26. package/src/backend/_core/vendor/symfony/polyfill-ctype/bootstrap.php +50 -50
  27. package/src/backend/_core/vendor/symfony/polyfill-ctype/bootstrap80.php +46 -46
  28. package/src/backend/_core/vendor/symfony/polyfill-ctype/composer.json +38 -38
  29. package/src/backend/_core/vendor/symfony/polyfill-mbstring/LICENSE +19 -19
  30. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Mbstring.php +1077 -1077
  31. package/src/backend/_core/vendor/symfony/polyfill-mbstring/README.md +13 -13
  32. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/caseFolding.php +119 -119
  33. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/lowerCase.php +1397 -1397
  34. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/titleCaseRegexp.php +5 -5
  35. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/upperCase.php +1489 -1489
  36. package/src/backend/_core/vendor/symfony/polyfill-mbstring/bootstrap.php +171 -171
  37. package/src/backend/_core/vendor/symfony/polyfill-mbstring/bootstrap80.php +167 -167
  38. package/src/backend/_core/vendor/symfony/polyfill-mbstring/composer.json +39 -39
  39. package/src/backend/_core/vendor/symfony/polyfill-php80/LICENSE +19 -19
  40. package/src/backend/_core/vendor/symfony/polyfill-php80/Php80.php +115 -115
  41. package/src/backend/_core/vendor/symfony/polyfill-php80/PhpToken.php +106 -106
  42. package/src/backend/_core/vendor/symfony/polyfill-php80/README.md +25 -25
  43. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/Attribute.php +31 -31
  44. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/PhpToken.php +16 -16
  45. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/Stringable.php +20 -20
  46. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/UnhandledMatchError.php +16 -16
  47. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/ValueError.php +16 -16
  48. package/src/backend/_core/vendor/symfony/polyfill-php80/bootstrap.php +42 -42
  49. package/src/backend/_core/vendor/symfony/polyfill-php80/composer.json +37 -37
  50. package/src/backend/_core/vendor/vlucas/phpdotenv/LICENSE +30 -30
  51. package/src/backend/_core/vendor/vlucas/phpdotenv/composer.json +60 -60
  52. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Dotenv.php +267 -267
  53. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/ExceptionInterface.php +12 -12
  54. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidEncodingException.php +12 -12
  55. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidFileException.php +12 -12
  56. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidPathException.php +12 -12
  57. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/ValidationException.php +12 -12
  58. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/Loader.php +48 -48
  59. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/LoaderInterface.php +20 -20
  60. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/Resolver.php +65 -65
  61. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Entry.php +59 -59
  62. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/EntryParser.php +299 -299
  63. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Lexer.php +58 -58
  64. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Lines.php +127 -127
  65. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Parser.php +53 -53
  66. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/ParserInterface.php +19 -19
  67. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Value.php +88 -88
  68. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/AdapterInterface.php +15 -15
  69. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ApacheAdapter.php +89 -89
  70. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ArrayAdapter.php +80 -80
  71. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/EnvConstAdapter.php +88 -88
  72. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/GuardedWriter.php +85 -85
  73. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ImmutableWriter.php +110 -110
  74. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/MultiReader.php +48 -48
  75. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/MultiWriter.php +64 -64
  76. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/PutenvAdapter.php +91 -91
  77. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ReaderInterface.php +17 -17
  78. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ReplacingWriter.php +104 -104
  79. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ServerConstAdapter.php +88 -88
  80. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/WriterInterface.php +27 -27
  81. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/AdapterRepository.php +107 -107
  82. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/RepositoryBuilder.php +272 -272
  83. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/RepositoryInterface.php +51 -51
  84. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/File/Paths.php +44 -44
  85. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/File/Reader.php +81 -81
  86. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/FileStore.php +72 -72
  87. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StoreBuilder.php +141 -141
  88. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StoreInterface.php +17 -17
  89. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StringStore.php +37 -37
  90. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Util/Regex.php +112 -112
  91. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Util/Str.php +108 -108
  92. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Validator.php +207 -207
  93. package/src/frontend/data/site.json +43 -43
  94. package/src/frontend/scss/modules/_root.scss +2 -4
  95. package/src/frontend/scss/modules/frameworks/_bootstrap.scss +0 -1
  96. package/src/frontend/scss/modules/frameworks/_bulma.scss +0 -1
  97. package/src/frontend/scss/modules/frameworks/_foundation.scss +0 -1
  98. package/src/frontend/scss/modules/frameworks/_uikit.scss +0 -1
  99. package/src/backend/_core/vendor/graham-campbell/result-type/.gitattributes +0 -9
  100. package/src/backend/_core/vendor/graham-campbell/result-type/.github/CODE_OF_CONDUCT.md +0 -132
  101. package/src/backend/_core/vendor/graham-campbell/result-type/.github/CONTRIBUTING.md +0 -31
  102. package/src/backend/_core/vendor/graham-campbell/result-type/.github/FUNDING.yml +0 -2
  103. package/src/backend/_core/vendor/graham-campbell/result-type/.github/SECURITY.md +0 -14
  104. package/src/backend/_core/vendor/graham-campbell/result-type/.github/workflows/stale.yml +0 -11
  105. package/src/backend/_core/vendor/graham-campbell/result-type/.github/workflows/tests.yml +0 -40
  106. package/src/backend/_core/vendor/graham-campbell/result-type/CHANGELOG.md +0 -53
  107. package/src/backend/_core/vendor/graham-campbell/result-type/README.md +0 -42
  108. package/src/backend/_core/vendor/graham-campbell/result-type/phpunit.xml.dist +0 -13
  109. package/src/backend/_core/vendor/graham-campbell/result-type/tests/ResultTest.php +0 -95
  110. package/src/backend/_core/vendor/phpoption/phpoption/.gitattributes +0 -13
  111. package/src/backend/_core/vendor/phpoption/phpoption/.github/CODE_OF_CONDUCT.md +0 -132
  112. package/src/backend/_core/vendor/phpoption/phpoption/.github/CONTRIBUTING.md +0 -30
  113. package/src/backend/_core/vendor/phpoption/phpoption/.github/FUNDING.yml +0 -2
  114. package/src/backend/_core/vendor/phpoption/phpoption/.github/SECURITY.md +0 -14
  115. package/src/backend/_core/vendor/phpoption/phpoption/.github/workflows/static.yml +0 -40
  116. package/src/backend/_core/vendor/phpoption/phpoption/.github/workflows/tests.yml +0 -40
  117. package/src/backend/_core/vendor/phpoption/phpoption/Makefile +0 -17
  118. package/src/backend/_core/vendor/phpoption/phpoption/README.md +0 -201
  119. package/src/backend/_core/vendor/phpoption/phpoption/phpstan-baseline.neon +0 -44
  120. package/src/backend/_core/vendor/phpoption/phpoption/phpstan.neon.dist +0 -7
  121. package/src/backend/_core/vendor/phpoption/phpoption/phpunit.xml.dist +0 -13
  122. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/EnsureTest.php +0 -72
  123. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/LazyOptionTest.php +0 -357
  124. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/NoneTest.php +0 -153
  125. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/OptionTest.php +0 -166
  126. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/SomeTest.php +0 -194
  127. package/src/backend/_core/vendor/phpoption/phpoption/tests/bootstrap.php +0 -8
  128. package/src/backend/_core/vendor/phpoption/phpoption/vendor-bin/phpstan/composer.json +0 -8
  129. package/src/backend/_core/vendor/vlucas/phpdotenv/.editorconfig +0 -15
  130. package/src/backend/_core/vendor/vlucas/phpdotenv/.gitattributes +0 -15
  131. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/CODE_OF_CONDUCT.md +0 -132
  132. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/CONTRIBUTING.md +0 -30
  133. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/FUNDING.yml +0 -2
  134. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/SECURITY.md +0 -14
  135. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/workflows/static.yml +0 -40
  136. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/workflows/tests.yml +0 -70
  137. package/src/backend/_core/vendor/vlucas/phpdotenv/Makefile +0 -17
  138. package/src/backend/_core/vendor/vlucas/phpdotenv/README.md +0 -370
  139. package/src/backend/_core/vendor/vlucas/phpdotenv/UPGRADING.md +0 -196
  140. package/src/backend/_core/vendor/vlucas/phpdotenv/phpstan-baseline.neon +0 -157
  141. package/src/backend/_core/vendor/vlucas/phpdotenv/phpstan.neon.dist +0 -7
  142. package/src/backend/_core/vendor/vlucas/phpdotenv/phpunit.xml.dist +0 -13
  143. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/DotenvTest.php +0 -387
  144. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Loader/LoaderTest.php +0 -86
  145. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/EntryParserTest.php +0 -234
  146. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/LexerTest.php +0 -40
  147. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/LinesTest.php +0 -53
  148. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/ParserTest.php +0 -98
  149. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/ArrayAdapterTest.php +0 -57
  150. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/EnvConstAdapterTest.php +0 -75
  151. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/PutenvAdapterTest.php +0 -52
  152. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/ServerConstAdapterTest.php +0 -75
  153. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/RepositoryTest.php +0 -305
  154. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Store/StoreTest.php +0 -141
  155. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/ValidatorTest.php +0 -479
  156. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/.env +0 -5
  157. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/assertions.env +0 -18
  158. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/booleans.env +0 -33
  159. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/commented.env +0 -15
  160. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/empty.env +0 -1
  161. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/example.env +0 -1
  162. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/exported.env +0 -7
  163. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/immutable.env +0 -1
  164. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/integers.env +0 -17
  165. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/large.env +0 -2
  166. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multibyte.env +0 -3
  167. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multiline.env +0 -14
  168. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multiple.env +0 -4
  169. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/mutable.env +0 -1
  170. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/nested.env +0 -15
  171. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/quoted.env +0 -11
  172. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/specialchars.env +0 -8
  173. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/unicodevarnames.env +0 -2
  174. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/utf8-with-bom-encoding.env +0 -3
  175. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/windows.env +0 -1
  176. package/src/backend/_core/vendor/vlucas/phpdotenv/vendor-bin/phpstan/composer.json +0 -15
@@ -1,299 +1,299 @@
1
- <?php
2
-
3
- declare(strict_types=1);
4
-
5
- namespace Dotenv\Parser;
6
-
7
- use Dotenv\Util\Regex;
8
- use Dotenv\Util\Str;
9
- use GrahamCampbell\ResultType\Error;
10
- use GrahamCampbell\ResultType\Result;
11
- use GrahamCampbell\ResultType\Success;
12
-
13
- final class EntryParser
14
- {
15
- private const INITIAL_STATE = 0;
16
- private const UNQUOTED_STATE = 1;
17
- private const SINGLE_QUOTED_STATE = 2;
18
- private const DOUBLE_QUOTED_STATE = 3;
19
- private const ESCAPE_SEQUENCE_STATE = 4;
20
- private const WHITESPACE_STATE = 5;
21
- private const COMMENT_STATE = 6;
22
- private const REJECT_STATES = [self::SINGLE_QUOTED_STATE, self::DOUBLE_QUOTED_STATE, self::ESCAPE_SEQUENCE_STATE];
23
-
24
- /**
25
- * This class is a singleton.
26
- *
27
- * @codeCoverageIgnore
28
- *
29
- * @return void
30
- */
31
- private function __construct()
32
- {
33
- //
34
- }
35
-
36
- /**
37
- * Parse a raw entry into a proper entry.
38
- *
39
- * That is, turn a raw environment variable entry into a name and possibly
40
- * a value. We wrap the answer in a result type.
41
- *
42
- * @param string $entry
43
- *
44
- * @return \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Entry, string>
45
- */
46
- public static function parse(string $entry)
47
- {
48
- return self::splitStringIntoParts($entry)->flatMap(static function (array $parts) {
49
- [$name, $value] = $parts;
50
-
51
- return self::parseName($name)->flatMap(static function (string $name) use ($value) {
52
- /** @var Result<Value|null, string> */
53
- $parsedValue = $value === null ? Success::create(null) : self::parseValue($value);
54
-
55
- return $parsedValue->map(static function (?Value $value) use ($name) {
56
- return new Entry($name, $value);
57
- });
58
- });
59
- });
60
- }
61
-
62
- /**
63
- * Split the compound string into parts.
64
- *
65
- * @param string $line
66
- *
67
- * @return \GrahamCampbell\ResultType\Result<array{string, string|null},string>
68
- */
69
- private static function splitStringIntoParts(string $line)
70
- {
71
- /** @var array{string, string|null} */
72
- $result = Str::pos($line, '=')->map(static function () use ($line) {
73
- return \array_map('trim', \explode('=', $line, 2));
74
- })->getOrElse([$line, null]);
75
-
76
- if ($result[0] === '') {
77
- /** @var \GrahamCampbell\ResultType\Result<array{string, string|null},string> */
78
- return Error::create(self::getErrorMessage('an unexpected equals', $line));
79
- }
80
-
81
- /** @var \GrahamCampbell\ResultType\Result<array{string, string|null},string> */
82
- return Success::create($result);
83
- }
84
-
85
- /**
86
- * Parse the given variable name.
87
- *
88
- * That is, strip the optional quotes and leading "export" from the
89
- * variable name. We wrap the answer in a result type.
90
- *
91
- * @param string $name
92
- *
93
- * @return \GrahamCampbell\ResultType\Result<string, string>
94
- */
95
- private static function parseName(string $name)
96
- {
97
- if (Str::len($name) > 8 && Str::substr($name, 0, 6) === 'export' && \ctype_space(Str::substr($name, 6, 1))) {
98
- $name = \ltrim(Str::substr($name, 6));
99
- }
100
-
101
- if (self::isQuotedName($name)) {
102
- $name = Str::substr($name, 1, -1);
103
- }
104
-
105
- if (!self::isValidName($name)) {
106
- /** @var \GrahamCampbell\ResultType\Result<string, string> */
107
- return Error::create(self::getErrorMessage('an invalid name', $name));
108
- }
109
-
110
- /** @var \GrahamCampbell\ResultType\Result<string, string> */
111
- return Success::create($name);
112
- }
113
-
114
- /**
115
- * Is the given variable name quoted?
116
- *
117
- * @param string $name
118
- *
119
- * @return bool
120
- */
121
- private static function isQuotedName(string $name)
122
- {
123
- if (Str::len($name) < 3) {
124
- return false;
125
- }
126
-
127
- $first = Str::substr($name, 0, 1);
128
- $last = Str::substr($name, -1, 1);
129
-
130
- return ($first === '"' && $last === '"') || ($first === '\'' && $last === '\'');
131
- }
132
-
133
- /**
134
- * Is the given variable name valid?
135
- *
136
- * @param string $name
137
- *
138
- * @return bool
139
- */
140
- private static function isValidName(string $name)
141
- {
142
- return Regex::matches('~(*UTF8)\A[\p{Ll}\p{Lu}\p{M}\p{N}_.]+\z~', $name)->success()->getOrElse(false);
143
- }
144
-
145
- /**
146
- * Parse the given variable value.
147
- *
148
- * This has the effect of stripping quotes and comments, dealing with
149
- * special characters, and locating nested variables, but not resolving
150
- * them. Formally, we run a finite state automaton with an output tape: a
151
- * transducer. We wrap the answer in a result type.
152
- *
153
- * @param string $value
154
- *
155
- * @return \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string>
156
- */
157
- private static function parseValue(string $value)
158
- {
159
- if (\trim($value) === '') {
160
- /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
161
- return Success::create(Value::blank());
162
- }
163
-
164
- return \array_reduce(\iterator_to_array(Lexer::lex($value)), static function (Result $data, string $token) {
165
- return $data->flatMap(static function (array $data) use ($token) {
166
- return self::processToken($data[1], $token)->map(static function (array $val) use ($data) {
167
- return [$data[0]->append($val[0], $val[1]), $val[2]];
168
- });
169
- });
170
- }, Success::create([Value::blank(), self::INITIAL_STATE]))->flatMap(static function (array $result) {
171
- if (in_array($result[1], self::REJECT_STATES, true)) {
172
- /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
173
- return Error::create('a missing closing quote');
174
- }
175
-
176
- /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
177
- return Success::create($result[0]);
178
- })->mapError(static function (string $err) use ($value) {
179
- return self::getErrorMessage($err, $value);
180
- });
181
- }
182
-
183
- /**
184
- * Process the given token.
185
- *
186
- * @param int $state
187
- * @param string $token
188
- *
189
- * @return \GrahamCampbell\ResultType\Result<array{string, bool, int}, string>
190
- */
191
- private static function processToken(int $state, string $token)
192
- {
193
- switch ($state) {
194
- case self::INITIAL_STATE:
195
- if ($token === '\'') {
196
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
197
- return Success::create(['', false, self::SINGLE_QUOTED_STATE]);
198
- } elseif ($token === '"') {
199
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
200
- return Success::create(['', false, self::DOUBLE_QUOTED_STATE]);
201
- } elseif ($token === '#') {
202
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
203
- return Success::create(['', false, self::COMMENT_STATE]);
204
- } elseif ($token === '$') {
205
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
206
- return Success::create([$token, true, self::UNQUOTED_STATE]);
207
- } else {
208
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
209
- return Success::create([$token, false, self::UNQUOTED_STATE]);
210
- }
211
- case self::UNQUOTED_STATE:
212
- if ($token === '#') {
213
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
214
- return Success::create(['', false, self::COMMENT_STATE]);
215
- } elseif (\ctype_space($token)) {
216
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
217
- return Success::create(['', false, self::WHITESPACE_STATE]);
218
- } elseif ($token === '$') {
219
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
220
- return Success::create([$token, true, self::UNQUOTED_STATE]);
221
- } else {
222
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
223
- return Success::create([$token, false, self::UNQUOTED_STATE]);
224
- }
225
- case self::SINGLE_QUOTED_STATE:
226
- if ($token === '\'') {
227
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
228
- return Success::create(['', false, self::WHITESPACE_STATE]);
229
- } else {
230
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
231
- return Success::create([$token, false, self::SINGLE_QUOTED_STATE]);
232
- }
233
- case self::DOUBLE_QUOTED_STATE:
234
- if ($token === '"') {
235
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
236
- return Success::create(['', false, self::WHITESPACE_STATE]);
237
- } elseif ($token === '\\') {
238
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
239
- return Success::create(['', false, self::ESCAPE_SEQUENCE_STATE]);
240
- } elseif ($token === '$') {
241
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
242
- return Success::create([$token, true, self::DOUBLE_QUOTED_STATE]);
243
- } else {
244
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
245
- return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
246
- }
247
- case self::ESCAPE_SEQUENCE_STATE:
248
- if ($token === '"' || $token === '\\') {
249
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
250
- return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
251
- } elseif ($token === '$') {
252
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
253
- return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
254
- } else {
255
- $first = Str::substr($token, 0, 1);
256
- if (\in_array($first, ['f', 'n', 'r', 't', 'v'], true)) {
257
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
258
- return Success::create([\stripcslashes('\\'.$first).Str::substr($token, 1), false, self::DOUBLE_QUOTED_STATE]);
259
- } else {
260
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
261
- return Error::create('an unexpected escape sequence');
262
- }
263
- }
264
- case self::WHITESPACE_STATE:
265
- if ($token === '#') {
266
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
267
- return Success::create(['', false, self::COMMENT_STATE]);
268
- } elseif (!\ctype_space($token)) {
269
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
270
- return Error::create('unexpected whitespace');
271
- } else {
272
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
273
- return Success::create(['', false, self::WHITESPACE_STATE]);
274
- }
275
- case self::COMMENT_STATE:
276
- /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
277
- return Success::create(['', false, self::COMMENT_STATE]);
278
- default:
279
- throw new \Error('Parser entered invalid state.');
280
- }
281
- }
282
-
283
- /**
284
- * Generate a friendly error message.
285
- *
286
- * @param string $cause
287
- * @param string $subject
288
- *
289
- * @return string
290
- */
291
- private static function getErrorMessage(string $cause, string $subject)
292
- {
293
- return \sprintf(
294
- 'Encountered %s at [%s].',
295
- $cause,
296
- \strtok($subject, "\n")
297
- );
298
- }
299
- }
1
+ <?php
2
+
3
+ declare(strict_types=1);
4
+
5
+ namespace Dotenv\Parser;
6
+
7
+ use Dotenv\Util\Regex;
8
+ use Dotenv\Util\Str;
9
+ use GrahamCampbell\ResultType\Error;
10
+ use GrahamCampbell\ResultType\Result;
11
+ use GrahamCampbell\ResultType\Success;
12
+
13
+ final class EntryParser
14
+ {
15
+ private const INITIAL_STATE = 0;
16
+ private const UNQUOTED_STATE = 1;
17
+ private const SINGLE_QUOTED_STATE = 2;
18
+ private const DOUBLE_QUOTED_STATE = 3;
19
+ private const ESCAPE_SEQUENCE_STATE = 4;
20
+ private const WHITESPACE_STATE = 5;
21
+ private const COMMENT_STATE = 6;
22
+ private const REJECT_STATES = [self::SINGLE_QUOTED_STATE, self::DOUBLE_QUOTED_STATE, self::ESCAPE_SEQUENCE_STATE];
23
+
24
+ /**
25
+ * This class is a singleton.
26
+ *
27
+ * @codeCoverageIgnore
28
+ *
29
+ * @return void
30
+ */
31
+ private function __construct()
32
+ {
33
+ //
34
+ }
35
+
36
+ /**
37
+ * Parse a raw entry into a proper entry.
38
+ *
39
+ * That is, turn a raw environment variable entry into a name and possibly
40
+ * a value. We wrap the answer in a result type.
41
+ *
42
+ * @param string $entry
43
+ *
44
+ * @return \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Entry, string>
45
+ */
46
+ public static function parse(string $entry)
47
+ {
48
+ return self::splitStringIntoParts($entry)->flatMap(static function (array $parts) {
49
+ [$name, $value] = $parts;
50
+
51
+ return self::parseName($name)->flatMap(static function (string $name) use ($value) {
52
+ /** @var Result<Value|null, string> */
53
+ $parsedValue = $value === null ? Success::create(null) : self::parseValue($value);
54
+
55
+ return $parsedValue->map(static function (?Value $value) use ($name) {
56
+ return new Entry($name, $value);
57
+ });
58
+ });
59
+ });
60
+ }
61
+
62
+ /**
63
+ * Split the compound string into parts.
64
+ *
65
+ * @param string $line
66
+ *
67
+ * @return \GrahamCampbell\ResultType\Result<array{string, string|null},string>
68
+ */
69
+ private static function splitStringIntoParts(string $line)
70
+ {
71
+ /** @var array{string, string|null} */
72
+ $result = Str::pos($line, '=')->map(static function () use ($line) {
73
+ return \array_map('trim', \explode('=', $line, 2));
74
+ })->getOrElse([$line, null]);
75
+
76
+ if ($result[0] === '') {
77
+ /** @var \GrahamCampbell\ResultType\Result<array{string, string|null},string> */
78
+ return Error::create(self::getErrorMessage('an unexpected equals', $line));
79
+ }
80
+
81
+ /** @var \GrahamCampbell\ResultType\Result<array{string, string|null},string> */
82
+ return Success::create($result);
83
+ }
84
+
85
+ /**
86
+ * Parse the given variable name.
87
+ *
88
+ * That is, strip the optional quotes and leading "export" from the
89
+ * variable name. We wrap the answer in a result type.
90
+ *
91
+ * @param string $name
92
+ *
93
+ * @return \GrahamCampbell\ResultType\Result<string, string>
94
+ */
95
+ private static function parseName(string $name)
96
+ {
97
+ if (Str::len($name) > 8 && Str::substr($name, 0, 6) === 'export' && \ctype_space(Str::substr($name, 6, 1))) {
98
+ $name = \ltrim(Str::substr($name, 6));
99
+ }
100
+
101
+ if (self::isQuotedName($name)) {
102
+ $name = Str::substr($name, 1, -1);
103
+ }
104
+
105
+ if (!self::isValidName($name)) {
106
+ /** @var \GrahamCampbell\ResultType\Result<string, string> */
107
+ return Error::create(self::getErrorMessage('an invalid name', $name));
108
+ }
109
+
110
+ /** @var \GrahamCampbell\ResultType\Result<string, string> */
111
+ return Success::create($name);
112
+ }
113
+
114
+ /**
115
+ * Is the given variable name quoted?
116
+ *
117
+ * @param string $name
118
+ *
119
+ * @return bool
120
+ */
121
+ private static function isQuotedName(string $name)
122
+ {
123
+ if (Str::len($name) < 3) {
124
+ return false;
125
+ }
126
+
127
+ $first = Str::substr($name, 0, 1);
128
+ $last = Str::substr($name, -1, 1);
129
+
130
+ return ($first === '"' && $last === '"') || ($first === '\'' && $last === '\'');
131
+ }
132
+
133
+ /**
134
+ * Is the given variable name valid?
135
+ *
136
+ * @param string $name
137
+ *
138
+ * @return bool
139
+ */
140
+ private static function isValidName(string $name)
141
+ {
142
+ return Regex::matches('~(*UTF8)\A[\p{Ll}\p{Lu}\p{M}\p{N}_.]+\z~', $name)->success()->getOrElse(false);
143
+ }
144
+
145
+ /**
146
+ * Parse the given variable value.
147
+ *
148
+ * This has the effect of stripping quotes and comments, dealing with
149
+ * special characters, and locating nested variables, but not resolving
150
+ * them. Formally, we run a finite state automaton with an output tape: a
151
+ * transducer. We wrap the answer in a result type.
152
+ *
153
+ * @param string $value
154
+ *
155
+ * @return \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string>
156
+ */
157
+ private static function parseValue(string $value)
158
+ {
159
+ if (\trim($value) === '') {
160
+ /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
161
+ return Success::create(Value::blank());
162
+ }
163
+
164
+ return \array_reduce(\iterator_to_array(Lexer::lex($value)), static function (Result $data, string $token) {
165
+ return $data->flatMap(static function (array $data) use ($token) {
166
+ return self::processToken($data[1], $token)->map(static function (array $val) use ($data) {
167
+ return [$data[0]->append($val[0], $val[1]), $val[2]];
168
+ });
169
+ });
170
+ }, Success::create([Value::blank(), self::INITIAL_STATE]))->flatMap(static function (array $result) {
171
+ if (in_array($result[1], self::REJECT_STATES, true)) {
172
+ /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
173
+ return Error::create('a missing closing quote');
174
+ }
175
+
176
+ /** @var \GrahamCampbell\ResultType\Result<\Dotenv\Parser\Value, string> */
177
+ return Success::create($result[0]);
178
+ })->mapError(static function (string $err) use ($value) {
179
+ return self::getErrorMessage($err, $value);
180
+ });
181
+ }
182
+
183
+ /**
184
+ * Process the given token.
185
+ *
186
+ * @param int $state
187
+ * @param string $token
188
+ *
189
+ * @return \GrahamCampbell\ResultType\Result<array{string, bool, int}, string>
190
+ */
191
+ private static function processToken(int $state, string $token)
192
+ {
193
+ switch ($state) {
194
+ case self::INITIAL_STATE:
195
+ if ($token === '\'') {
196
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
197
+ return Success::create(['', false, self::SINGLE_QUOTED_STATE]);
198
+ } elseif ($token === '"') {
199
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
200
+ return Success::create(['', false, self::DOUBLE_QUOTED_STATE]);
201
+ } elseif ($token === '#') {
202
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
203
+ return Success::create(['', false, self::COMMENT_STATE]);
204
+ } elseif ($token === '$') {
205
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
206
+ return Success::create([$token, true, self::UNQUOTED_STATE]);
207
+ } else {
208
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
209
+ return Success::create([$token, false, self::UNQUOTED_STATE]);
210
+ }
211
+ case self::UNQUOTED_STATE:
212
+ if ($token === '#') {
213
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
214
+ return Success::create(['', false, self::COMMENT_STATE]);
215
+ } elseif (\ctype_space($token)) {
216
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
217
+ return Success::create(['', false, self::WHITESPACE_STATE]);
218
+ } elseif ($token === '$') {
219
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
220
+ return Success::create([$token, true, self::UNQUOTED_STATE]);
221
+ } else {
222
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
223
+ return Success::create([$token, false, self::UNQUOTED_STATE]);
224
+ }
225
+ case self::SINGLE_QUOTED_STATE:
226
+ if ($token === '\'') {
227
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
228
+ return Success::create(['', false, self::WHITESPACE_STATE]);
229
+ } else {
230
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
231
+ return Success::create([$token, false, self::SINGLE_QUOTED_STATE]);
232
+ }
233
+ case self::DOUBLE_QUOTED_STATE:
234
+ if ($token === '"') {
235
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
236
+ return Success::create(['', false, self::WHITESPACE_STATE]);
237
+ } elseif ($token === '\\') {
238
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
239
+ return Success::create(['', false, self::ESCAPE_SEQUENCE_STATE]);
240
+ } elseif ($token === '$') {
241
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
242
+ return Success::create([$token, true, self::DOUBLE_QUOTED_STATE]);
243
+ } else {
244
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
245
+ return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
246
+ }
247
+ case self::ESCAPE_SEQUENCE_STATE:
248
+ if ($token === '"' || $token === '\\') {
249
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
250
+ return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
251
+ } elseif ($token === '$') {
252
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
253
+ return Success::create([$token, false, self::DOUBLE_QUOTED_STATE]);
254
+ } else {
255
+ $first = Str::substr($token, 0, 1);
256
+ if (\in_array($first, ['f', 'n', 'r', 't', 'v'], true)) {
257
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
258
+ return Success::create([\stripcslashes('\\'.$first).Str::substr($token, 1), false, self::DOUBLE_QUOTED_STATE]);
259
+ } else {
260
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
261
+ return Error::create('an unexpected escape sequence');
262
+ }
263
+ }
264
+ case self::WHITESPACE_STATE:
265
+ if ($token === '#') {
266
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
267
+ return Success::create(['', false, self::COMMENT_STATE]);
268
+ } elseif (!\ctype_space($token)) {
269
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
270
+ return Error::create('unexpected whitespace');
271
+ } else {
272
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
273
+ return Success::create(['', false, self::WHITESPACE_STATE]);
274
+ }
275
+ case self::COMMENT_STATE:
276
+ /** @var \GrahamCampbell\ResultType\Result<array{string, bool, int}, string> */
277
+ return Success::create(['', false, self::COMMENT_STATE]);
278
+ default:
279
+ throw new \Error('Parser entered invalid state.');
280
+ }
281
+ }
282
+
283
+ /**
284
+ * Generate a friendly error message.
285
+ *
286
+ * @param string $cause
287
+ * @param string $subject
288
+ *
289
+ * @return string
290
+ */
291
+ private static function getErrorMessage(string $cause, string $subject)
292
+ {
293
+ return \sprintf(
294
+ 'Encountered %s at [%s].',
295
+ $cause,
296
+ \strtok($subject, "\n")
297
+ );
298
+ }
299
+ }