create-berna-stencil 1.0.49 → 1.0.51

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 (177) hide show
  1. package/_tools/modules/updateIncludes.js +1 -1
  2. package/bin/create.js +2 -2
  3. package/docs/Components.md +54 -33
  4. package/docs/Head and SEO.md +33 -6
  5. package/docs/Styling with SCSS.md +8 -6
  6. package/package.json +58 -62
  7. package/src/backend/_core/vendor/composer/autoload_static.php +10 -10
  8. package/src/backend/_core/vendor/composer/installed.json +6 -6
  9. package/src/backend/_core/vendor/composer/installed.php +2 -2
  10. package/src/backend/_core/vendor/graham-campbell/result-type/.gitattributes +9 -0
  11. package/src/backend/_core/vendor/graham-campbell/result-type/.github/CODE_OF_CONDUCT.md +132 -0
  12. package/src/backend/_core/vendor/graham-campbell/result-type/.github/CONTRIBUTING.md +31 -0
  13. package/src/backend/_core/vendor/graham-campbell/result-type/.github/FUNDING.yml +2 -0
  14. package/src/backend/_core/vendor/graham-campbell/result-type/.github/SECURITY.md +14 -0
  15. package/src/backend/_core/vendor/graham-campbell/result-type/.github/workflows/stale.yml +11 -0
  16. package/src/backend/_core/vendor/graham-campbell/result-type/.github/workflows/tests.yml +40 -0
  17. package/src/backend/_core/vendor/graham-campbell/result-type/CHANGELOG.md +53 -0
  18. package/src/backend/_core/vendor/graham-campbell/result-type/LICENSE +21 -21
  19. package/src/backend/_core/vendor/graham-campbell/result-type/README.md +42 -0
  20. package/src/backend/_core/vendor/graham-campbell/result-type/composer.json +33 -33
  21. package/src/backend/_core/vendor/graham-campbell/result-type/phpunit.xml.dist +13 -0
  22. package/src/backend/_core/vendor/graham-campbell/result-type/src/Error.php +121 -121
  23. package/src/backend/_core/vendor/graham-campbell/result-type/src/Result.php +69 -69
  24. package/src/backend/_core/vendor/graham-campbell/result-type/src/Success.php +120 -120
  25. package/src/backend/_core/vendor/graham-campbell/result-type/tests/ResultTest.php +95 -0
  26. package/src/backend/_core/vendor/phpoption/phpoption/.gitattributes +13 -0
  27. package/src/backend/_core/vendor/phpoption/phpoption/.github/CODE_OF_CONDUCT.md +132 -0
  28. package/src/backend/_core/vendor/phpoption/phpoption/.github/CONTRIBUTING.md +30 -0
  29. package/src/backend/_core/vendor/phpoption/phpoption/.github/FUNDING.yml +2 -0
  30. package/src/backend/_core/vendor/phpoption/phpoption/.github/SECURITY.md +14 -0
  31. package/src/backend/_core/vendor/phpoption/phpoption/.github/workflows/static.yml +40 -0
  32. package/src/backend/_core/vendor/phpoption/phpoption/.github/workflows/tests.yml +40 -0
  33. package/src/backend/_core/vendor/phpoption/phpoption/LICENSE +200 -200
  34. package/src/backend/_core/vendor/phpoption/phpoption/Makefile +17 -0
  35. package/src/backend/_core/vendor/phpoption/phpoption/README.md +201 -0
  36. package/src/backend/_core/vendor/phpoption/phpoption/composer.json +50 -50
  37. package/src/backend/_core/vendor/phpoption/phpoption/phpstan-baseline.neon +44 -0
  38. package/src/backend/_core/vendor/phpoption/phpoption/phpstan.neon.dist +7 -0
  39. package/src/backend/_core/vendor/phpoption/phpoption/phpunit.xml.dist +13 -0
  40. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/LazyOption.php +175 -175
  41. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/None.php +136 -136
  42. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/Option.php +434 -434
  43. package/src/backend/_core/vendor/phpoption/phpoption/src/PhpOption/Some.php +169 -169
  44. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/EnsureTest.php +72 -0
  45. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/LazyOptionTest.php +357 -0
  46. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/NoneTest.php +153 -0
  47. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/OptionTest.php +166 -0
  48. package/src/backend/_core/vendor/phpoption/phpoption/tests/PhpOption/Tests/SomeTest.php +194 -0
  49. package/src/backend/_core/vendor/phpoption/phpoption/tests/bootstrap.php +8 -0
  50. package/src/backend/_core/vendor/phpoption/phpoption/vendor-bin/phpstan/composer.json +8 -0
  51. package/src/backend/_core/vendor/symfony/polyfill-ctype/Ctype.php +232 -232
  52. package/src/backend/_core/vendor/symfony/polyfill-ctype/LICENSE +19 -19
  53. package/src/backend/_core/vendor/symfony/polyfill-ctype/README.md +12 -12
  54. package/src/backend/_core/vendor/symfony/polyfill-ctype/bootstrap.php +50 -50
  55. package/src/backend/_core/vendor/symfony/polyfill-ctype/bootstrap80.php +46 -46
  56. package/src/backend/_core/vendor/symfony/polyfill-ctype/composer.json +38 -38
  57. package/src/backend/_core/vendor/symfony/polyfill-mbstring/LICENSE +19 -19
  58. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Mbstring.php +1077 -1077
  59. package/src/backend/_core/vendor/symfony/polyfill-mbstring/README.md +13 -13
  60. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/caseFolding.php +119 -119
  61. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/lowerCase.php +1397 -1397
  62. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/titleCaseRegexp.php +5 -5
  63. package/src/backend/_core/vendor/symfony/polyfill-mbstring/Resources/unidata/upperCase.php +1489 -1489
  64. package/src/backend/_core/vendor/symfony/polyfill-mbstring/bootstrap.php +171 -171
  65. package/src/backend/_core/vendor/symfony/polyfill-mbstring/bootstrap80.php +167 -167
  66. package/src/backend/_core/vendor/symfony/polyfill-mbstring/composer.json +39 -39
  67. package/src/backend/_core/vendor/symfony/polyfill-php80/LICENSE +19 -19
  68. package/src/backend/_core/vendor/symfony/polyfill-php80/Php80.php +115 -115
  69. package/src/backend/_core/vendor/symfony/polyfill-php80/PhpToken.php +106 -106
  70. package/src/backend/_core/vendor/symfony/polyfill-php80/README.md +25 -25
  71. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/Attribute.php +31 -31
  72. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/PhpToken.php +16 -16
  73. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/Stringable.php +20 -20
  74. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/UnhandledMatchError.php +16 -16
  75. package/src/backend/_core/vendor/symfony/polyfill-php80/Resources/stubs/ValueError.php +16 -16
  76. package/src/backend/_core/vendor/symfony/polyfill-php80/bootstrap.php +42 -42
  77. package/src/backend/_core/vendor/symfony/polyfill-php80/composer.json +37 -37
  78. package/src/backend/_core/vendor/vlucas/phpdotenv/.editorconfig +15 -0
  79. package/src/backend/_core/vendor/vlucas/phpdotenv/.gitattributes +15 -0
  80. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/CODE_OF_CONDUCT.md +132 -0
  81. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/CONTRIBUTING.md +30 -0
  82. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/FUNDING.yml +2 -0
  83. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/SECURITY.md +14 -0
  84. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/workflows/static.yml +40 -0
  85. package/src/backend/_core/vendor/vlucas/phpdotenv/.github/workflows/tests.yml +70 -0
  86. package/src/backend/_core/vendor/vlucas/phpdotenv/LICENSE +30 -30
  87. package/src/backend/_core/vendor/vlucas/phpdotenv/Makefile +17 -0
  88. package/src/backend/_core/vendor/vlucas/phpdotenv/README.md +370 -0
  89. package/src/backend/_core/vendor/vlucas/phpdotenv/UPGRADING.md +196 -0
  90. package/src/backend/_core/vendor/vlucas/phpdotenv/composer.json +60 -60
  91. package/src/backend/_core/vendor/vlucas/phpdotenv/phpstan-baseline.neon +157 -0
  92. package/src/backend/_core/vendor/vlucas/phpdotenv/phpstan.neon.dist +7 -0
  93. package/src/backend/_core/vendor/vlucas/phpdotenv/phpunit.xml.dist +13 -0
  94. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Dotenv.php +267 -267
  95. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/ExceptionInterface.php +12 -12
  96. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidEncodingException.php +12 -12
  97. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidFileException.php +12 -12
  98. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/InvalidPathException.php +12 -12
  99. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Exception/ValidationException.php +12 -12
  100. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/Loader.php +48 -48
  101. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/LoaderInterface.php +20 -20
  102. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Loader/Resolver.php +65 -65
  103. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Entry.php +59 -59
  104. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/EntryParser.php +299 -299
  105. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Lexer.php +58 -58
  106. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Lines.php +127 -127
  107. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Parser.php +53 -53
  108. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/ParserInterface.php +19 -19
  109. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Parser/Value.php +88 -88
  110. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/AdapterInterface.php +15 -15
  111. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ApacheAdapter.php +89 -89
  112. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ArrayAdapter.php +80 -80
  113. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/EnvConstAdapter.php +88 -88
  114. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/GuardedWriter.php +85 -85
  115. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ImmutableWriter.php +110 -110
  116. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/MultiReader.php +48 -48
  117. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/MultiWriter.php +64 -64
  118. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/PutenvAdapter.php +91 -91
  119. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ReaderInterface.php +17 -17
  120. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ReplacingWriter.php +104 -104
  121. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/ServerConstAdapter.php +88 -88
  122. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/Adapter/WriterInterface.php +27 -27
  123. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/AdapterRepository.php +107 -107
  124. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/RepositoryBuilder.php +272 -272
  125. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Repository/RepositoryInterface.php +51 -51
  126. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/File/Paths.php +44 -44
  127. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/File/Reader.php +81 -81
  128. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/FileStore.php +72 -72
  129. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StoreBuilder.php +141 -141
  130. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StoreInterface.php +17 -17
  131. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Store/StringStore.php +37 -37
  132. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Util/Regex.php +112 -112
  133. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Util/Str.php +108 -108
  134. package/src/backend/_core/vendor/vlucas/phpdotenv/src/Validator.php +207 -207
  135. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/DotenvTest.php +387 -0
  136. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Loader/LoaderTest.php +86 -0
  137. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/EntryParserTest.php +234 -0
  138. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/LexerTest.php +40 -0
  139. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/LinesTest.php +53 -0
  140. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Parser/ParserTest.php +98 -0
  141. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/ArrayAdapterTest.php +57 -0
  142. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/EnvConstAdapterTest.php +75 -0
  143. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/PutenvAdapterTest.php +52 -0
  144. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/Adapter/ServerConstAdapterTest.php +75 -0
  145. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Repository/RepositoryTest.php +305 -0
  146. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/Store/StoreTest.php +141 -0
  147. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/Dotenv/ValidatorTest.php +479 -0
  148. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/.env +5 -0
  149. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/assertions.env +18 -0
  150. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/booleans.env +33 -0
  151. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/commented.env +15 -0
  152. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/empty.env +1 -0
  153. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/example.env +1 -0
  154. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/exported.env +7 -0
  155. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/immutable.env +1 -0
  156. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/integers.env +17 -0
  157. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/large.env +2 -0
  158. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multibyte.env +3 -0
  159. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multiline.env +14 -0
  160. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/multiple.env +4 -0
  161. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/mutable.env +1 -0
  162. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/nested.env +15 -0
  163. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/quoted.env +11 -0
  164. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/specialchars.env +8 -0
  165. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/unicodevarnames.env +2 -0
  166. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/utf8-with-bom-encoding.env +3 -0
  167. package/src/backend/_core/vendor/vlucas/phpdotenv/tests/fixtures/env/windows.env +1 -0
  168. package/src/backend/_core/vendor/vlucas/phpdotenv/vendor-bin/phpstan/composer.json +15 -0
  169. package/src/frontend/components/layouts/includes.njk +1 -0
  170. package/src/frontend/data/site.json +43 -43
  171. package/src/frontend/llms.njk +18 -0
  172. package/src/frontend/robots.njk +8 -0
  173. package/src/frontend/scss/modules/frameworks/_bootstrap.scss +63 -63
  174. package/src/frontend/scss/modules/frameworks/_bulma.scss +64 -64
  175. package/src/frontend/scss/modules/frameworks/_foundation.scss +97 -97
  176. package/src/frontend/scss/modules/frameworks/_uikit.scss +78 -78
  177. package/src/frontend/robots.txt +0 -4
@@ -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
+ }